第12篇:Java 的访问修饰符(public、protected、private)、封装、继承

简介: 🍀public: 在所有类中都是可见的🍀protected: 仅在自己的包中、自己的子类中可见🍀package-private(没有修饰符): 仅在自己的包中可见🍀private: 仅在自己的类中可见

一、访问控制修饰符

🍀 Access level modifiers determine whether other classes can use a particular field or invoke a particular method.
访问级别修饰符决定其他类是否可以使用特定成员变量或调用特定成员方法方法。

Java 中有 4 个级别的访问控制:
🍀public: 在所有类中都是可见的
🍀protected: 仅在自己的中、自己的子类中可见
🍀package-private(没有修饰符): 仅在自己的中可见
🍀private: 仅在自己的中可见

修饰符 Class Package Subclass World
public 🌼 🌼 🌼 🌼
protected 🌼 🌼 🌼
package-private 🌼 🌼
private 🌼

---
🍀 这四个访问控制修饰符可以修饰类的成员【eg:成员变量、成员方法、嵌套类(Nested Class)】
🍀 只有 publicpackage-private(没有修饰符)可以修饰顶级类(Top-level Class)【顶级类可以有多个,但被 public 修饰的顶级类只能有一个】

public class Person {
   public class Fahter {
       class Son {
           protected class Grandson {
               private class Dog {

               }
           }
       }
   }
}
上面代码中的 Person 就是顶级类,Person 类(顶级类)只能被 public 修饰、或没有访问修饰符

🍀 访问修饰符不能修饰局部类(Local Class)、局部变量

二、封装(Encapsulation)

