深入理解 Java 修饰符与封装:访问权限、行为控制与数据隐藏

本文涉及的产品
可观测可视化 Grafana 版,10个用户账号 1个月
函数计算FC,每月15万CU 3个月
应用实时监控服务-应用监控,每月50GB免费额度
简介: ava 修饰符 用于控制类、属性、方法和构造函数的访问权限和行为。它们可以分为两组:访问修饰符:public: 意味着代码对所有类可访问。private: 意味着代码只能在声明的类内部访问。default: 意味着代码只能在同一包中访问。protected: 意味着代码在同一包和子类中可访问。非访问修饰符:final: 意味着类不能被继承,属性和方法不能被重写。static: 意味着属性和方法属于类,而不属于对象。abstract: 意味着类不能用于创建对象,方法没有主体,必须由子类提供。transient: 意味着在序列化包含它们的对象时,属性和方法将被跳过。sync

Java 修饰符

Java 修饰符 用于控制类、属性、方法和构造函数的访问权限和行为。它们可以分为两组:

  1. 访问修饰符:
  • public: 意味着代码对所有类可访问。
  • private: 意味着代码只能在声明的类内部访问。
  • default: 意味着代码只能在同一包中访问。
  • protected: 意味着代码在同一包和子类中可访问。
  1. 非访问修饰符:
  • final: 意味着类不能被继承,属性和方法不能被重写。
  • static: 意味着属性和方法属于类,而不属于对象。
  • abstract: 意味着类不能用于创建对象,方法没有主体,必须由子类提供。
  • transient: 意味着在序列化包含它们的对象时,属性和方法将被跳过。
  • synchronized: 意味着方法一次只能由一个线程访问。
  • volatile: 意味着属性的值不会在本地线程缓存,总是从“主内存”读取。

示例:

访问修饰符:

// public类可以被任何类访问
public class Main {
   

  // private属性只能在 Main 类内部访问
  private int x = 10;

  // default方法只能在同一个包中访问
  void myMethod() {
   
    System.out.println("This is a default method.");
  }

  // protected方法可以在同一个包和子类中访问
  protected void myProtectedMethod() {
   
    System.out.println("This is a protected method.");
  }

  public static void main(String[] args) {
   
    Main myObj = new Main();
    System.out.println(myObj.x); // 错误,无法访问私有属性
    myObj.myMethod(); // 可以访问 default 方法
    myObj.myProtectedMethod(); // 可以访问 protected 方法
  }
}

非访问修饰符:

public class Main {
   

  // final属性不能被修改
  final int y = 20;

  // static方法可以在不创建对象的情况下访问
  static void myStaticMethod() {
   
    System.out.println("This is a static method.");
  }

  // abstract方法没有主体,必须由子类提供
  abstract void myAbstractMethod();

  public static void main(String[] args) {
   
    // y = 30; // 错误,无法修改 final 属性
    myStaticMethod(); // 可以访问 static 方法
    // myAbstractMethod(); // 错误,无法直接调用抽象方法
  }
}

// 子类继承 Main 类并提供抽象方法的主体
class Child extends Main {
   

  @Override
  void myAbstractMethod() {
   
    System.out.println("This is the implementation of myAbstractMethod.");
  }
}

一些额外的说明:

  • 可以使用多个修饰符来修饰类、属性、方法和构造函数。例如,public final 表示类是公开的并且不能被继承。
  • 修饰符是 Java 语言的重要组成部分,它们可以帮助您控制代码的访问权限和行为。

以下是一些有关 Java 修饰符的其他重要信息:

  • default 修饰符 也可以称为 package-private 修饰符。
  • protected 修饰符允许子类访问父类中的成员,但不允许外部类访问。
  • static 变量和方法也称为 类变量 和 类方法。
  • abstract 类不能被实例化,只能被继承。
  • transient 变量不会被序列化。
  • synchronized 方法可以防止多个线程同时执行。
  • volatile 变量保证每次读取都将获取最新的值。

Java 封装

封装 的含义是确保敏感数据对用户隐藏。为了实现这一点,你需要:

  1. 将类变量/属性声明为私有
  2. 提供公共的 get 和 set 方法以访问和更新私有变量的值

Get 和 Set 方法

私有变量只能在同一类中访问(外部类无法访问)。但是,如果我们提供了公共的 get 和 set 方法,就可以访问它们。

get 方法 返回变量值,而 set 方法 设置值。

两者的语法是它们都以 get 或 set 开头,后面跟着变量的名称,首字母大写:

public class Person {
   
  private String name; // private = 受限制的访问

  // Getter
  public String getName() {
   
    return name;
  }

  // Setter
  public void setName(String newName) {
   
    this.name = newName;
  }
}

示例解释

  • get 方法返回变量 name 的值。
  • set 方法接受一个参数(newName)并将其分配给 name 变量。关键字 this 用于引用当前对象。

然而,由于 name 变量被声明为私有,我们无法从此类外部访问它:

public class Main {
   
  public static void main(String[] args) {
   
    Person myObj = new Person();
    myObj.name = "John"; // 错误
    System.out.println(myObj.name); // 错误
  }
}

如果变量被声明为 public,我们期望以下输出:

John

然而,由于我们尝试访问一个私有变量,我们得到一个错误:

MyClass.java:4: error: name has private access in Person
  myObj.name = "John";
    ^
MyClass.java

:5: error: name has private access in Person
    System.out.println(myObj.name);
        ^
2 errors

