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文件。