详解Java中的访问控制修饰符(public, protected, default, private)

时间:2022-03-26 05:12:42

Java中的访问控制修饰符已经困惑笔者多时,其中较复杂的情况一直不能理解透彻。今天下定决心,系统、全面地研究Java中的访问控制修饰符的所有方面,并整理成这篇文章,希望有同样疑惑的读者读完后能有所收获。如果文章中出现错误,欢迎评论指出,共同交流~

说在前面:这篇文章只研究Java中访问控制修饰符声明类的变量/方法的情况。

先抛出结论:

* 成员变量/方法的访问权限

*                                        private        default        protected        public

* 自己包自己类                          √               √                  √                √

* 自己包别的类                                           √                  √                √

* 别的包别的类有继承关系②                                            ①               √

* 别的包别的类无继承关系                                                                 √

①:子类可以继承,但是不能访问父类的成员变量/方法(一般来说,可以访问就可以继承)。

②:有继承关系说明访问对象所在的类是父类。

1. 让我们来看一下Java中访问控制修饰符的定义。

Java中,可以使用访问控制符来保护对类、变量、方法和构造方法的访问。

访问的形式有以下四种:

· 某个类的成员变量访问某个类的成员变量

· 某个类的成员变量访问某个类的成员方法

· 某个类的成员方法访问某个类的成员变量

· 某个类的成员方法访问某个类的成员方法

ps:以下代码均以第三种形式为例,其他形式基本一致。

根据访问对象的不同,访问的方式又可划分为两大类:

· 访问对象在同一个类,此时可以通过[成员变量/方法的名字]直接访问。

class A {
int a = 10; void printA() {
System.out.println(a);
}
}

printA()要访问a,因为它们在同一个类,所以可以通过a直接访问。

· 访问对象在不同类(假设访问对象在类B),此时可以通过声明、初始化B的一个对象,通过[对象名.成员变量/方法的名字]进行访问。

ps:这种情况仅限于成员方法访问成员变量/方法。

class A {
void printB() {
B ob = new B();
System.out.println(ob.b);
}
} class B {
int b = 10;
}

A中的printB()要访问B中的b,因为它们不在同一个类,所以可以在printB()中声明、初始化B的一个对象ob,通过ob.b进行访问。

此外,当访问对象为静态变量/方法时,可以通过[访问对象所在类的类名.成员变量/方法的名字]进行访问。

class A {
static int a = 10; int doubleA = A.a * 2; void printB() {
System.out.println(B.b);
}
} class B {
static int b = 10;
}

doubleA要访问a,由于a为静态变量,因此可以通过A.a进行访问。

A中的printB()要访问B中的b,由于b为静态变量,因此可以通过B.b进行访问。

2.结论中提到了包,我们来看一下Java中包的定义和作用。

为了更好地组织类,Java提供了包机制,用于区别类名的命名空间。

包的作用

  • 1 把功能相似或相关的类或接口组织在同一个包中,方便类的查找和使用。
  • 2 如同文件夹一样,包也采用了树形目录的存储方式。同一个包中的类名字是不同的,不同的包中的类的名字是可以相同的,当同时调用两个不同包中相同类名的类时,应该加上包名加以区别。因此,包可以避免名字冲突。
  • 3 包也限定了访问权限,拥有包访问权限的类才能访问某个包中的类。

Java使用包(package)这种机制是为了防止命名冲突,访问控制,提供搜索和定位类(class)、接口、枚举(enumerations)和注释(annotation)等。

关于包的使用方法,请参考Java教程 包(package),在此不详细赘述。

值得注意的是,import关键字引入的是class文件,而非java文件。

3.结论中还提到了继承,我们来看一下Java中继承的定义。

继承是java面向对象编程技术的一块基石,因为它允许创建分等级层次的类。继承可以理解为一个对象从另一个对象获取属性的过程。

关于继承的细节,请参照Java教程 继承,在此不详细赘述。

需要理解的是,子类继承父类的成员变量/方法时,是先访问再继承。因此上面访问权限的规则同样适用于继承。

在同一个包里,如果父类的某个成员变量/方法可以被访问,则该成员变量/方法可以被继承。即如果在子类成员方法中,声明、初始化父类的一个对象后,可以通过[对象名.成员变量/方法a]访问a,则声明、初始化子类的一个对象后,也一定可以通过[对象名.成员变量/方法a]访问a。

class A extends B {
void printB() {
B ob = new B();
System.out.println(ob.b);
A ob2 = new A();
System.out.println(ob2.b);
}
} class B {
int b = 10;
}

A继承B,因此A继承B的成员变量b。由于A在printB()中,声明、初始化B的一个对象ob后,可以通过ob.b访问b,则声明、初始化A的一个对象ob2后,可能通过ob2.b访问b。(可以访问则可以继承)。

然而,在不同包里,子类继承父类时,子类只能访问父类的public型成员变量/方法,却能继承父类的protected和public型成员变量/方法。(请看下面的例子)

值得注意的是,子类继承父类的成员变量/方法,并不意味着这些成员变量/方法存在于子类,因此不能通过[成员变量/方法的名字]直接访问。可以理解为继承而来的成员变量/方法进入了子类的异次元(雾)。

当然,如果继承而来的成员变量/方法被重写,这些成员变量/方法就存在于子类了,此时可以通过[成员变量/方法的名字]直接访问。

此处不讨论多态的情况,请参照Java教程 多态

回到结论,让我们来一层层地验证Java中的访问控制修饰符。

/* Stark.java */

package winter.is.coming;

public class Stark {
private boolean ned;
boolean robb;
protected boolean sansa;
public boolean arya; void howIsNed() {
System.out.println(ned);
}
} class Snow {
void whoseBastard() {
Stark stark = new Stark();
// System.out.println(stark.ned); 不可访问
System.out.println(stark.robb);
}
} /* Greyjoy.java */ import winter.is.coming.Stark; public class Greyjoy extends Stark {
void betray() { Stark stark = new Stark();
// System.out.println(stark.robb); 不可访问
// System.out.println(stark.sansa); 不可访问 Greyjoy greyjoy = new Greyjoy();
// System.out.println(greyjoy.robb); 不可访问
System.out.println(greyjoy.sansa); }
} /* Bolton.java */ import winter.is.coming.Stark; public class Bolton {
void flay() {
Stark stark = new Stark();
System.out.println(stark.arya);
}
}

① 自己包自己类 -- private可访问

Stark中的howIsNed()可以访问Stark中private型的ned。

② 自己包别的类 -- default可访问

Snow中的whoseBastard()可以访问Stark中default型的robb,不可以访问Stark中private型的ned。

③ 别的包别的类有继承关系 -- protected可继承,不可访问

Greyjoy中的betray()可以继承Stark中protected型的sansa,不可以访问Stark中protected型的sansa,也不可以继承和访问Stark中default型的robb。

④ 别的包别的类无继承关系 -- public可访问

Bolton中的flay()可以访问Stark中public型的arya。