先看两段小程序:
第一段定义了一个类BaseClass, 具有包级访问权限
第二段在同一个包内定义了另一个类,其中包含了一个List<BaseClass>
你可能会认为这个程序不会有任何问题,虽然在getFirstName()方法中的写法有点奇怪。但是第二段根本就无法通过编译,原因是list中的元素无法得到getName()方法的访问权限。但是我们的两段程序都是在同一个包里的呀,而且第一段程序也有着包级访问权限,那问题出在哪里呢?
很明显,List<? extends BaseClass> list = getItems();这句奇怪的语句导致了问题。通配符似乎扩大了访问范围。现在我们来假设一下如果该句是合法的会出现什么情况?那将意味着我们可以在包外访问一个包级的方法!虽然BaseClass是包级的,但是我们仍然可以通过继承在包外得到它的一个子类实例!因此,在这里java禁止我们调用包级的方法是出于安全的考虑,也是完全合情合理的。我们想要访问到这个方法,必须得经过显式转换。但最好的方法是,不要尝试去写这种奇怪的代码。
第一段定义了一个类BaseClass, 具有包级访问权限
package aboutClass;
class BaseClass {
private String name = "";
BaseClass(String name){
this.name = name;
}
String getName(){
return name;
}
}
第二段在同一个包内定义了另一个类,其中包含了一个List<BaseClass>
package aboutClass;
import java.util.ArrayList;
import java.util.List;
class TestBaseClass {
private List<BaseClass> items = new ArrayList<BaseClass>();
void addItem(BaseClass b){
items.add(b);
}
List<BaseClass> getItems(){
return items;
}
String getFirstName(){
List<? extends BaseClass> list = getItems();
if(list.size() >= 1)
return list.get(0).getName();
else return "";
}
}
你可能会认为这个程序不会有任何问题,虽然在getFirstName()方法中的写法有点奇怪。但是第二段根本就无法通过编译,原因是list中的元素无法得到getName()方法的访问权限。但是我们的两段程序都是在同一个包里的呀,而且第一段程序也有着包级访问权限,那问题出在哪里呢?
很明显,List<? extends BaseClass> list = getItems();这句奇怪的语句导致了问题。通配符似乎扩大了访问范围。现在我们来假设一下如果该句是合法的会出现什么情况?那将意味着我们可以在包外访问一个包级的方法!虽然BaseClass是包级的,但是我们仍然可以通过继承在包外得到它的一个子类实例!因此,在这里java禁止我们调用包级的方法是出于安全的考虑,也是完全合情合理的。我们想要访问到这个方法,必须得经过显式转换。但最好的方法是,不要尝试去写这种奇怪的代码。