一、访问权限简介
访问权限控制: 指的是本类及本类内部的成员(成员变量、成员方法、内部类)对其他类的可见性,即这些内容是否允许其他类访问。Java 中一共有四种访问权限控制,其权限控制的大小情况是这样的:
public > protected > default(包访问权限) > private
public:Java语言中访问限制最宽的修饰符,一般称之为“公共的”。被其修饰的类、属性以及方法不仅可以跨类访问,而且允许跨包访问。
protected:介于public 和 private 之间的一种访问修饰符,一般称之为“保护访问权限”。被其修饰的属性以及方法只能被类本身的方法及子类访问,即使子类在不同的包中也可以访问。对外包的非子类是不可以访问。
default:即不加任何访问修饰符,通常称为“默认访问权限“或者“包访问权限”。该模式下,只允许在同一个包中进行访问,外包的所有类都不能访问。
private:Java语言中对访问权限限制的最窄的修饰符,一般称之为“私有的”。被其修饰的属性以及方法只能被该类的对象访问,其子类不能访问,更不能允许跨包访问。
注意:
1、所谓的访问,可以分为两种不同方式:通过对象实例访问;直接访问。
比如说,某父类protected权限的成员,子类是可以直接访问的,换一种说法是子类其实继承了父类的除了private成员外的所有成员,包括protected成员,所以与其说是子类访问了父类的protected成员,不如说子类访问了自己的从父类继承来的protected成员。另一方面,如果该子类与父类不在同一个包里,那么通过父类的对象实例是不能访问父类的protected成员的。
2、要区分开 protected 权限、包访问权限,正确使用它们;
当某个成员能被所有的子类继承,但不能被外包的非子类访问,就是用protected;
当某个成员的访问权限只对同包的类开放,包括不能让外包的类继承这个成员,就用包访问权限;
3、使用访问权限控制的原因:
1)使用户不要碰触那些他们不该碰触的部分;
2)类库设计者可以更改类的内部工作的方式,而不会担心这样会对用户产生重大影响;
二、下面用表格来展示四种修饰符的访问权限范围。列所指定的类是否有权限允许访问行的权限控制下的内容:
三、访问权限控制的使用场景
访问权限使用的场景可以总结为下面的五种场景,分别对访问权限的使用有不同的限制:
1、外部类的访问控制
外部类(外部接口) 是相对于内部类(也称为嵌套类)、内部接口而言的。外部类的访问控制只能是这两种:public、default 。
//public 访问权限的外部类,所有类都可以使用这个类
public class OuterClass {
}
//default 访问权限的外部接口,所有类、接口均可以使用此接口
interface OuterInterface{
}
2、类里面的成员的访问控制
类里面的成员分为三类:成员变量、成员方法、成员内部类(内部接口)
类里面的成员的访问控制可以是四种,也就是可以使用所有的访问控制权限
public class OuterClass {
public int aa; //可以被所有的类访问
protected boolean bb; //可以被所有子类以及本包的类使用
void cc() { //default 访问权限,能在本包范围内使用
System.out.println("包访问权限");
}
//private权限的内部类,即这是私有的内部类,只能在本类使用
private class InnerClass{
}
}
注意:
这里的类里面的成员是指类的全局成员,并没有包括局部的成员(局部变量、局部内部类,没有局部内部接口)。或者说,局部成员是没有访问权限控制的,因为局部成员只在其所在的作用域内起作用,不可能被其他类访问到。
public void count(){
//局部成员变量
public int amount;//编译无法通过,不能用public修饰
int money;//编译通过
//局部嵌套接口
class customer{//编译通过
}
}
上面的两种场景几乎可以适应所有的情况,但有一些情况比较特殊,还做了有些额外访问权限的要求。
3、抽象方法的访问权限
普通方法是可以使用四种访问权限的,但抽象方法是有一个限制:不能用private 来修饰,也即抽象方法不能是私有的。否则,子类就无法继承实现抽象方法。
4、接口成员的访问权限
接口由于其自身特殊性,所有成员的访问权限都规定得死死的。下面是接口成员的访问权限:
变量: public static final
抽象方法: public abstract
静态方法: public static,JDK1.8后才支持
内部类、内部接口 : public static
也因为所有的一切都默认强制规定好了,所以我们在用的时候,并不一定需要完整写出所有的修饰符,编译器会帮我们完成的。也就是,可以少写修饰符,但不能写错修饰符。
public interface Interface_Test {
public int aa = 6; //少写了 static final
int bb = 5; //
//嵌套接口,可以不写public static
interface cc{
}
}
5、构造器的访问权限
构造器的访问权限可以是以上四种权限中的任意一种:
A 采用 private:一般是不允许直接构造这个类的对象,再结合工厂方法(static方法),实现单例模式。注意:所有子类都不能继承它。
B 采用包访问控制:比较少用,这个类的对象只能在本包中使用,但是如果这个类有static 成员,那么这个类还是可以在外包使用;(也许可以用于该类的外包单例模式)。
注意:外包的类不能继承这个类。
C 采用 protected:就是为了能让所有子类继承这个类,但是外包的非子类不能访问这个类。
D 采用 public:对于内外包的所有类都是可访问的;
注意: 构造方法有点特殊。因为子类的构造器初始化时,都要调用父类的构造器,所以一旦父类构造器不能被访问,那么子类的构造器调用失败,意味着子类继承父类失败!