第五章
- 有些人认为 super 与 this 引用是类似的概念,实际上,这样比较并不太恰当。这是因为 super 不是一个对象的引用,不能将 super 赋给另一个对象变量,它只是一个指示编译器调用超类方法的特殊关键字。
- 如果子类的构造器没有显式地调用超类的构造器,则将自动地调用超类默认( 没有参数 )的构造器。如果超类没有不带参数的构造器 , 并且在子类的构造器中又没有显式地调用超类的其他构造器 ’ 则 Java 编译器将报告错误。
- 一个对象变量可以指示多种实际类型的现象被称为多态。在运行时能够自动地选择调用哪个方法的现象称为 动态绑定 。
-
Manager boss = new Manager ( . . . ) ;Employee [ ] staff = new Employee [ 3 ] ;staff [ 0 ] = boss ;
在这个例子中,变量staffl与boss引用同一个对象。但编译器将staff [ 0 ] 看成 Employee 对象。这意味着,可以这样调用boss.setBonus(5000);//OK,但不能这样调用staff[0].setBonus(5000);//Error。这是因为staff[0]声明的类型是Employee,而seffionus不是Employee类的方法。
- 如何在对象上应用方法调用 :一. 编译器査看对象的声明类型和方法名。假设调用x.f(param), 且隐式参数 x 声明为 C类的对象。编译器将会一一列举所有C 类中名为 f 的方法和其超类中访问属性为 public 且名为 f 的方法 ( 超类的私有方法不可访问 )。至此,编译器已获得所有可能被调用的候选方法。 二. 接下来,编译器将査看调用方法时提供的参数类型。如果在所有名为 f 的方法中存在一个与提供的参数类型完全匹配,就选择这个方法。这个过程被称为重栽解析。例如,对于调用 x.f ( “ Hello ”)来说,编译器将会挑选f(String),而不是f( int ) 。由于允许类型转换(int 可以转换成 double ,等等 ),所以这个过程可能很复杂。如果编译器没有找到与参数类型匹配的方法,或者发现经过类型转换后有多个方法与之匹配,就会报告一个错误。至此,编译器已获得需要调用的方法名字和参数类型。 三. 如果是private 方法、static方法、final方法或者构造器 ,那么编译器将可以准确地知道应该调用哪个方法,我们将这种调用方式称为静态绑定。与此对应的是,调用的方法依赖于隐式参数的实际类型,并且在运行时实现动态绑定。 四. 当程序运行,并且采用动态绑定调用方法时,虚拟机一定调用与 x 所引用对象的实际类型最合适的那个类的方法。假设x的实际类型是 D,它是 C 类的子类。如果 D 类定义了方法 f(String),就直接调用它,否则,将在 D 类的超类中寻找 f ( String ),以此类推。
- 每次调用方法都要进行搜索,时间开销相当大。因此,虚拟机预先为每个类创建了一个方法表, 其中列出了所有方法的签名和实际调用的方法。这样一来 ,在真正调用方法的时候,虚拟机仅查找这个表就行了。在前面的例子中,虚拟机搜索D类的方法表,以便寻找与调用 f ( Sting ) 相匹配的方法。这个方法既有可能是 D .f ( String ) ,也有可能是X . f ( String ) , 这里的X是D的超类。这里需提醒一点 ,如果调用 super . f ( param ) , 编译器将对隐式参数超类的方法表进行搜索。
- 为了防备参数可能为null 的情况,需要使用 Objects.equals方法。如果两个参数都为null,Objects.equals( a,b)调用将返回true;如果其中一个参数为null,则返回false;否则,如果两个参数都不为null,则调用a.equals ( b )。
- 在子类中定义 equals 方法时 , 首先调用超类的 equals。 如果检测失败 ,对象就不可能相等。 如果超类中的域都相等 ,就需要比较子类中的实例域。
- Java 语言规范要求 equals 方法具有下面的特性 :
- 最好使用null安全的方法 Objects.hashCode。如果其参数为null,这个方法会返回0,否则返回对参数调用 hashCode的结果。
- 数组列表的容量与数组的大小有一个非常重要的区别 。 如果为数组分配 100 个元素的存储空间, 数组就有100 个空位置可以使用 。 而容量为 100 个元素的数组列表只是拥有保存 100 个元素的潜力 ( 实际上, 重新分配空间的话 , 将会超过100 ) , 但是在最初 ,甚至完成初始化构造之后 , 数组列表根本就不含有任何元素 。
- 一旦能够确认数组列表的大小不再发生变化, 就可以调用 trimToSize 方法 。 这个方法将存储区域的大小调整为当前元素数量所需要的存储空间数目。 垃圾回收器将回收多余的存储空间。一旦整理了数组列表的大小, 添加新元素就需要花时间再次移动存储块 , 所以应该在确认不会添加任何元素时 , 再调用 trimToSize 。
- 反射机制的功能极其强大,在下面可以看到,反射机制可以用来 :在运行时分析类的能力。在运行时查看对象,例如,编写一个 toString 方法供所有类使用。实现通用的数组操作代码。利用Method对象,这个对象很像中的函数指针。
原文
https://segmentfault.com/a/1190000020187463