🌻 类中的成员变量私有化(private
🌻 提供公共的(public)方法【Getter 和 Setter】让外界操纵成员变量

public class Person {
    // 成员变量私有化
    private String name;
    private int age;

    /* 提供公共的(public)方法(Getter 和 Setter)让外界操纵私有的属性 */
    public String getName() {
        return name;
    }

    public int getAge() {
        return age;
    }

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

    public void setAge(int age) {
        this.age = age;
    }
}
public class Main {
    public static void main(String[] args) {
        Person p = new Person();

        /* 通过公共的 Getter/Setter 方法操作成员变量 */

        p.setName("张浩男");
        p.setAge(11);

        System.out.println(p.getName());
        System.out.println(p.getAge());
    }
}

🌻 封装避免外界直接修改属性(而是通过 Setter 方法修改属性),把实现细节封装起来;Setter 可对数据进行验证,保证数据安全合理

三、继承(Inheritance)

(1) 引出继承

学生类:👨‍🎓

public class Student {
    private String name;
    private int age;
    private double score;
    private String gender;
}

员工类:👨‍💻

public class Employee {
    private String name;
    private int age;
    private String gender;
    private double salary;
}
🌻 上面代码中的员工类和学生类都有 name、age 和 gender 属性,员工类和学生类唯一不同的属性是 salary 💴 和 score 💯
🌻 可以把 name、age 和 gender 等三个属性也抽取在一个类( Person)中,把 name、age 和 gender 作为 Person 类的属性。然后学生类和员工类 继承 Person 类。这样学生类和员工类也拥有了 Person 类的 name、age 和 gender 等三个属性,而学生类和员工类中可编写它们独有的属性。

人类【包含 name、age 和 gender 三个属性(这三个属性是员工类和学生类也共有的)】👶

public class Person {
    private String name;
    private int age;
    private String gender;
}

学生类【独有的属性只有 score】👨‍🎓

public class Student {
    private double score;
}

员工类【独有的属性只有 salary】👨‍✈️

public class Student {
    private double salary;
}
💚 Person 类拥有 Student 类和 Employee 类共有的属性(name、age、gender)
💚 可使用 extends 关键字把 Person 类和 Student 类、把 Person 类和 Employee 类产生关联(让 Student 类和 Employee 类中也拥有 Person 类中所拥有的属性)

(2) 继承介绍

💮 The idea of inheritance(继承) is simple but powerful: When you want to create a new class and there is already a class that includes some of the code that you want, you can derive(产生、获得) your new class from the existing class. In doing this, you can reuse the fields and methods of the existing class without having to write them yourself.
💮 继承思想简单,却强大:当你想要创建一个新的类的时候。如果有一个已存在的类,该类中包含你的新类中所需要的代码。你可以从已存在的类中产生你的新类。这样做的话,你就可以重复使用已存在的类中的成员变量和成员方法,而无需自己重新写。

🌷 A subclass(子类) inherits all the members (fields, methods, and nested classes) from its superclass(父类、超类). Constructors are not members, so they are not inherited by subclasses, but the constructor of the superclass can be invoked from the subclass.
🌷 子类继承了父类(或超类)中的所有成员(成员变量、成员方法和嵌套类)。构造方法不是成员,所以构造方法不会被子类继承,但父类(或超类)的构造方法可以被子类调用。


继承可实现代码的复用。当多个类中存在相同的属性和方法时,可以从这些类中抽象出父类,在父类中定义这些相同的属性和方法。然后所有的子类不需要重新定义这些属性和方法,只需要通过 extends 关键字来继承父类即可。
在这里插入图片描述

public class Person {
    private String name;
    private int age;
    private String gender;
}
// Student 类继承 Person 类
public class Student extends Person {
    private double score;
}
// Employee 类继承 Person 类
public class Employee extends Person {
    private double salary;
}
🌺 Person 可叫做超类、父类、基类
🌺 Student 和 Employee 可叫做子类或派生类

(3) 继承细节

🌾 ① 子类继承了父类的成员变量和成员方法后,非私有的(publicprotectedpackage-private)属性和方法可以在子类直接访问。私有的属性和方法子类不能够直接访问,可通过非私有的方法间接访问。
父类 Father:

/**
 * 父类
 */
public class Father {
    public String name = "父类 name";
    protected String age = "父类 age";
    String money = "父类 money";
    // 被 private 修饰的属性无法被子类直接访问到
    private String hobby = "父类 hobby";

    public String getHobbyByPublic() {
        return hobby + "_ByPublic";
    }

    protected String getHobbyByProtected() {
        return hobby + "_ByProtected";
    }

    String getHobbyByPackagePrivate() {
        return hobby + "_ByPackagePrivate";
    }
}

子类 Son 继承 Father:

/**
 * 子类
 */
public class Son extends Father {
    public static void main(String[] args) {
        Son son = new Son();
        son.printFatherClassFields();
    }

    private void printFatherClassFields() {
        // output: 父类 name
        System.out.println(name);
        // output: 父类 age
        System.out.println(age);
        // output: 父类 money
        System.out.println(money);

        // 无法访问到父类被 private 关键字修饰的属性
        // System.out.println(hobby); // ERROR

        // 可通过父类的非私有的方法间接访问到父类被 private 关键字修饰的属性
        // output: 父类 hobby_ByPublic
        System.out.println(getHobbyByPublic());
        // output: 父类 hobby_ByProtected
        System.out.println(getHobbyByProtected());
        // output: 父类 hobby_ByPackagePrivate
        System.out.println(getHobbyByPackagePrivate());
    }
}

在这里插入图片描述

🌾 ② 实例化子类的时候,先调用父类的构造器完成父类的初始化,后调用子类的构造器完成子类的初始化 【“父子”,“父子”,先初始化“父”,后初始化“子”】

/**
 * 父类
 */
public class Father {
    public Father() { // 父类构造器
        System.out.println("1.public Father() 构造器");
    }
}
/**
 * 子类
 */
public class Son extends Father {
    public Son() { // 子类构造器
        System.out.println("2.public Son() 构造器");
    }

    public static void main(String[] args) {
        new Son();
        /*
            output:
                1.public Father() 构造器
                2.public Son() 构造器
         */
    }
}

🌾 ③ 创建子类对象的时候,不管使用子类的哪个构造器,默认情况下都会先去调用父类的无参构造器。如果父类没有提供无参构造器,则子类的构造器中必须用 super 关键字去指定使用父类的哪个构造器。否则,编译无法通过。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

🌾 ④ 子类可通过 super(参数列表) 显示调用父类的某个构造器

🌾 ⑤ super 使用的时候必须放在构造器第一行,且 super 只能在构造器中使用
在这里插入图片描述

🌾 ⑥ super() this() 都只能放在构造器的第一行,所以这两个方法不能在一个构造器中同时存在

🌾 ⑦ Java 中所有的类都有一个共同的父类 Object;Java 中所有的类都是 Object 类的子类;Object类是所有类的基类】

public class Father {
    
    public static void main(String[] args) {
        // output: class java.lang.Object
        // Father 类继承 Object 类
        System.out.println(Father.class.getSuperclass());
    }
    
}

🌾 ⑧ 父类构造器的调用不仅限于直接父类,将会一直往上追溯,直到 Object 类

public class Grandpa {

    public Grandpa() {
        System.out.println("1. class Grandpa - 构造器");
    }

}
public class Father  extends Grandpa{

    public Father() {
        System.out.println("2. class Father  extends Grandpa - 构造器");
    }

}
public class Son extends Father {

    public Son() {
        System.out.println("3. class Son extends Father - 构造器");
    }

}
public class Grandson extends Son {

    public Grandson() {
        System.out.println("4. class Grandson extends Father - 构造器");
    }

    public static void main(String[] args) {
        new Grandson();
    }

}
output:
        1. class Grandpa - 构造器
        2. class Father  extends Grandpa - 构造器
        3. class Son extends Father - 构造器
        4. class Grandson extends Father - 构造器

在这里插入图片描述

🌾 ⑨ Java 中是单继承机制(子类最多只能继承一个直接父类)
在这里插入图片描述

思考:如何让 A 类继承 B 类和 C 类 ?

A extends B;B extends C
public class C {
   public String candy = "candy";
}
public class B extends C {
    public String boy = "boy";
}
public class A extends B {
    public static void main(String[] args) {
        A a = new A();
        a.test();
    }

    private void test() {
        // boy
        System.out.println(boy);
        // candy
        System.out.println(candy);
    }
}

在这里插入图片描述

bye-bye! 下篇文章分析类之间产生继承关系后的内存布局...

若文章中发现错误,请不吝赐教

相关文章
|
7月前
|
Java 数据库连接 数据库
Java 组件详细使用方法与封装实战指南
本指南详解Java核心组件使用与封装技巧,涵盖跨平台开发、面向对象编程、多线程、数据库操作等关键内容,并提供工具类、连接池、异常及响应结果的封装方法。结合Spring框架、MyBatis、Spring Boot等主流技术,助你掌握高质量Java组件设计与开发实践。
239 2
|
10月前
|
人工智能 JSON Java
列表结构与树结构转换分析与工具类封装(java版)
本文介绍了将线性列表转换为树形结构的实现方法及工具类封装。核心思路是先获取所有根节点,将其余节点作为子节点,通过递归构建每个根节点的子节点。关键在于节点需包含 `id`、`parentId` 和 `children` 三个属性。文中提供了两种封装方式:一是基于基类 `BaseTree` 的通用工具类,二是使用函数式接口实现更灵活的方式。推荐使用后者,因其避免了继承限制,更具扩展性。代码示例中使用了 Jackson 库进行 JSON 格式化输出,便于结果展示。最后总结指出,理解原理是进一步优化和封装的基础。
345 0
|
11月前
|
安全 Java 开发者
【JAVA】封装多线程原理
Java 中的多线程封装旨在简化使用、提高安全性和增强可维护性。通过抽象和隐藏底层细节,提供简洁接口。常见封装方式包括基于 Runnable 和 Callable 接口的任务封装,以及线程池的封装。Runnable 适用于无返回值任务,Callable 支持有返回值任务。线程池(如 ExecutorService)则用于管理和复用线程,减少性能开销。示例代码展示了如何实现这些封装,使多线程编程更加高效和安全。
|
11月前
|
安全 网络协议 Java
Java网络编程封装
Java网络编程封装原理旨在隐藏底层通信细节,提供简洁、安全的高层接口。通过简化开发、提高安全性和增强可维护性,封装使开发者能更高效地进行网络应用开发。常见的封装层次包括套接字层(如Socket和ServerSocket类),以及更高层次的HTTP请求封装(如RestTemplate)。示例代码展示了如何使用RestTemplate简化HTTP请求的发送与处理,确保代码清晰易维护。
|
Java
Java 面向对象编程的三大法宝:封装、继承与多态
本文介绍了Java面向对象编程中的三大核心概念:封装、继承和多态。
637 15
|
Java
在Java中,接口之间可以继承吗?
接口继承是一种重要的机制,它允许一个接口从另一个或多个接口继承方法和常量。
1128 60
|
Java
在Java多线程编程中,实现Runnable接口通常优于继承Thread类
【10月更文挑战第20天】在Java多线程编程中,实现Runnable接口通常优于继承Thread类。原因包括:1) Java只支持单继承,实现接口不受此限制;2) Runnable接口便于代码复用和线程池管理;3) 分离任务与线程,提高灵活性。因此,实现Runnable接口是更佳选择。
291 2