访问修饰符是Java语法中很基础的一部分,但是能正确的使用Java访问修饰符的程序员只在少数。在Java组件开发中,如果能够恰到好处的使用访问修饰符,就能很好的隐藏组件内部数据和不必公布的实现细节,从而把组件API和实现细节隔离;正确的使用访问修饰符开发的Java组件,在组件与组件的调用和依赖过程中,也能很好的解耦程序,以至于整个组件能够持续开发、持续测试、持续更新。
小捌温馨总结:
Java语法提供了四种级别的访问修饰符,作用于域、方法、类、接口,它们的可访问性如下所示:
访问修饰符 | 名称 | 访问性 |
---|---|---|
private | 私有的 | 声明该成员的类才可以访问。注意:顶层类不能被private和protected修饰,内部类可以 |
default/package-private | 包级私有的 | 声明该成员的类同一包下的任何类均可以访问 |
protected | 受保护的 | 声明该成员的类同一包下、子类可以访问 |
public | 共有的 | 任何地方均可访问 |
注意:private和default并不是绝对安全,如果类实现了Serializable,这些被private和defaulte修饰的域同一可能被导出;其次反射也是可以跨过访问修饰符的限制。
Java访问修饰符使用的原则非常简单:在实现Java组件的过程中,保证组件功能一致的同时,尽可能让类、类成员不被外界访问。
这一条规则看似非常简单,但是往往给让程序员产生一种误导,他把类所有的方法和属性都不假思索的设置为private。
这会导致一个什么问题呢?在组件对外公布的时候或者迭代更新的时候,需要不断的颠覆以前的设计,把更多的API对外公出来,但是总的来说这也好过把类中所有成员都用public修饰,这种方式是完全不能接收的,兄弟们。
那问题来了,具体应该怎么搞呢?
其实小捌觉得只需要明白三个点,因为访问修饰符作用于类、方法、属性;所以针对如下三者分析它们应该怎么选择访问修饰符。
对于类来说有如下规则:
对于方法来说有如下规则:
对于属性来说有如下规则:
关于常量域中定义数组对象带来的危险性,小捌做个Demo演示
定义Person对象:
public class Person { private String name; public Person(String name) { this.name = name; } public String getName() { return name; } @Override public String toString() { return "Person{" + "name='" + name + ''' + '}'; } }
定义数组域所属类:
public class PersonDemo { public static final Person[] PERSONS = new Person[] {new Person("李子柒"), new Person("李子捌")}; }
测试代码:
class Test { public static void main(String[] args) { Arrays.stream(PersonDemo.PERSONS).forEach(System.out::println); for (int i = 0; i < PersonDemo.PERSONS.length; i++) { PersonDemo.PERSONS[i] = new Person(PersonDemo.PERSONS[i].getName() + "被修改啦!"); } System.out.println(); Arrays.stream(PersonDemo.PERSONS).forEach(System.out::println); } }
测试结果可以看出,数组内容被修改了,这往往不是我们定义一个常量时所希望看到的。
关于这种方式的处理也很简单,可以将数组域私有化,并且提供一个API来访问数组的拷贝
public class PersonDemo { private static final Person[] PERSONS = new Person[] {new Person("李子柒"), new Person("李子捌")}; public static final Person[] getPersons() { return PERSONS.clone(); } }
此时外部无法直接访问PERSONS数组,访问的只是数组的拷贝,修改的也只是数组的拷贝,无法修改到数组域的内容。
此外也可以使用Collections工具类将其包装为不可变集合,包装成UnmodifiableCollection对象之后,set、add、remove等方法调用会抛出UnsupportedOperationException:
public class PersonDemo { private static final Person[] PERSONS = new Person[] {new Person("李子柒"), new Person("李子捌")}; public static final List<Person> getPersons() { return Collections.unmodifiableList(Arrays.asList(PERSONS)); } }