abstract类中可以有抽象方法,也可以没有。但是有抽象方法的类一定是抽象类。
抽象类还可以包含具体数据和具体方法。不能被实例化。可以引用子类的实例。
abstract和static,private final不能出现在一起。这里说的是外部类。
Java语言规范要求equals方法具有下面的特性:
1) 自反性:对于任何非空引用x, x.equals(x) 应该返回true。
2) 对称性:对于任何引用x和y,当且仅当y.equals(x)返回true, x.equals(y)也应该返回true。
3) 传递性:对于任何引用x、 y和z,如果x.equals(y)返回true, y.equals(z)返回true,
x.equals(z) 也应该返回true。
4) 一致性:如果x和y引用的对象没有发生变化,反复调用x.equals(y) 应该返回同样的结果。
5)对于任意非空引用x, x.equals(null) 应该返回false。
下面给出编写一个完美的equals方法的建议:
1)显式参数命名为otherObject,稍后需要将它转换成另一个叫做other的变量。
2)检测this与otherObject是否引用同一个对象:
if(this==otherObject)return true;
这条语句只是一个优化。实际上,这是一种经常采用的形式。因为计算这个等式要比一个一个
地比较类中的域所付出的代价小得多。
3)检测otherObject是否为null,如果为null,返回false。这项检测是很必要的。
if(otherObject==null)return false;
4)比较this与otherObject是否属于同一个类。如果equals的语义在每个子类中有所改变,
就使用getClass检测:
if(getClass()!=otherObject.getClass())return false;
如果所有的子类都拥有统一的语义,就使用instanceof检测:
if(xxx instanceof xxx)
5)将otherObject转换为相应的类类型变量:
xxxxx other = (xxxx)otherObject
6)现在开始对所有需要比较的域进行比较了。使用 = = 比较基本类型域,使用equals比较
对象域。如果所有的域都匹配,就返回true;否则返回false。
return xxx==other.xxx &&xxx.equals(other.xxx);
如果在子类中重新定义equals,就要在其中包含调用super.equals(other)。
Equals与hashCode的定义必须一致:如果x.equals(y)返回true,那么x.hashCode( )就必须与
y.hashCode( )具有相同的值。
反射
Date a = new Date();
Class b = a.getClass();
String c = b.getName();
Class d = Class.forName(c);
Class e = xxx.class;
可以用==比较两个类对象的class,可以通过newInstance()来实例化。调用的是默认构造器。
a.getClass().newInstance();Class.forName("A").newInstance();
在java.lang.reflect包中有三个类Field、 Method和Constructor分别用于描述类的域、方法和构造器。这三个类都有一个叫做getName的方法,用来返回项目的名称。 Field类有一个getType方法,用来返回描述域所属类型的Class对象。 Method和Constructor类有能够报告参数类型的方法, Method类还有一个可以报告返回类型的方法。这三个类还有一个叫做getModifiers的方法,它将返回一个整型数值,用不同的位开关描述public和static这样的修饰符使用状况。另外,还可以利用java.lang.reflect包中的Modifier类的静态方法分析getModifiers返回的整型数值。例如,可以使用Modifier类中的isPublic、 isPrivate或isFinal判断方法或构造器是否是public、 private或final。我们需要做的全部工作就是调用Modifier类的相应方法,并对返回的整型数值进行分析,另外,还可以利用Modifier.toString方法将修饰法打印出来。Class类中的getFields、 getMethods和getConstructors方法将分别返回类提供的public域、方法和构造器数组,其中包括超类的公有成员。 Class类的getDeclareFields、 getDeclareMethods和getDeclaredConstructors方法将分别返回类中声明的全部域、方法和构造器,其中包括私有和受保护成员,但不包括超类的成员。
Field、 Method或Constructor,用get()得到值。使用PropertyDescriptor字段要有get和set方法
Class c = a.getClass();
Field[] fields = c.getDeclaredFields();
fields[0].set(a, "qqq");
System.out.println(fields[0].get(a));
for(Field d:fields){
PropertyDescriptor p = new PropertyDescriptor(d.getName(), c);
Method method = p.getWriteMethod();
method.invoke(a, "qqqqq");
}
for(Field f:fields){
PropertyDescriptor p = new PropertyDescriptor(f.getName(), c);
Method m = p.getReadMethod();
String string = (String)m.invoke(a);
System.out.println(string);
}
private要用setAccessible(true)( jvm加上安全管理期及相关策略就不能用了);
可以自动打包成Double等类型,也可以用getDouble();
method.invoke()调用方法用的。
继承设计的技巧
将公共操作和域放在超类,不要使用受保护的域。使用继承实现“ is-a”关系。
除非所有继承的方法都有意义,否则不要使用继承。在覆盖方法时,不要改变预期的行为
使用多态,而非类型信息。不要过多地使用反射。