protected修饰符详解

protected这个修饰符,各大参考书都会这样说:访问权限为类内,包内和子类,因此在父类中定义的方法和成员变量如果为protected修饰的,是可以在不同包中的子类进行访问的,示例代码如下:

1 package cn.tedu.object.a;
2 
3 public class A {
4     
5     protected void m(){
6         System.out.println("A m~~~");
7     }
8 
9 }
 1 package cn.tedu.object.b;
 2 
 3 import cn.tedu.object.a.A;
 4 
 5 public class B extends A {
 6     void callM() {
 7         m();
 8         super.m();
 9         B b = new B();
10         b.m();
11     }
12 }

如上代码所示,class B继承了class A,但是两个类位于a、b两个不同的包中,此时class B可以直接访问class A中的protected修饰的方法。一共有三种方法访问:

第一种为第7行,直接调用;

第二种为第8行,加super关键字;其实前两种是等效的。

第三种为第9、10行,实例化一个对象进行调用。

那么问题来了,如果说在b包中新建一个class C,并且有如下代码,会报错吗?

 1 package cn.tedu.object.b;
 2 
 3 import cn.tedu.object.a.A;
 4 
 5 public class C extends A {
 6 
 7     void callM() {
 8         m();
 9         super.m();
10         B b = new B();
11         b.m();     //The method m() from type A is not visible
12     }
13 
14 }

会报错!那为什么会报错呢?

因为b.m()这种调用属于子类对象调用,并不是子类调用。

这个是非常需要引起注意的。因此,当创建子类对象调用父类的protected成员变量时,必须要注意:子类对象和子类是对应的!

通过这个例子其实也可以看出来,当一个包外子类继承保护成员时,该成员在这个子类内实际上变为私有。

事实上,除了以上三种调用父类protected成员变量的方法外,其余都是会报错的!

包括用超类实例去访问。代码如下:

 1 package cn.tedu.object.b;
 2 
 3 import cn.tedu.object.a.A;
 4 
 5 public class B extends A {
 6     
 7     void callM() {
 8         A a=new A();
 9         a.m();  //The method m() from type A is not visible
10     }
11 
12 }

同样的,会报错!因此,可以得到结论:

包外子类有权访问超类成员,它指子类继承该成员,然而,这并不意味着包外子类能够使用超类实例的引用访问该成员。

 

 

 
posted @ 2018-04-09 20:38  豪杰杰杰  阅读(8172)  评论(1编辑  收藏  举报