Java访问修饰符的正确使用姿势-阿里云开发者社区

开发者社区> 李子捌> 正文

Java访问修饰符的正确使用姿势

简介: Java访问修饰符的正确使用姿势
+关注继续查看
(福利推荐:你还在原价购买阿里云服务器?现在阿里云0.8折限时抢购活动来啦!4核8G企业云服务器仅998元/3年,立即抢购>>>:9i0i.cn/aliyun

福利推荐:阿里云、腾讯云、华为云等大品牌云产品全线2折优惠活动来袭,4核8G云服务器899元/3年,新老用户共享优惠,点击这里立即抢购>>>

1、简介

访问修饰符是Java语法中很基础的一部分,但是能正确的使用Java访问修饰符的程序员只在少数。在Java组件开发中,如果能够恰到好处的使用访问修饰符,就能很好的隐藏组件内部数据和不必公布的实现细节,从而把组件API和实现细节隔离;正确的使用访问修饰符开发的Java组件,在组件与组件的调用和依赖过程中,也能很好的解耦程序,以至于整个组件能够持续开发、持续测试、持续更新。

小捌温馨总结:

  1. 通过限制访问范围达到信息隐藏或封装的效果,保证程序实现细节的安全
  2. 解耦组件,使得组件之间的耦合关系降低,从而能够低成本、低风险(不影响其他组件)的迭代


2、访问修饰符

Java语法提供了四种级别的访问修饰符,作用于域、方法、类、接口,它们的可访问性如下所示:

访问修饰符

名称

访问性

private

私有的

声明该成员的类才可以访问。注意:顶层类不能被private和protected修饰,内部类可以

default/package-private

包级私有的

声明该成员的类同一包下的任何类均可以访问

protected

受保护的

声明该成员的类同一包下、子类可以访问

public

共有的

任何地方均可访问

注意:private和default并不是绝对安全,如果类实现了Serializable,这些被private和defaulte修饰的域同一可能被导出;其次反射也是可以跨过访问修饰符的限制。


3、原则

Java访问修饰符使用的原则非常简单:在实现Java组件的过程中,保证组件功能一致的同时,尽可能让类、类成员不被外界访问。

这一条规则看似非常简单,但是往往给让程序员产生一种误导,他把类所有的方法和属性都不假思索的设置为private。这会导致一个什么问题呢?在组件对外公布的时候或者迭代更新的时候,需要不断的颠覆以前的设计,把更多的API对外公出来,但是总的来说这也好过把类中所有成员都用public修饰,这种方式是完全不能接收的,兄弟们。

那问题来了,具体应该怎么搞呢?

其实小捌觉得只需要明白三个点,因为访问修饰符作用于类、方法、属性;所以针对如下三者分析它们应该怎么选择访问修饰符。

对于类来说有如下规则:

  1. 接口没得选,默认就是public
  2. 顶层普通类,我们可以选择public和default,此时应该着重考虑这个顶层类是否只是在当前包中提供的抽象,如果满足这个条件就可以好不由于的设置为default,但是如果这个顶层类需要被包外其他类直接使用,那就只能设置为public
  3. 非顶层普通类,这种类主要是内部类,内部类有匿名内部类、非匿名内部类;匿名内部类不考虑;非匿名内部类又有静态内部类和非静态内部类,这两者在选择访问修饰符的时候小捌认为没有区别,尽可能的选择私有,因为你都将他设计为内部类,说明这个类抽象就是给外层类提供抽象支持的;所以处于组件设计安全性考虑,尽可能设计为私有,如果在外部需要使用,可以通过外层类提供API访问。

对于方法来说有如下规则:

  1. 接口方法没得选,默认public,根据里氏替换原则,任何使用超类的地方均可以使用子类实例,子类的访问修饰符必须大于等于超类,所以子类也只有public一种选择
  2. 普通类方法,设计类之前要先设计类需要对外公布的API,也就是类需要对外提供那些功能/服务,这个一定要先于写代码之前设计好,之后我们再考虑将这些API设计为default、protected、public,关于具体细节必须使用private修饰

对于属性来说有如下规则:

  1. 如果类是共有的,一定不能将实例域公开;因为一旦公开实例域,等于其他类中可以修改这个实例域,无法保证实例域的安全性
  2. 如果属性能够定义为常量,我们一定要使用static final进行修饰,这样对外暴露的域具有较高安全性。注意不要在常量域中定义数组等可变对象。


关于常量域中定义数组对象带来的危险性,小捌做个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);


