java笔记(this,super,override,instanceof,static)

打印 上一主题 下一主题

主题 969|帖子 969|积分 2907

super关键字的一些注意事项


  • 子类在执行构造方法时,如果显式使用super()显式调用父类构造方法,则该调用必须放代码块在第一行
  • super必须出现在子类的方法或者构造方法中
  • 使用this()显示调用构造方法,则该调用必须放在代码块第一行
  • 由于第一条和第三条限制,super和this不能同时调用构造方法
  1. class P1{
  2.   public P1(){
  3.     //this(); //不可以递归调用构造方法
  4.   }
  5.   public P1(String str){
  6.     this();//可以调用重载的构造方法,该调用必须放在第一行
  7.   }
  8. }
  9. class S1 extends P1{
  10.   public S1(){
  11.     super();//显式调用父类构造方法,该调用必须放在第一行
  12.   }
  13.   public S1(String str){
  14.     this();//无论调用自己的构造方法还是父类的构造方法,都要求在第一行,因此只能调用其中一个
  15.     System.out.println("constructor with parameter");
  16.   }
  17. }
复制代码
super 和 this的不同


  • 代表的对象不同:


  • this:本身调用者这个对象
  • super:代表父类对象的引用

  • 使用的前提不一样:


  • this没有发生继承也可以使用
  • super只有在发生继承时才可以使用,用来表示父类

  • 构造方法不一样:


  • this() 调用本类的构造方法
  • super() 调用父类的构造方法
方法重写

方法重写是多态的一个前提,需要在有继承关系的前提下,由子类重写父类的方法,并且存在几点要求:

  • 方法名必须相同
  • 参数列表必须相同
  • 重写的方法修饰符范围可以扩大: public>protected>default>private
  • 抛出的异常范围可以缩小,但不能扩大:ClassNotFoundException(小)  ==> Exception(大)
  • 返回值类型可以缩小
  1. class P2{
  2.         protected S1 run() throws Exception{
  3.                 System.out.println("Father running!");
  4.                 return new S1();
  5.         }
  6. }
  7. class S2 extends P2{
  8. //修饰符范围可以扩大,抛出异常范围可以缩小
  9. //子类方法与父类方法方法名、参数列表必须完全一致
  10.         public P1 run() throws ClassNotFoundException{
  11.                 System.out.println("Son running!");
  12.                 return new P1();//返回值范围可以缩小
  13.         }
  14. }
复制代码
多态注意事项


  • 多态是方法的多态,属性没有多态
  • 如果没有继承关系的类强制转换,会导致ClassCastException类型转换异常
  • 多态的存在条件:继承关系、方法重写、父类引用指向子类对象, Father f1 = new Son();
    因此,有一些方法是不能实现多态的:
  • static 方法不能实现多态,因为static方法是属于类的,不属于实例,当子类定义同名static方法时,会隐藏父类的同名方法。
  • final 修饰的类和方法。因为final修饰的类为最终类,不能被继承,断子绝孙了,不能满足继承关系,自然也就无法实现多态;final修饰的方法不能被重写,也不能实现多态
  • private 修饰的方法。这类方法子类无法访问,无法产生重写,因此无法产生多态。
  • 在实验中,笔者注意到父类中 protected 修饰的方法在重写后,只有在父类包路径中可以实现多态,当实例范围超出父类的包之后,该方法即只能被子类实例调用,父类引用指向的子类实例此时不能再调用该方法。
  1. public class Test{
  2.         public static void mail(String args[]) throws Exception{
  3.         S2 stu = new S2();
  4.         stu.run();//无论在哪里都可以正常调用
  5.         P2 per = stu;//父类引用指向子类实例,此时发生多态
  6.         per.run();//只有当Test与父类P2在同一个包中时,父类引用per才可以访问重写的run()方法。
  7.         }
  8. }
复制代码
instanceof

instanceof的作用是判断实例的类型

  • 当类型由低向高转化时,如子类型转化为夫类型,直接使用父类引用指向子类实例即可。
  • 当类型由高向低转化时,需要进行强制类型转换。这是因为子类实例一定是父类的实例,包含父类的所有域,而父类实例不一定时子类型的实例,包含的域可能更小。
  • 子类转换为父类时,可能丢失自己的方法。
  1. class P3{
  2. }
  3. class S3 extends P3{
  4.         static void talk(){
  5.                 System.out.println("Son talk sth");
  6.         }
  7.         public class Test{
  8.                 public static void main(String[] args){
  9.                         S3 son = new S3();
  10.                         //此的S3的类方法talk()
  11.                         son.talk();
  12.                         P3 per = son;//直接使用父类应用指向子类对象,即完成了类型由低到高的转换
  13.                         //转换成更高等级的父类后,子类中的talk不能被访问了
  14.                         //per.talk()//不能访问
  15.                         S3 son1 = (S3)per;//由父类型向子类型转换则需要强制类型转换
  16.                 }
  17.         }
  18. }
复制代码
static

static在类加载时的一些作用
  1. public class P4{
  2.         {
  3.                 System.out.println("Anonymous code block!");
  4.         }
  5.         static {
  6.                 System.out.println("Static anonymous code block!");
  7.         }
  8.         public P4(){
  9.                 System.out.println("No parameter constructor!");
  10.         }
  11.         public static void main(String[] args){
  12.         P4 per1 = new P4();
  13.         System.out.println("==============");
  14.         P4 per2 = new P4();
  15.         }
  16. }
复制代码
执行结果如下:

可见静态代码块在类加载时首先执行,且在类生命周期中只被执行一次,而匿名代码块则在构造方法之前执行,且每次实例化都会执行。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

天空闲话

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表