第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! 下篇文章分析类之间产生继承关系后的内存布局...

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

相关文章
|
4天前
|
Java
Java 面向对象编程的三大法宝:封装、继承与多态
本文介绍了Java面向对象编程中的三大核心概念:封装、继承和多态。
47 15
|
2月前
|
Java
在Java中,接口之间可以继承吗?
接口继承是一种重要的机制,它允许一个接口从另一个或多个接口继承方法和常量。
161 1
|
3月前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
53 3
|
3月前
|
Java
在Java多线程编程中,实现Runnable接口通常优于继承Thread类
【10月更文挑战第20天】在Java多线程编程中,实现Runnable接口通常优于继承Thread类。原因包括:1) Java只支持单继承,实现接口不受此限制;2) Runnable接口便于代码复用和线程池管理;3) 分离任务与线程,提高灵活性。因此,实现Runnable接口是更佳选择。
76 2
|
3月前
|
Java
Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口
【10月更文挑战第20天】《JAVA多线程深度解析:线程的创建之路》介绍了Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口。文章详细讲解了每种方式的实现方法、优缺点及适用场景,帮助读者更好地理解和掌握多线程编程技术,为复杂任务的高效处理奠定基础。
55 2
|
3月前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
58 1
|
3月前
|
Java 测试技术 编译器
Java零基础-继承详解!
【10月更文挑战第6天】Java零基础教学篇,手把手实践教学!
45 0
Java中访问修饰符public、private、protect、default范围
Java中访问修饰符public、private、protect、default范围
|
18天前
|
监控 Java
java异步判断线程池所有任务是否执行完
通过上述步骤,您可以在Java中实现异步判断线程池所有任务是否执行完毕。这种方法使用了 `CompletionService`来监控任务的完成情况,并通过一个独立线程异步检查所有任务的执行状态。这种设计不仅简洁高效,还能确保在大量任务处理时程序的稳定性和可维护性。希望本文能为您的开发工作提供实用的指导和帮助。
80 17
|
28天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者

热门文章

最新文章