? ? }


}

测试结果可以看出,数组内容被修改了,这往往不是我们定义一个常量时所希望看到的。

image

关于这种方式的处理也很简单,可以将数组域私有化,并且提供一个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));

? ? }


}

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
阿里云服务器怎么设置密码?怎么停机?怎么重启服务器?
如果在创建实例时没有设置密码,或者密码丢失,您可以在控制台上重新设置实例的登录密码。本文仅描述如何在 ECS 管理控制台上修改实例登录密码。
9473 0
Java对文件的读、写随机访问,RandomAccessFile类的使用分析
  在网上看了一些关于java中的RandomAccessFile类的介绍,又经过查看Java API和自己编的测试程序,总算是对RandomAccessFile的使用有了一定的了解。自己做了以下比较详细的总结吧。
1028 0
Java访问修饰符的正确使用姿势
Java访问修饰符的正确使用姿势
917 0
阿里云服务器ECS远程登录用户名密码查询方法
阿里云服务器ECS远程连接登录输入用户名和密码,阿里云没有默认密码,如果购买时没设置需要先重置实例密码,Windows用户名是administrator,Linux账号是root,阿小云来详细说下阿里云服务器远程登录连接用户名和密码查询方法
11178 0
阿里云服务器端口号设置
阿里云服务器初级使用者可能面临的问题之一. 使用tomcat或者其他服务器软件设置端口号后,比如 一些不是默认的,?mysql的 3306,?mssql的1433,有时候打不开网页, 原因是没有在ecs安全组去设置这个端口号. 解决: 点击ecs下网络和安全下的安全组 在弹出的安全组中,如果没有就新建安全组,然后点击配置规则 最后如上图点击添加...或快速创建. ? have?fun! ?将编程看作是一门艺术,而不单单是个技术。
10839 0
阿里云服务器如何登录?阿里云服务器的三种登录方法
购买阿里云ECS云服务器后如何登录?场景不同,阿里云优惠总结大概有三种登录方式: 登录到ECS云服务器控制台 在ECS云服务器控制台用户可以更改密码、更换系.
13151 0
阿里云ECS云服务器初始化设置教程方法
阿里云ECS云服务器初始化是指将云服务器系统恢复到最初状态的过程,阿里云的服务器初始化是通过更换系统盘来实现的,是免费的,阿里云百科网分享服务器初始化教程: 服务器初始化教程方法 本文的服务器初始化是指将ECS云服务器系统恢复到最初状态,服务器中的数据也会被清空,所以初始化之前一定要先备份好。
11493 0
Java访问修饰符的正确使用方式(备战2022春招或暑期实习,每天进步一点点,打卡100天,Day6)
Java访问修饰符的正确使用方式(备战2022春招或暑期实习,每天进步一点点,打卡100天,Day6)
7 0
阿里云ECS云服务器初始化设置教程方法
阿里云ECS云服务器初始化是指将云服务器系统恢复到最初状态的过程,阿里云的服务器初始化是通过更换系统盘来实现的,是免费的,阿里云百科网分享服务器初始化教程: 服务器初始化教程方法 本文的服务器初始化是指将ECS云服务器系统恢复到最初状态,服务器中的数据也会被清空,所以初始化之前一定要先备份好。
6882 0
+关注
李子捌
简介: CSDN优秀作者、华为云专家 领域: Java框架、并发编程、分布式、微服务、Redis、HarmonyOS、中间件等技术
240
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
《2021云上架构与运维峰会演讲合集》
立即下载
《零基础CSS入门教程》
立即下载
《零基础HTML入门教程》
立即下载


http://www.vxiaotou.com