相反,我们使用 getName() 和 setName() 方法来访问和更新变量:

public class Main {
   
  public static void main(String[] args) {
   
    Person myObj = new Person();
    myObj.setName("John"); // 将 name 变量的值设置为 "John"
    System.out.println(myObj.getName());
  }
}

// 输出 "John"

为什么封装?

  • 更好地控制类的属性和方法
  • 类属性可以被设置为只读(如果仅使用 get 方法)或只写(如果仅使用 set 方法)
  • 灵活性:程序员可以更改代码的一部分而不影响其他部分
  • 数据的安全性增加

封装的优点:

  • 提高安全性:隐藏内部实现细节,防止意外修改或访问敏感数据。
  • 提高代码的模块化:将代码组织成独立的模块,每个模块只暴露必要的接口。
  • 提高代码的可重用性:封装的模块可以被其他代码重用,降低代码的重复性。
  • 提高代码的维护性:易于理解和维护代码,降低代码的复杂度。

一些额外的说明:

  • 可以使用多个修饰符来修饰 get 和 set 方法,例如 public、protected 和 private。
  • 可以使用 final 修饰符来声明常量,常量不能被修改。
  • 可以使用 static 修饰符来声明静态方法和变量,静态方法和变量属于类,而不是对象。

最后

为了方便其他设备和平台的小伙伴观看往期文章:

微信公众号搜索:Let us Coding,关注后即可获取最新文章推送

看完如果觉得有帮助,欢迎 点赞、收藏、关注

相关文章
|
20天前
|
前端开发 JavaScript Java
java常用数据判空、比较和类型转换
本文介绍了Java开发中常见的数据处理技巧,包括数据判空、数据比较和类型转换。详细讲解了字符串、Integer、对象、List、Map、Set及数组的判空方法,推荐使用工具类如StringUtils、Objects等。同时,讨论了基本数据类型与引用数据类型的比较方法,以及自动类型转换和强制类型转换的规则。最后,提供了数值类型与字符串互相转换的具体示例。
|
27天前
|
JSON Java 程序员
Java|如何用一个统一结构接收成员名称不固定的数据
本文介绍了一种 Java 中如何用一个统一结构接收成员名称不固定的数据的方法。
26 3
|
1月前
|
Java 程序员 容器
Java中的变量和常量:数据的‘小盒子’和‘铁盒子’有啥不一样?
在Java中,变量是一个可以随时改变的数据容器,类似于一个可以反复打开的小盒子。定义变量时需指定数据类型和名称。例如:`int age = 25;` 表示定义一个整数类型的变量 `age`,初始值为25。 常量则是不可改变的数据容器,类似于一个锁死的铁盒子,定义时使用 `final` 关键字。例如:`final int MAX_SPEED = 120;` 表示定义一个名为 `MAX_SPEED` 的常量,值为120,且不能修改。 变量和常量的主要区别在于变量的数据可以随时修改,而常量的数据一旦确定就不能改变。常量主要用于防止意外修改、提高代码可读性和便于维护。
|
1月前
|
存储 缓存 安全
在 Java 编程中,创建临时文件用于存储临时数据或进行临时操作非常常见
在 Java 编程中,创建临时文件用于存储临时数据或进行临时操作非常常见。本文介绍了使用 `File.createTempFile` 方法和自定义创建临时文件的两种方式,详细探讨了它们的使用场景和注意事项,包括数据缓存、文件上传下载和日志记录等。强调了清理临时文件、确保文件名唯一性和合理设置文件权限的重要性。
96 2
|
1月前
|
Java
Java 8 引入的 Streams 功能强大,提供了一种简洁高效的处理数据集合的方式
Java 8 引入的 Streams 功能强大,提供了一种简洁高效的处理数据集合的方式。本文介绍了 Streams 的基本概念和使用方法,包括创建 Streams、中间操作和终端操作,并通过多个案例详细解析了过滤、映射、归并、排序、分组和并行处理等操作,帮助读者更好地理解和掌握这一重要特性。
33 2
|
1月前
|
存储 分布式计算 Java
存算分离与计算向数据移动:深度解析与Java实现
【11月更文挑战第10天】随着大数据时代的到来,数据量的激增给传统的数据处理架构带来了巨大的挑战。传统的“存算一体”架构,即计算资源与存储资源紧密耦合,在处理海量数据时逐渐显露出其局限性。为了应对这些挑战,存算分离(Disaggregated Storage and Compute Architecture)和计算向数据移动(Compute Moves to Data)两种架构应运而生,成为大数据处理领域的热门技术。
65 2
|
1月前
|
SQL Java 数据库连接
在Java应用中,数据库访问常成为性能瓶颈。连接池技术通过预建立并复用数据库连接,有效减少连接开销,提升访问效率
在Java应用中,数据库访问常成为性能瓶颈。连接池技术通过预建立并复用数据库连接,有效减少连接开销,提升访问效率。本文介绍了连接池的工作原理、优势及实现方法,并提供了HikariCP的示例代码。
55 3
|
前端开发 Java 数据安全/隐私保护
Java教程 (Java 修饰符)
Java教程 (Java 修饰符)
|
机器学习/深度学习 Java 数据安全/隐私保护
JAVA 入坑教程 | 章节四 修饰符
Java语言提供了很多修饰符,主要分为以下两类: 访问修饰符 非访问修饰符 例子 public class className { // ... } private boolean myFlag; static final double weeks = 9.
958 0
|
1天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者