1. protected 访问控制符能被用于方法和成员变量。 2. 声明为protected的方法和成员变量能被同一个包里的所有类所访问,就像默认修饰符package一样。 3. 能被该类的子类所访问,子类可以和父类不在一个包中。 这样,当你想让一个类中的某个方法或成员变量在包中都可见,而且其子类也能访问(子类有可能和父类不在同一个包中)但又不想让所有类都可以访问该类时,就可以用protected修饰符。 可访问性: public > protected > package >private 注意: 4. But a subclass in another package can access the protected members in the super-class via only the references of subclass or its subclasses. A subclass in the same package doesn’t have this restriction. This ensures that classes from other packages are accessing only the members that are part of their inheritance hierarchy.
下面的例子可以说明上述几点:(特别注意第4点) 我们将创建一个父类Bird.java,放在birdpack包中,父类中有一个protected int的成员变量nFeathers; 再分别创建4个Bird类的子类Duck1.java,Duck2.java,Duck3.java,Swan.java,放在duckpack包中,通过在每个子类中调用nFeathers的不同方法说明上述几点
下面的程序并不用于运行,因为访问控制在编译期间就要确定,我们只需编译下述文件,看是否能通过。在编译下述文件前,先想想能不能编译通过?
//Bird.java------------------------------ package birdpack;
public class Bird{ protected int nFeathers; }
//Duck1.java----------------------------- package duckpack;
import birdpack.Bird;
public class Duck1 extends Bird{ public void setn(int duck1n){ //在子类中直接访问父类中的protected变量 nFeathers=duck1n; } } //Duck2.java------------------------------ package duckpack;
import birdpack.Bird;
public class Duck2 extends Bird{
public void construct(int newduck2){ Duck2 d2 = new Duck2(); //在子类中通过子类的对象访问父类中的protected变量 d2.nFeathers=newduck2; } }
//Duck3.java------------------------------ package duckpack;
import birdpack.Bird;
public class Duck3 extends Bird{
public void construct(int newduck3){ Bird b = new Bird(); //子类中用父类对象反而不能访问父类中的protected变量 b.nFeathers=newduck3; } }
//Swan.java-------------------------------- package duckpack;
import birdpack.Bird;
public class Swan extends Bird{
public void construct(int swan){ Duck1 d1 = new Duck1(); //子类中用另外一个子类的对象也不能访问父类中的protected变量 d1.nFeathers=swan; } }
编译上述几个文件,后2个不能通过。编译器提示: " nFeathers has protected access in birdpack.Bird"。
//Bird.java------------------------------ package birdpack;
public class Bird{ protected int nFeathers; }
//Duck1.java----------------------------- package duckpack;
import birdpack.Bird;
public class Duck1 extends Bird{ public void setn(int duck1n){ //在子类中直接访问父类中的protected变量 nFeathers=duck1n; } } //Duck2.java------------------------------ package duckpack;
import birdpack.Bird;
public class Duck2 extends Bird{
public void construct(int newduck2){ Duck2 d2 = new Duck2(); //在子类中通过子类的对象访问父类中的protected变量 d2.nFeathers=newduck2; } }
//Duck3.java------------------------------ package duckpack;
import birdpack.Bird;
public class Duck3 extends Bird{
public void construct(int newduck3){ Bird b = new Bird(); //子类中用父类对象反而不能访问父类中的protected变量 b.nFeathers=newduck3; } }
//Swan.java-------------------------------- package duckpack;
import birdpack.Bird;
public class Swan extends Bird{
public void construct(int swan){ Duck1 d1 = new Duck1(); //子类中用另外一个子类的对象也不能访问父类中的protected变量 d1.nFeathers=swan; } }
编译上述几个文件,后2个不能通过。编译器提示: " nFeathers has protected access in birdpack.Bird"。
第4点说明,就算在子类中,也只能通过子类(或子类的子类)的引用来访问父类中的protected方法和成员变量。 在Duck3和Swan两个子类中,直接通过父类和另一个子类来访问父类中的protected方法和成员变量就不行。
附:编译技巧 当编译含有包声明的源文件时,使用 -d 选项会带来很多方便,它能保证编译后的class文件存放在正确的目录路径下。 我们可以把上述5个文件都放在protectedtest目录下,然后用 -d 选项去编译: javac -d . Bird.java javac -d . Duck1.java ..... 其中.表示以当前目录创建包结构的目录层次。编译成功后,会在protectedtest下生成2个目录:birdpack和duckpack,里面是生成的class文件。
|