首页 > 编程语言 >06-JavaSE:面向对象编程

06-JavaSE:面向对象编程

时间:2023-01-25 22:55:47浏览次数:45  
标签:06 Student void class 面向对象编程 JavaSE 方法 public out

  • 面向过程的思维模式

面向过程的思维模式是简单的线性思维,思考问题首先陷入第一步做什么、第二步做什么的细节中。这种思维模式适合处理简单的事情,比如:上厕所。

如果面对复杂的事情,这种思维模式会陷入令人发疯的状态!比如:如何造神舟十号!

  • 面向对象的思维模式

面向对象的思维模式说白了就是分类思维模式。思考问题首先会解决问题需要哪些分类,然后对这些分类进行单独思考。最后,才对某个分类下的细节进行面向过程的思索。

这样就可以形成很好的协作分工。比如:设计师分了10个类,然后将10个类交给了10个人分别进行详细设计和编码!

显然,面向对象适合处理复杂的问题,适合处理需要多人协作的问题!

如果一个问题需要多人协作一起解决,那么你一定要用面向对象的方式来思考!

对于描述复杂的事物,为了从宏观上把握、从整体上合理分析,我们需要使用面向对象的思路来分析整个系统。但是,具体到微观操作,仍然需要面向过程的思路去处理。

一、OOP详解

1.1 何为面向对象

java的编程语言是面向对象的,采用这种语言进行编程称为面向对象编程(Object-Oriented Programming,简称OOP)

本质:以类的方式组织代码,以对象的组织(封装)数据。

  • 抽象(abstract)
忽略一个主题中与当前目标无关的那些方面,以便更充分地注意与当前目标有关的方面。抽象并不打算了解全部问题,而只是选择其中的一部分,暂时不用关注细节。
例如:要设计一个学生成绩管理系统,那么对于学生,只关心他的班级、学号、成绩等,而不用去关心他的身高、体重这些信息。 抽象是什么?就是将多个物体共同点归纳出来,就是抽出像的部分!
  • 封装(Encapsulation)
封装是面向对象的特征之一,是对象和类概念的主要特性。封装是把过程和数据包围起来,对数据的访问只能通过指定的方式。
在定义一个对象的特性的时候,有必要决定这些特性的可见性,即哪些特性对外部是可见的,哪些特性用于表示内部状态。
通常,应禁止直接访问一个对象中数据的实际表示,而应通过操作接口来访问,这称为信息隐藏。
信息隐藏是用户对封装性的认识,封装则为信息隐藏提供支持。
封装保证了模块具有较好的独立性,使得程序维护修改较为容易。对应用程序的修改仅限于类的内部,
因而可以将应用程序修改带来的影响减少到最低限度。
  • 继承(inheritance
继承是一种联结类的层次模型,并且允许和支持类的重用,它提供了一种明确表述共性的方法。
新类继承了原始类后,新类就继承了原始类的特性,新类称为原始类的派生类(子类),而原始类称为新类的基类(父类)。
派生类(子类)可以从它的基类(父类)那里继承方法和实例变量,并且派生类(子类)中可以修改或增加新的方法使之更适合特殊的需要继承性很好的解决了软件的可重用性问题。比如说,所有的Windows应用程序都有一个窗口,它们可以看作都是从一个窗口类派生出来的。但是有的应用程序用于文字处理,有的应用程序用于绘图,这是由于派生出了不同的子类,各个子类添加了不同的特性。
  • 多态(polymorphism)
多态性是指允许不同类的对象对同一消息作出响应。
多态性语言具有灵活、抽象、行为共享、代码共享的优势,很好的解决了应用程序函数同名问题。

从认识论角度考虑,先有对象后有类,对象是具体的事物。类是抽象的,是对对象的抽象

从代码运行的角度考虑,先有类后有对象。类是对象的模板

1.2 类与对象的关系

类是一种抽象的数据类型,它是对某一类事物整体描述/定义,但是并不能代表某一个具体的事物。

例如:生活中我们说的词语,动物,植物,手机,电脑等,这些也都是抽象的概念而不是一个具体的东西。

张三是一个具体的实例,张三家的旺财就是狗的一个具体的实例,能够体现出特点,展现功能的就是具体的实例,而不是一个抽象的概念。

Student s = new Student(1L, "Tom", 20);
s.study();

Car c = new Car(1, "BWM", 500000);
c.run();

对象s就是Student类的一个实例,对象c就是Car类的一个具体实例,能够使用的是具体实例,而不是类。类只是给对象的创建提供了一个参考的模板。

在java中, 没有类就没有对象,然而类有时根据具体的功能需求,进行实际的分析,最终抽象出来的。

1.3 对象和引用的关系

引用“指向”对象。

使用类类型,数组类型,接口类型声明出来的变量,都可以指向对象,这种变量就是引用类型变量,简称引用。

在程序中,创建出对象后,直接使用并不方便,所以一般会用一个引用类型的变量去接收这个对象,这个就是所说的引用指向对象。

二、方法回顾和加深

方法一定是在定义在类中的,属于类的成员

2.1 方法的定义

格式: 修饰符 返回值类型 方法名(参数列表) 异常抛出类型{.....}
  1. 修饰符
public、static、abstract、final等等都是修饰符,一个方法可以有多个修饰符。例如程序入口main方法,就使用了public static这两个修饰符
注:如果一个方法或者属性有多个修饰符,这多个修饰符是没有先后顺序的。
  1. 返回类型
方法执行完,如果有要返回的数据,那么就要声明返回数据类型,如果没有返回的数据,那么返回类型就必须写void
只有构造方法(构造器)不写任何返回类型也不写void

【实例】

public String sayHello(){
	return "hello";
}

public int max(int a, int b){
	return a>b?a:b;
}

public void print(String msg){
    System.out.println(msg);
}
  • return 语句的作用:

    • return从当前的方法中退出,返回到该调用的方法的语句处,继续执行
    • return返回一个值给调用该方法的语句,返回值的数据类型必须与方法的声明中的返回值的类型一致
    • return后面也可以不带参数,不带参数就是返回空,其实主要目的就是用于想中断函数的执行,返回调用函数处
  • break语句的作用

    • break在循环体内,强行结束循环的执行,也就是结束整个循环过程,不在判断执行循环的条件是否成立,直接转向循环语句下面的语句。
    • 当break出现在循环体中的switch语句体内时,其作用知识跳出该switch语句体。
  1. 方法名
遵守java中标示符的命名规则即可
  1. 参数列表
根据需求定义,方法可以是无参的,也可以有一个参数,也可以有多个参数
  1. 异常抛出类型
如果方法中的代码在执行过程中,可能会出现一些异常情况,那么就可以在方法上把这些异常声明抛出,也可以同时声明抛出多个异常,使用逗号隔开即可。

【示例】

public void readFile(String file) throws IOException{
    
}

public void readFile(String file)  throws IOException, ClassNotFoundException {

}

2.2 方法调用

在类中定义了方法,这个方法中的代码并不会执行,当这个方法被调用的时候,方法中的代码才会被一行一行顺序执行。

  1. 非静态方法
没有使static修饰符的方法,就是非静态方法
调用这种方法的时候,是“一定”要使用对象的,因为非静态方法是属于对象的(非静态属性也是一样的)

【例子】

public class Student{
    public void say(){}
}
public class Demo01{
    public static void main(String[] args){
        Student s = new Student();
        s.say();
    }
}
  1. 静态方法
使用static修饰符修饰的方法,就是静态方法
调用这种方法的时候,可以使用对象调用,也可以使用类来调用,但是推荐使用类进行调用,因为静态方法是属于类的。(静态属性也是一样的)

【例子】

public class Student{
    public static void say(){}
}

public class Demo01{
	public static void main(String[] args){
        Student.say();
    }
}
  1. 类方法之间的调用

​ 假设同一个类中有俩个方法,a方法和b方法,a和b都是非静态方法,相互之间可以直接调用。

public class Demo02{
    public void a(){
        b();
    }
    public void b(){}
}

​ a和b都是静态方法,相互之间可以直接调用。

public class Demo03{
    public static void a(){
    	b();
    }
    public static void b(){}
}

​ a静态方法,b是非静态方法,a方法中不能直接调用b方法,但是b方法中可以直接调用a方法.

静态方法不能调用非静态方法!

public class Demo03{
    public static void a(){
    	b(); //报错
    }
    public  void b(){}
}

另外:在同一个类中,静态方法内不能直接访问到类中的非静态属性.

总结:类中方法中的调用,两个方法都是静态或者非静态都可以互相调用,当一个方法是静态,一个方法是非静态的时候,非静态方法可以调用静态方法,反之不能。

2.3 调用方法时的传参

  1. 形参和实参

【例子】

public class Demo01 {
    public static void main(String[] args){
        int x = 1;
        new Demo1().test(x);
        return;
    } 

    public void test(int a){
        System.out.println(a);
    }
}

参数列表中的a是方法test的形参(形式上的参数)

调用方法时的x是方法test的实参(实际上的参数)

注意:形参的名字和实参的名字都只是一个变量的名字,是可以随便写的,并不关心这个名字,而是关心变量的类型以及变量接收的值。

  1. 值传递和引用传递

调用方法进行传参时,分为值传递和引用传递两种。

如果参数的类型是基本数据类型,那么就是值传递

如果参数的类型是引用数据类型,那么就是引用传递

值传递是实参把自己变量本身存的简单数值赋值给形参

引用传递就是实参把自己变量本身存的对象内存地址赋值给形参

所以值传递和引用传递本质上是一样的,只不过传递的东西的意义不同。

【示例:值传递】

public class Test {
    public static void main(String[] args){
        int a = 1;
        System.out.println("before: a = " + a); //1
        changeNum(a);
        System.out.println("after: a = " + a); //1
    }

    public static void changeNum(int a){
        a = 10;
    }
}

【示例:引用传递】

public class Demo01 {
    public static void changeName(Student s) {
        s.name = "java";
    }

    public static void main(String[] args) {
        Student s = new Student();
        System.out.println("before: name = " + s.name); //null
        changeName(s);
        System.out.println("after: name = " + s.name); //java
    }
}

public class Student {
    String name;
}

2.4 this关键字

在类中,可以使用this关键字表示一些特殊的作用。

2.4.1 this在类中的作用

【区别成员变量和局部变量】

public class Student {
    private String name;
    public void setName(String name){
        //this.name 表示类中的属性name
        this.name = name;
    }
}

【调用类中的其他方法】

public class Student {
    private String name;
    public void setName(String name){
        this.name = name;
    }
    public void print(){
        //表示调用当前类中的setName方法
        this.setName("java");
    }
}

注意:默认情况下,setName("java")this.setName("java")的效果是一样的。

【调用类中的其他构造器】

//创建Student的实例的时候,如果没有赋初始值,那么就会调用没有写形参的构造器,这样的会给该对象的name赋值为java
//构造器重载
public class Student{
    private String name;
    public Student(){
        //调用一个构造器,并且参数的类型是String
        this("java");
    }
    public Student(String name){
        this.name = name;
    }
}

注:this的这种用法,只能在构造器中使用,普通的方法是不能这样使用的,并且这调用的代码只能出现在构造器中的第一句。

2.4.2 this关键字在类中的意义

this在类中表示当前类将来创建出来的对象

【例子】

public class Student{
    private String name;
    public Student(){
        System.out.println("this = " + this);
    }
    public static void main(String[] args){
        Student s = new Student();
        System.out.println("s = " + s);
    }
}

​ 运行后看结果可以发现,this和的打印结果一样,那么其实也就是变量s是从对象的外部执行对象,而this是对象的内部执行对象。

【this和s打印出来的内存地址是一样的,使用==比较的结果为true】

public class Student {
    public Student getStudent(){
        return this;
    }

    public static void main(String[] args) {
        Student s1 = new Student();
        Student s2 = s1.getStudent();
        System.out.println(s1 == s2); //true
    }
}

【类中的this是和s1相等还是和s2相等呢?】

public class Student{
    private String name;
    public void test(){
        System.out.println(this);
    }
    public static void main(String[] args){
        Student s1 = new Student();
        Student s2 = new Student();
        s1.test();
        s2.test();
    }
}

注:this指代的时对象的地址,而不是类的,对象不同this指代的地址则不同。

三、创建与初始化对象

  • 使用new关键字创建对象

使用new关键字创建的时候,除了分配内存空间之外,还会给创建好的对象进行默认的初始化,以及对类中构造器的使用。

main方法中的以下代码:
	Student s = new Student();
1) 为对象分配内存空间,将对象的实例变量自动初始化默认值为0/false/null。
2) 代码中实例变量有显式值,那么就将之前的默认值覆盖掉。
  	private String name = "java"
3) 调用构造器

4) 把对象内存地址赋值给变量(“=”操作)

四、构造器

类中的构造器也称为构造方法,是在进行创建对象的时候必须要调用的,并且构造器有以下两个特点:

  1. 必须和类的名字相同
  2. 必须没有返回值类型,也不能写void

构造器的作用:

  1. 使用new创建对象的时候必须使用类的构造器
  2. 构造器中的代码执行后,可以给对象中的属性初始化赋值

【演示】

public class Student {
    private String name;
    public Student(){
        name = "java"
    }
}

构造器重载:

除了无参构造器之外,很多时候还会使用有参构造器,在创建对象时,可以给属性赋值

public class Student{
    private String name;

    public Student(){
        name = "java";
    }

    public Student(String name) {
        this.name = name;
    }
}

构造器之间的调用:

使用this关键字,在一个构造器中可以调用另一个构造器代码

注意:this的这种用法不会产生新的对象,只是调用了构造器中的代码而已,一般情况下只有使用new关键字才会创建新对象

【演示】

public class Student{
    private String name;
    public Student(){
        this("java");
    }
    public Student(String name){
        this.name = name;
    }
}

默认构造器:

在java中,即使我们在编写类的时候没有写构造器,那么在编译之后会自动的添加一个无参构造器,这个无参构造器也被称为默认的构造器。

【示例】

public class Demo01{
    public static void main(String[] args){
        Student s = new Student(); 
    }
}

class Student {}

如果手动编写了一个构造器,编译后就不会添加任何构造器了

public class Demo01 {
    public static void main(String[] args){
        Student s = new Student();
    }
}

class Student {
    public Student(){
        System.out.println(this)
    }
}

五、内存分析

JAVA程序运行的内存分析

栈stack:

  1. 每个线程私有,不能实现线程间的共享
  2. 局部变量放置于栈中
  3. 栈是由系统自动分配,速度快!栈是一个连续的内存空间

堆heap:

  1. 放置new出来的对象!
  2. 堆是一个不连续的内存空间,分配灵活,速度慢!

方法区(也是堆):

  1. 被所有线程共享!
  2. 用来存放程序中永远不变或唯一的内容(类代码信息、静态变量、字符串常量)

引用类型的概念:

  1. java中,除了基本数据类型以外的其他类型称之为引用类型。
  2. java中的对象是通过引用来操作的(引用:reference,引用对象的地址)

属性(field,成员变量):

  1. 属性用于定义该类或该类对象包含的数据或者静态属性

  2. 属性作用范围是整个类体

  3. 属性的默认初始化:

    1. 在定义成员变量时,可以对其初始化,如果不对其初始化,Java使用默认的值对其初始化(数值:0,0.0 char:u0000,boolean:false,所有引用类型:null)
  4. 属性定义格式

[修饰符] 属性类型  属性名 = [默认值]

类的方法:

方法是类和对象动态行为特征的抽象。方法很类似于面向对象中的过程,面向过程中,函数是最基本单位,整个程序由一个个函数调用组成;面向对象中,整个程序的基本单位是类,方法是从属于类或对象的。

方法定义格式:

[修饰符] 方法返回值类型  方法名(形参列表) {
	//n条语句
}

java对象的创建和使用:

  • 必须使用new关键字创建对象

    • Person person = new Person();
  • 调用对象的成员变量

    • person.age
  • 调用对象的方法

    • person.setAge(23)

六、封装

定义一个对象的特性,有必要决定这些特性的可见性,即哪些特性对外部是可见的,哪些特性用于表示内部状态

通常,应禁止直接访问一个对象中属性的值,而应通过操作接口来访问,这称为信息隐藏

6.1 封装的步骤

  1. 使用private修饰需要封装的成员变量
  2. 提供一个公开的方法设置或者访问私有的属性,设置通过set/get方法,命名格式:set属性名()、get属性名(),属性的首字母要大写

【演示(不推荐)】

//对象能在类的外部“直接”访问 (不推荐)
public class Student {
    public String name;
    public void println(){
        System.out.println(this.name);
    }
}

public class Test {
    public static void main(String[] args){
        Student s = new Student();
        s.name = "java";
    }
}

【演示】

在类中一般不会把数据直接暴露在外部,而使用private(私有)关键字把数据隐藏起来

public class Student {
    private String name;

    //设置钩子方法
	public void setName(String name){
        this.name = name;
    }
	
    //设置钩子方法
    public void getName(){
        return this.name;
    }
    
}

public class Test {
    public static void main(String[] args){
        Student s = new Student();
        s.setName("java");
        System.out.println(s.getName());
    }
}

6.2 作用和意义

  1. 提高程序的安全性,保护数据
  2. 隐藏代码的实现细节
  3. 统一用户的调用接口
  4. 提高系统的可维护性
  5. 便于调用者调用

良好的封装,便于修改内部代码,提高可维护性

良好的封装,可进行数据完整性检测,保护数据的有效性

6.3 方法重载

类中有多个方法,有着相同的方法名,但是方法的参数各不相同,这种情况被称为方法的重载。方法的重载可以提供方法调用的灵活性。

方法重载必须满足以下条件:

  1. 方法名必须相同
  2. 参数列表必须不同(参数的类型、个数、顺序的不同)
public void test(String str){}
public void test(int a){}

在java中,判断一个类中的两个方法是否相同,主要参考两个方面:方法面子和参数列表

七、继承

提高代码的复用性

Java中的类没有多继承,只有单继承!!接口可以多继承!

  1. 继承是类和类之间的一种关系。除此之外,类和类之间的关系还有依赖、组合、聚合等。
  2. 继承关系的俩个类,一个为子类(派生类),一个为父类(基类)。子类继承父类,使用关键字extends来表示。
public class Student extends Person{}
  1. 子类和父类之间,从意义上讲应该具有“is a”的关系
student is a person
dog is a animal
  1. 类和类之间的继承是单继承
一个子类只能"直接"继承一个父类,就像是一个人只能有一个亲生父亲
一个父类可以被多子类继承,就像一个父亲可以有多个孩子
注:java中接口和接口之间,有可以继承,并且是多继承。
  1. 父类中的属性和方法可以被子类继承

​ 子类中继承了父类中的属性和方法后,在子类中能不能直接使用这些属性和方法,是和这些属性和方法原有的修饰符(public protected default private)相关的。

例如:

​ 父类中的属性和方法使用public修饰,在子类中继承后"可以直接"使用

​ 父类中的属性和方法使用private修饰,在子类中继承后"不可以直接"使用

父类中的构造器是不能被子类继承的,但是子类的构造器中,会隐式的调用父类中的无参构造器(默认使用super关键字)。

7.1 Object类

java中的每一个类都是"直接" 或者 "间接"的继承了Object类,所以每一个对象都和Object类有“is a”的关系。从API文档中,可以看到任何一个类最上层的父类都是Object。(Object类本身除外)AnyClass is aObject。

System.out.println("任何对象" instanceof Object);
//输出结果:true
//注:任何对象也包含数组对象

例如:
//编译后,Persion类会默认继承Object
public class Person{}

//student 是间接继承了object
public class Student extents Person{}

在Object类中供了一些方法被子类继承,那么就意味着,在java中,任何一个对象都可以调用这些被继承过来的方法。(因为Object是所以类的父类)

例如:toString方法、equals方法、getClass方法等

注:Object类中的每一个方法之后都会使用到.

7.2 Super关键字

子类继承父类之后,在子类中可以使用this来表示访问或调用子类中的属性或方法,使用super就表示访问或调用父类中的属性和方法。

  1. super的使用

【访问父类中的属性】

public class Person {
    public String name = "zs";
}

public class Student extends Person{
    public String name = "ls";

    public void test(String name){
        System.out.println(name);
        System.out.println(this.name);
        System.out.println(super.name);
    }

    public static void main(String[] args) {
        Student s = new Student();
        s.test("ceshi");
    }
}

【调用父类中的方法】

public class Person {
    public void print(){
        System.out.println("Person");
    }
}

public class Student extends Person {
    public void print(){
        System.out.println("student");
    }
    public void test() {
        print();
        this.print();
        super.println();
    }
    public static void main(String[] args) {
        Student s = new Student();
        s.test();
    }
}

【调用父类中的构造器】

//如果没有写构造方法  编译的时候会默认加上一个无参构造器
public class Person{}

public class Student extends Person {
    //编译通过,子类构造器中会隐式的调用父类的无参构造器
    //super();
    public Student(){}
}

父类没有无参构造

public class Person{
    public String name;
    public Person(String anme){
        this.name = name;
    }
}
public class Student extends Persion(){
    //编译报错,子类构造器中会隐式的调用父类的无参构造器,但是父类中没有无参构造器
	//super();
    public Student(){}
    public static void main(String[] args) {
        Student s = new Student();
        s.test();
    }
}

【显示的调用父类的有参构造器】

public class Person{
    public String name;
    public Person(String anme){
        this.name = name;
    }
}
public class Student extends Persion(){
    //编译通过,子类构造器中显式的调用父类的有参构造器
    //指定了调用父类的有参构造器,不再使用父类默认的无参构造,报错消失
    public Student(){
        super("java");
    }
    public static void main(String[] args) {
        Student s = new Student();
        s.test();
    }
}

注:不管是显式还是隐式的父类的构造器,super语句一定要出现在子类构造器中第一行代码。所以this和super不可能同时使用它们调用构造器的功能,因为它们都要出现在第一行代码位置。

  • super使用的注意的地方
  1. 用super调用父类构造方法,必须是构造方法中的第一个语句。
  2. super只能出现在子类的方法或者构造方法中
  3. super和this不能同时调用构造方法(因为this也是在构造方法的第一个语句)
  • super和this的区别
  1. 代表的事物不一样:

    1. this:代表所属方法的调用者(对象)
    2. super:代表父类对象的引用空间
  2. 使用前提不一样

    1. this:在非继承的条件下也可以使用。
    2. super:只能在继承的条件下才能使用。
  3. 调用构造方法:

    1. this:调用本类的构造方法。
    2. super:调用的父类的构造方法

7.3 方法重写

  1. 方法重写只存在于子类和父类(包括直接父类和间接父类)之间,在同一个类中方法只能被重载,不能被重写
  2. 静态方法不能重写
    1. 父类的静态方法不能被子类重写为非静态方法
    2. 父类的非静态方法不能被子类重写为静态方法
    3. 子类可以定义与父类的静态方法同名的静态方法(但是这个不是覆盖)

【例子】

A类继承B类 A类和B类中都有一个相同的静态方法test

B a = new A();
a.test(); //调用到的是B类中的静态方法test

A a = new A();
a.test(); //调用的是A类中的静态方法test

可以看出静态方法的调用只和变量声明的类型相关,这个和非静态方法的重写之后的效果完全不同

测试

  • 私有方法不能被子类重写,子类继承父类后,是不能直接访问父类中的私有方法的,就更谈不上重写了。
public class Person {
    private void run() {}
}

//编译通过,但这不是重写,只是两个类中分别有自己的私有方法。
public class Student extends Person{
    private void run(){}
}
  1. 重写的语法
    1. 方法名必须相同
    2. 参数列表必须相同
    3. 访问控制修饰符可以被扩大,不能被缩小: **public > protected > default > private**
    4. 抛出异常类型的范围可以被缩小,但是不能被扩大:**ClassNotFoundException <--- Exception**
    5. 返回值类型可以相同,也可以不同,如果不同,子类重写后的方法返回类型必须是父类方法返回类型的子类型
      1. 例如:父类返回的是Person,子类重写后返回的可以是Person,也可以是Person的子类

注意:

一般情况下,重写的方法会和父类中的方法的声明完全保持一致,只是方法的是实现不同(大括号中代码不一样)

public class Person{
    public void run(){}

    protected object test() throw Exception {
        return null;
    }
}
//编译通过,子类继承父类,重写了run和test方法
public class Student extends Person{
	public void run(){}

    public String test(){
        return "";
    }
}

为什么要重写?

子类继承父类,继承了父类中的方法,但是父类中的方法并不一定能满足子类中的功能需要,所以子类中需要把方法进行重写。

八、多态

8.1 认识多态

允许不同类的对象对同一消息做出响应,即同一消息可以根据发送对象的不同而采用不同的行为方式。

  1. 一个对象的实际类型是确定的:

    1. 例如:new Student(); new Person();等
  2. 可以指向对象的引用类型有很多

    1. 一个对象的实现类型虽然是确定的,但是这个对象所属的类型可能有很多种
Student s1 = new Student();
Person s2 = new Student();
Object s3 = new Student();

因为Person和Object都是Student的父类型。如果声明的变量类型是父类,但是赋值的时候赋值的是子类的引用,当调用这个引用的时候,可以被调用的方法列表是按照父类的方法列表读取,如果父类的方法被子类重写了,那么调用的方法则为子类的方法。如果一个方法在子类中存在,但是在父类中不存在,那么将无法调用(编译报错)。

一个对象的实际类型是确定,但是可以指向这个对象的引用的类型,却是可以是这对象实际类型的任意父类型。

  1. 一个父类引用可以指向它的任何一个子类对象
Object o = new AnyClass();
Person p = null;
p = new Student();
p = new Teacher();
p = new Person();
  1. 多态中的方法调用
public class Person{
    public void run(){}
}
public class Student extends Person{
    
}
  1. 调用到的run方法,是Student从Person继承过来的run方法
main:
Person p = new Student();
p.run();

例如:

public class Person{
    public void run(){}
}

public class Student extends Person{
    public void run(){
        //重写run方法
    }
}

//调用到的run方法,是Student中重写的run方法
main:
Person p = new Student();
p.run();

注:子类继承父类,调用a方法,如果a方法在子类中没有重写,那么就是调用的是子类继承父类的a方法,如果重写了,那么调用的就是重写之后的方法。

子类中独有方法的调用:

public class Person{
	public void run(){}
}
public class Student extends Person{
	public void test(){}

    public static void main(String[] args){
        Person p = new Student();
        //调用到继承的run方法
        p.run();
        //编译报错,因为编译器检查变量p的类型是Person,但是在Person类中并没有发现test方法,所以编译报错。
        p.test();
    }
}

注:一个变量x,调用一个方法test,编译器是否能让其编译通过,主要是看声明变量x的类型中有没有定义test方法,如果有则编译通过,如果没有则编译报错,而不是看x所指向的对象中有没有test方法。

原理:编译看左边,运行不一定看右边

编译看左边的意思:java 编译器在编译的时候会检测引用类型中含有指定的成员,如果没有就会报错。子类的成员是特有的,父类的没有的,所以他是找不到的。

8.2 重写、重载和多态的关系

重载是编译时的多态

调用重载的方法,在编译期间就要确定调用的方法是谁,如果不能确定则编译报错

重写是运行时的多态

调用重写的方法,在运行期间才能确定这个方法到底是哪个对象中的。这个取决于调用方法的引用,在运行期间所指向的对象是谁,这个引用指向哪个对象那么调用的就是哪个对象中的方法。(java中的方法调用,是运行时动态和对象绑定的)

8.3 多态注意事项

  1. 多态是方法的多态,属性没有多态性
  2. 编写程序时,如果想调用运行时类型的方法,只能进行类型转换。不然通不过编译器的检查,但是如果两个没有关联的类型进行强制转换,会报:ClassCastException。比如:本来是狗,把它转换成猫,就会报这个异常
  3. 多态的存在要有3个必要条件:继承,重写,父类引用指向子类对象。

8.4 多态存在的条件

  1. 有继承关系

  2. 子类重写父类方法

    1. 以下三种类型的方法是没办法表现出多态特性的(因为不能被重写):
      1. static方法:因为被static修饰的方法是属于类的,不属于实例
      2. final方法:因为被final修饰的方法无法被子类重写
      3. private方法和protected方法:前者是因为private修饰的方法对子类不可见,后者是因为尽管被protected修饰的方法可以被子类见到,也可以被子类重写,但是它是无法被外部所引用的,一个不能被外部引用的方法,所以也就无法谈多态。
  3. 父类引用指向子类对象

8.5 方法绑定(method binding)

​ 执行调用方法时,系统根据相关信息,能够执行内存地址中代表该方法的代码,分为静态绑定和动态绑定。

  • 静态绑定:

​ 在编译期完成,可以提高代码执行速度。

  • 动态绑定

​ 通过对象调用的方法,采用动态绑定机制,这虽然让编程变得灵活,但是降低了代码的执行速度,这也是java比c/c++速度慢的主要因素之一。Java中除了final类、final方法,static方法,所有的方法都在jvm在运行期间才进行动态绑定的。

  • 多态:

​ 如果编译时类型和运行时类型不一致,就会造成多态。

8.6 instanceof和类型转换

在 Java 中可以使用 instanceof 关键字判断一个对象是否为一个类(或接口、抽象类、父类)的实例,语法格式如下所示。

  1. instanceof
public class Person {
    public void run(){}
}
public class Student extends Person{}

public class Teacher extends Person{}
public class Demo01{
    public static void main(String[] args){
        Person s = new Student();
        System.out.println(s instanceof Student); //true
        System.out.println(s instanceof Object);  //true
        System.out.println(s instanceof Teacher); //false
        System.out.println(s instanceof String); //编译不通过

        Person s = new Person();
        System.out.println(s instanceof Student); //false
        System.out.println(s instanceof Object);  //true
        System.out.println(s instanceof Teacher); //false
    }
}

【分析2】

System.out.println(x instanceof y);
输出结果是true还是false,主要是看变量x所指向的对象实际类型是不是Y类型的"子类型"。
  1. 类型转换
public class Person {
    public void run(){}
}
public class Student extends Person{
    public void go(){}
}
public class Teacher extends Person{}

【为什么要类型转换】

//编译报错,因为P声明的类型Person中没有go方法
Person p = new Student();
p.go();

//需要把变量p的类型进行转化
Person p = new Student();
Student s = (Student)p;
s.go();

//注意这两种形式前面必须要有两个小括号
((Student)p).go();

【类型转换中的问题】

//编译通过,运行不报错
Object o = new Student();
Person p = (Person)o;

//编译通过 运行没问题
Object o = new Student();
Student s = (Student)o;

//编译通过  运行报错
Object o = new Teacher();
Student s = (Student)o;

转换关系可以升级转换,但是不可以降级或者平调


即:
X x = (X)o;
运行是否报错,主要是变量o所指向的对象实现类型,是不是X类型的子类型,如果不是则运行就会报错

【总结】

  1. 父类引用可以指向子类对象,子类引用不能指向父类对象

  2. 如果子类对象直接赋值给父类引用交upcasting向上转型,向上转型不用强制转型

    1. Father f = new Son();
  3. 把指向子类对象的父类引用赋给子类引用叫向下转型,要强制转型。

    1. father就是一个指向子类对象父类引用,把father赋值给子类引用son,即Son s = (Son)father;
    2. 其中**father**前面的(Son)必须添加,进行强制转换
  4. upcasting会丢失子类特有的方法,但是子类overriding父类的方法,子类方法有效

  5. 向上转型的作用,减少了重复代码,父类为参数,调用时用子类作为参数,就是利用了向上转型,这样使代码变得简洁,体现了Java的抽象编程思想。

九、修饰符

9.1 static 修饰符

  1. static变量

在类中,使用static修饰的成员变量,就是静态变量,反之为非静态变量。

静态变量和非静态变量的区别

静态变量属于类的,“可以”使用类名来访问,非静态变量是属于对象的,“必须”使用对象来访问。

public class Student {
    private static int age;
    private double score;

    public static void main(String[] args){
        Student s = new Student();

        //推荐使用类名访问静态成员
        System.out.println(Student.age);
        System.out.println(s.age);

        System.out.println(s.score);
    }
}

静态变量对于类而言在内存中只有一个,能被类的所有实例共享,实例变量对于类的每个实例都有一份,它们之间互不影响。

public class Student {
    private static int count;
    private int num;
    public Student(){
        count++;
        num++;
    }
    public static void main(String[] args){
        Student s1 = new Student();
        Student s2 = new Student();

        System.out.println(Student.count); // 2
        System.out.println(s2.num); // 1
        System.out.println(s2.count); // 2
    }
}

​ 在加载类的过程中为静态变量分配在内存,实例变量在创建对象时分配内存,所以静态变量可以使用类名来直接访问,而不需要使用对象来访问。

  1. static方法

在类中,使用static修饰的成员变量,就是静态方法,反之为非静态方法。

  • 静态方法和非静态方法的区别

    • 静态方法“不可以”直接访问类中的非静态变量和非静态方法,但是“可以”访问类中的静态变量和静态方法。
  • 注意:

    • this和super在类中属于非静态的变量。(静态方法中不能使用)
public class Student{
    private static int count;
    private int num;
    public void run(){}
    public static void go(){}

    public static void test(){
        //编译通过
        System.out.println(count);
        go();

        //编译报错
        System.out.println(num);
        run():
    }
}

​ 非静态方法“可以”直接访问类中的非静态变量和非静态方法,也“可以”直接访问类中的静态变量和静态方法。

public class Student{
    private static int count;
    private int num;
    public void run(){}
    public static void go(){}

    public void test(){
        //编译通过
        System.out.println(count);
        go();

        //编译通过
        System.out.println(num);
        run();
    }
}

思考:为什么静态方法和非静态方法不能直接相互访问?加载顺序的问题!

父类的静态方法可以被子类继承,但是不能被子类重写。

public class Person {
	public static void method() {}
}

//编译报错
public class Student extends Person {
	public void method(){}
}

例如:
public class Person {
	public static void test() {
		System.out.println("Person");
	}
}
//编译通过,但不是重写
public class Student extends Person {
	public static void test(){
		System.out.println("Student");
	}
}

main:
Perosn p = new Student();
p.test();//输出Person
p = new Person();
p.test();//输出Perosn

父类的非静态方法不能被子类重写为静态方法

public class Person {
    public void test() {
    	System.out.println("Person");
    }
}

//编译报错
public class Student extends Person {
    public static void test(){
    	System.out.println("Student");
    }
}
  1. 代码块和静态代码块
public class Person{
    {
        //代码块(匿名代码块)
    }

    static {
        // 静态代码块
    }
}

【匿名代码块和静态代码块的执行】

因为没有名字,在程序并不能主动调用这些代码块。

匿名代码块是在创建对象的时候自动执行的,并且在构造器执行之前。同时匿名代码块在每次创建对象的时候都会自动执行。

静态代码块是在类加载完成之后就自动执行,并且只执行一次.

注:每个类在第一次被使用的时候就会被加载,并且一般只会加载一次。

public class Student {
	{
		System.out.println("匿名代码块");
	}
    
    static{
    	System.out.println("静态代码块");
    }
    
	public Person(){
    	System.out.println("构造器");
    }
}

main:
Student s1 = new Student();
Student s2 = new Student();
Student s3 = new Student();


//输出
静态代码块
匿名代码块
构造器

匿名代码块
构造器

匿名代码块
构造器

【作用】

匿名代码块的作用是给对象的成员变量初始化赋值,但是因为构造器也能完成这项工作,所以匿名代码块使用的并不多

静态代码块的作用是给类中的静态成员变量初始化赋值

例如:

public class Person {
    public static String name;
    static {
        name = "java";
    }
    public Person(){
        name = "zs";
    }
}

maim:
System.out.println(Person.name); //java

注:在构造器中给静态变量赋值,不能保证能赋值成功,因为构造器是在创建对象的时候才执行的,但是静态变量可以不创建对象而直接使用类名来访问。

  1. 创建和初始化对象的过程
Student s = new Student();

【Student类之前没有进行类加载——执行流程】

  • 类加载,同时初始化类中静态的属性
  • 执行静态代码块
  • 分配内存空间,同时初始化非静态的属性(默认0/false/null)
  • 调用student中父类构造器
  • 对student中的属性进行显示赋值(如果有的话)
  • 执行匿名代码块
  • 执行构造器
  • 返回内存地址

注:子类中非静态属性的显示赋值在父类构造器执行完之后的子类中的匿名代码执行之前的时候。

public class Person{
	private String name = "zs";
	public Person() {
		System.out.println("Person构造器");
		print();
	}
	public void print(){
        System.out.println("Person print方法: name = "+name);
	}
}

public class Student extends Person{
	private String name = "tom";
	{
		System.out.println("Student匿名代码块");
	}
	static{
		System.out.println("Student静态代码块");
	}
	public Student(){
		System.out.println("Student构造器");
	}
	public void print(){
		System.out.println("student print方法: name = "+name);
	}
	public static void main(String[] args) {
		new Student();
	}
}

输出:
Student静态代码块
Person构造器
student print方法: name = null
Student匿名代码块
Student构造器
    
Student s = new Student();
Student类之前已经进行了类加载
1.分配内存空间,同时初始化非静态的属性(赋默认值,0/false/null)
2.调用Student的父类构造器
3.对Student中的属性进行显示赋值(如果有的话)
4.执行匿名代码块
5.执行构造器
6.返回内存地址
  1. 静态导入

静态导包就是java包的静态导入,用import static代替import静态导入包是JDK1.5中的新特性。意思是导入这个类里的静态方法。

好处:这种方法的好处就是可以简化一些操作,例如打印操作System.out.println(…);就可以将其写入一个静态方法print(…),在使用时直接print(…)就可以了。但是这种方法建议在有很多重复调用的时候使用,如果仅有一到两次调用,不如直接写来的方便。

import static java.lang.Math.random;
import static java.lang.Math.PI;

public class Test {
	public static void main(String[] args) {
		//之前是需要Math.random()调用的
		System.out.println(random());
		System.out.println(PI);
	}
}

9.2 final修饰符

  1. 修饰类

用final修饰的类不能被继承,没有子类

例如:

我们是无法写一个类去继承String类,然后对String类型扩展的,因为API中已经被String类定义为final的了。也可以定义final修饰的类:

public final class Action {
    
}
//编译报错
public class Go extends Action{
    
}
  1. 修饰方法

​ 用final修饰的方法可以被继承,但是不能被子类重写

例如:

​ 每个类都是Object类的子类,继承了Object中的众多方法,在子类中可以重写toString方法,equals方法,但是不能重写getClass方法wait方法等,因为这些方法都是使用final修饰的。

public class Person{
    public final void print(){}
}

//编译报错
public class Student extends Person{
    public void print(){}
}
  1. 修饰变量

用final修饰的变量表示常量,只能被赋一次值,使用final修饰的变量也就变成了常量,因为值不再变了。

【修饰局部变量】

public class Person{
    public void print(final int a){
        a = 1; //编译报错,不能再次赋值,传参的时候已经赋值过了
    }
}

public class Person{
    public void print(){
        final int a;
        a = 1;
        //编译报错
        a = 2;
    }
}

【修饰成员变量-非静态成员变量】

public class Person{
    private final int a;
}

只有一次机会,可以给此变量a赋值的位置:
    声明的同时赋值
    匿名代码中赋值
    构造器中赋值(类中出现的所有构造器都要写)

【修饰成员变量-静态成员变量】

public class Person{
    private static final int a;
}
只有一次机会,可以给此变量a赋值的位置:
    声明的同时赋值
    静态代码块中赋值

【修饰引用变量】

main:
final Student s = new Student();
//编译通过
s.setName("java");
s.setName("java01");

//编译报错,不能修改引用s指向的内存地址
s = new Student();

9.3 abstract修饰符

abstract修饰符可以用来修饰类和方法,如果修饰方法,那么被修饰的方法就叫做重新方法,如果修饰类,那么叫做抽象类

  1. 抽象类和抽象方法的关系

抽象类中可以没有抽象方法,但是存在抽象方法的类必须是抽象类。

  1. 语法
public abstract class Action{
    public abstract void doSomething();
}

public void doSomething(){...}

对于这个普通方法来讲:

public void doSomething()这部分是方法的声明

{...}这部分是方法的实现,如果大括号中什么都没有写,就叫做方法的空实现

声明类的同时,加上abstract修饰符就是抽象类

声明方法的时候,加上abstract修饰符,并且去掉方法的大口号,同时结尾加上分号,该方法就是抽象方法。

  1. 特点及应用

抽象类不能使用new关键字类创建对象,它是用来让子类继承的。

抽象方法,只有方法的声明,没有方法的实现,它是用来让子类实现的。

注:子类继承抽象后,需要实现抽象类中没有实现的抽象方法,否则这个子类也要声明为抽象类。

public abstract class Action{
    public abstract void doSomething();
}

main:
Action a = new Action();

//子类继承抽象类
public class Eat extends Action{
    //实现父类中没有实现抽象方法
    public void doSomething(){
        //code
    }
}

main:
Action a = new Eat();
a.doSomething();

思考:

  1. 抽象类不能new对象,那么抽象类中有没有构造器?
抽象类是不能被实例化,抽象类的目的就是为实现多态中的共同点,抽象类的构造器会在子类实例化时调用,因此它也是用来实现多态中的共同点构造,不建议这样使用!
  1. 抽象类和抽象方法意义(为什么要编写抽象类、抽象方法)
打个比方,要做一个游戏。如果要创建一个角色,如果反复创建类和方法会很繁琐和麻烦。建一个抽象类后。若要创建角色可直接继承抽象类中的字段和方法,而抽象类中又有抽象方法。如果一个角色有很多种职业,每个职业又有很多技能,要是依次实例这些技能方法会显得想当笨拙。定义抽象方法,在需要时继承后重写调用,可以省去很多代码。
总之抽象类和抽象方法起到一个框架作用。很方便后期的调用和重写
抽象方法是为了程序的可扩展性。重写抽象方法时即可实现同名方法但又非同目的的要求。

十、接口

10.1 接口的本质

普通类:只有具体实现

抽象类:具体实现和规范(抽象方法)都有!

接口:只有规范!

【为什么要使用接口?就扣和抽象类的区别是什么?】

  • 接口就是比“抽象类”还抽象的“抽象类”,可以更加规范的对子类进行约束,全面地专业地实现了:规范和具体实现的分离
  • 抽象类还提供某些具体的实现,接口不提供任何实现,接口中所有的方法都是抽象方法。接口是完全面向规范的,规定了一批类具有的公共方法规范。
  • 从接口的实现者角度看,接口定义了可以向外提供的服务
  • 从接口的调用者角度看,接口定义了实现者提供哪些服务
  • 接口是两个模块之间通讯的标准,通信的规范,如果能把你要审计的系统之间模块之间的接口定义好,就相当于完成了系统的设计大纲,剩下的就是舔砖加瓦的具体实现了。以后的代码中要有“面向接口”的思想方式去编程

【接口的本质探讨】

  • 接口就是规范,定义的是一组规则,体现了现实世界中“如果你是。。。。则必须。。。”的思想。如果你是天使,则必须能飞,如果你是汽车,则必须能跑。
  • 接口的本质是契约,就像我们人间的法律一样,制定好后大家都遵守。
  • OO的精髓,是对对象的抽象,最能体现这一点的就是接口。为什么我们讨论设计模式都只针对具备了抽象能力的语言(比如c++、java、c#等),就是因为设计模式所研究的,实际上就是如何合理的去抽象。

10.2 接口与抽象类的区别

抽象类也是类,除了可以写抽象方法以及不能直接new对象之外,其他的和普通类没有什么区别。接口已经另一种类型了,和类的本质是有区别的,所以不能用类的标准去衡量接口。

  • 声明类的关键字是class,声明接口的关键字是interface

    • 抽象类是用来被继承的,java中的类是单继承
    • 类A继承了抽象类B,那么类A的对象就属于B类型了,可以使用多态。(一个父类的引用,可以指向这个父类的任意子类对象) 注:继承的关键字是extends
    • 接口是用来类实现的,java中的接口可以被类实现。
    • 类A实现接口B,C,D,E... 那么类A的对象就属于B,C,D,E等类型了,可以使用多态。注意:实现的关键字是implements。

10.3 接口中的方法都是抽象方法

public interface Action {
    public abstract void run();

    //默认就是public abstract 修饰的
    void test();
    public void go();
}

10.4 接口中的变量都是静态常量(public static final修饰)

接口中可以不写任何属性,如果写属性了,该属性必须是public static final修饰的常量。

  • 注意:
    • 可以直接使用接口名访问其属性,因为是**public static**修饰的
    • 声明的同时必须赋值。(因为接口中不能编写静态代码块)
public interface Action{
    public static final String NAME = "java";
	//系统默认public static final
    int AGE = 20;
}

main:
System.out.println(Action.NAME);
System.out.println(Action.AGE);

10.5 一个类可以实现多个接口

public class Student implements A, B, C, D{
    //Student需要实现接口A  B  C  D中所有的抽象方法
    //否则Student类就要声明为抽象类,因为有抽象方法没有实现
}

main:
A s1 = new Student(); //s1只能调用接口A中声明的方法以及Object中的方法
B s2 = new Student(); //s2只能调用接口B中声明的方法以及Object中的方法
C s3 = new Student(); //s3只能调用接口C中声明的方法以及Object中的方法
D s4 = new Student(); //s4只能调用接口D中声明的方法以及Object中的方法

注意:必要可以类型强制转换

例如:接口中有test(),接口B中有run()

A s1 = new Student();
s1.test();

B s2 = new Student();
s2.run();

if (s1 interface B){
    ((B)s1).run();
}

10.6 一个接口可以继承多个父类接口

public interface A {
    public void testA();
}

public interface B {
    public void testB();
}
//接口C把接口A B中的方法都继承过来了
public interface C extends A, B {
    public void testc();
}

//Student 相当于实现了A B C三个接口,需要实现所有的抽象方法
//Student 的对象也就同时属于A类型  B类型  C类型
public class Student implements c {
    public void testA(){}
    public void testB(){}
    public void testC(){}
}

main:
C o = new Student();
System.out.println(o interface A);  //true
System.out.println(o interface B);  //true
System.out.println(o interface C);  //true
System.out.println(o interface Student);  //true
System.out.println(o interface Object);  //true
System.out.println(o interface Teacher);  //false

//编译报错
System.out.println(o interface String);
  • 注意:System.out.println(o interface X);
    • 如果o是一个接口类型声明的变量,那么只要X不是一个final修饰的类,该代码就能通过编译,至于其结果是不是true,就要看变量o指向的对象的实际类型,是不是X的子类或者实现类了。
    • 一个引用所指向的对象,是有可能实现任何一个接口的(多态)

10.7 接口的作用

接口的最主要的左右是达到统一访问,就是在创建对象的时候用接口创建

[接口名] [对象名] = new [实现接口的类]
  • 总结:

    • Java接口中的成员变量默认都是public,static,final类型的(都可忽略),必须被显示初始化,即接口中的成员变量为常量(大写,单词之间用“_”分隔)
    • Java接口中的方法默认都是public,abstract类型的(都可忽略),没有方法体,不能被实例化
    • Java接口中只能包含public,static,final类型的成员变量和public,sabstract类型的成员方法
    • 接口中没有构造方法,不能被实例化
    • 一个接口不能实现(implements)另一个接口,但是他可以继承多个其他的接口
    • java接口必须通过类来实现它的抽象方法
    • 当类实现了某个Java接口时,他必须实现接口中所有的抽象方法,否则这个类必须声明为抽象类
    • 不允许创建接口的实例(实例化),但允许定义接口类型的引用变量,给变量引用实现了这个接口的类的实例
    • 一个类只能继承一个直接父类,但是可以实现多个接口,间接的实现了多继承
  • 【实例】

interface SwimInterface {
    void swim();
}
class Fish {
    int fins = 4;
}
class Duck {
    int leg = 2;
    void egg(){};
}
class Goldfish extends Fish implements SwimInterface {
    @Override
    public void swim(){
        System.out.println("Goldfish can swim!");
    }
}

class SmallDuck extends Duck implements SwimInterface {
    public void egg(){
        System.out.println("SmallDuck can lay eggs");
    }
    @Override
    public void swim(){
        System.out.println("smallduck can swim");
    }
}

public class InterfaceDemo{
    public static void main(String[] args){
        Goldfish goldfish = new Goldfish();
        goldfish.swim();

        SmallDuck smallDuck = new SmallDuck();
        smallDuck.swim();
        smallDuck.egg();
    }
}

十一、内部类

11.1 内部类概述

内部类就是一个类的内部定义一个类,比如,A类中定义一个B类,那么B类相对于A类来说就称为内部类,而A类相对B类就是外部类了。

内部类不是在一个java源文件中编写两个平行的两个类,而是在一个类的内部再定义另外一个类,我们可以把外部的类称为外部类,在其内部编写的类称为内部类。

  • 内部类分为四种:
    • 成员内部类
    • 静态内部类
    • 局部内部类
    • 匿名内部类

11.2 成员内部类(实例内部类,非静态内部类)

注意:

成员内部类中不能写静态属性和方法

【定义一个内部类】

//在A类中申明了一个B类,此B类就在A的内部,并且在成员变量的位置上,所以就称为成员内部类
public class Outer {
    private int id;
    public void out(){
    	System.out.println("这是外部类方法");
    }
    class Inner{
        public void in(){
        	System.out.println("这是内部类方法");
        }
    }
}

【实例化内部类】

实例化内部类,首先需要实例化外部类,通过外部类调用内部类

public class Outer{
    private int id;
    public void out(){
        System.out.println("这是内部类方法");
    }
    class Inner{
        public void in(){
            System.out.println("这是内部类方法");
        }
    }
}

public class Test{
    public static void main(String[] args){
        //实例化成员内部类分两步
        //1. 实例化外部类
        Outer outObject = new Outer();
        //2. 通过外部类调用内部类
        Outer.Inner inObject = outObject.new Inner();
        //测试:调用内部类中的方法
        inObject.in();
    }
}

【成员内部类能干什么?】

  1. 访问外部类的所有属性(包括外部类私有的成员变量,方法)
public class Outer {
    private int id;
    public void out(){
        System.out.println("这是外部方法");
    }

    class Inner{
        public void in(){
            System.out.println("这是内部类方法");
        }
        
        //内部类访问外部私有的成员变量
        public void useId(){
            System.out.println(id + 3);
        }

        //内部类访问外部类
        public void useOut(){
            out();
        }
    }
}

public class Test{
    public static void main(String[] args){
        //实例化成员内部类分两步
    	//1、实例化外部类
    	Outer outObject = new Outer();
    	//2、通过外部类调用内部类
    	Outer.Inner inObject = outObject.new Inner();
    	//测试
    	inObject.useId();//打印3,因为id初始化值为0,0+3就为3,其中在内部类就使用了
    	外部类的私有成员变量id。
    	inObject.useOut();//打印:这是外部类方法
    }
}
  1. 如果内部类中的变量名和外部类的成员变量名一样,要通过创建外部类对象.属性来访问外部类属性,通过this.属性访问内部类成员属性
public class Outer {
    private int id;//默认初始化0
    public void out(){
    	System.out.println("这是外部类方法");
    }
    class Inner{
    	private int id=8; //这个id跟外部类的属性id名称一样。
    	public void in(){
    		System.out.println("这是内部类方法");
    	}
		public void test(){
			System.out.println(id);//输出8,内部类中的变量会暂时将外部类的成员变量给隐藏
    		//如何调用外部类的成员变量呢?通过Outer.this,想要知道为什么能通过这个来调用,
            //就得明白下面这个原理,想实例化内部类对象,就必须通过外部类对象,
            //当外部类对象来new出内部类对象时,会把自己(外部类对象)的引用传到了内部类中,
            //所以内部类就可以通过Outer.this来访问外部类的属性和方法,
            //到这里,你也就可以知道为什么内部类可以访问外部类的属性和方法,
            //这里由于有两个相同的属性名称,所以需要显示的用Outer.this来调用外部类的属性,
            //平常如果属性名不重复,那么我们在内部类中调用外部类的属性和方法时,前面就隐式的调用了Outer.this。
			System.out.println(Outer.this.id);//输出外部类的属性id。也就是输出0
		}
	}
}
  • 借助成员内部类,来总结内部类(包括4种内部类)的通用用法:
    • 要想访问内部类中的内容,必须通过外部类对象来实例化内部类。
    • 能够访问外部类所有的属性和方法,原理就是在通过外部类对象实例化内部类对象时,外部类对象把自己的引用传进了内部类,使内部类可以用通过Outer.this去调用外部类的属性和方法,一般都是隐式调用了,但是当内部类中有属性或者方法名和外部类中的属性或方法名相同的时候,就需要通过显式调用Outer.this了。

【例子】

public class MemberInnerClassTest {
    private String name;
    private static int age;
    public void run(){}
    public static void go(){}
    public class MemberInnerClass{
   		private String name;
    	//内部类访问外部类
   		public void test(String name){
            System.out.println(name);
            System.out.println(this.name);
            System.out.println(MemberInnerClassTest.this.name);
            System.out.println(MemberInnerClassTest.age);
            MemberInnerClassTest.this.run();
            MemberInnerClassTest.go();
    	}
    }
    //外部类访问成员内部类
	//成员内部类的对象要 依赖于外部类的对象的存在
	public void test(){
		//MemberInnerClass mic = MemberInnerClassTest.this.new
		//MemberInnerClass();
		//MemberInnerClass mic = this.new MemberInnerClass();
		MemberInnerClass mic = new MemberInnerClass();
		mic.name = "tom";
		mic.test("hua");
	}
	public static void main(String[] args) {
    	//MemberInnerClass mic = new MemberInnerClass();这个是不行的,this是动态的。
    	//所以要使用要先创建外部类对象,才能使用
    	MemberInnerClassTest out = new MemberInnerClassTest();
    	MemberInnerClass mic = out.new MemberInnerClass();
    	//如果内部类是private,则不能访问,只能通过内部方法来调用内部类
    	mic.name="jik";
    	mic.test("kkk");
	}
}

11.3 静态内部类

static关键字一般只修饰变量和方法,平常不可以修饰类,但是内部类却可以被static修饰。

  1. static修饰成员变量:整个类的实例共享静态变量
  2. static修饰方法:静态方法,只能够访问用static修饰的属性或方法,而非静态方法可以访问static修饰的方法或属性
  3. 被static修饰了的成员变量和方法能直接被类名调用。
  4. static不能修饰局部变量,切记,不要搞混淆了,static平常就用来修饰成员变量和方法。
package com.haining.opp;

public class StaticInnerClassTest {

    private String name;
    private static int age;

    public void run() {
    }

    public static void go() {
    }

    //外部类访问静态内部类
    public void test() {
        StaticInnerClass sic = new StaticInnerClass(); //静态的内部类不需要依赖外部类,所以不用this
        sic.name = "tom";
        sic.test1("jack");
        StaticInnerClass.age = 10;
        StaticInnerClass.test2("xixi");
    }

    private static class StaticInnerClass {
        private String name;
        private static int age;

        public void test1(String name) {
            System.out.println(name);
            System.out.println(this.name);
            System.out.println(StaticInnerClass.age);
            System.out.println(StaticInnerClassTest.age);
            //System.out.println(StaticInnerClassTest.this.name);静态类不能访问非静态属性
            StaticInnerClassTest.go();
            //StaticInnerClassTest.this.run();静态类不能访问非静态方法
        }

        public static void test2(String name) {
            //只能访问自己和外部类的静态属性和方法
            System.out.println(name);
            //System.out.println(this.name);静态方法里面连自己类的非静态属性都不能访问
            System.out.println(StaticInnerClass.age);
            System.out.println(StaticInnerClassTest.age);
            //System.out.println(StaticInnerClassTest.this.name);静态方法不能访问非静态属性
            StaticInnerClassTest.go();
            //StaticInnerClassTest.this.run();静态方法不能访问非静态方法
        }
    }

}

注意:

  1. 我们上面说的内部类能够调用外部类的方法和属性,在静态内部类中就行了,因为静态内部类没有了指向外部类对象的引用。除非外部类中的方法或者属性也是静态的。这就回归到了static关键字的用法。
  2. 静态内部类能够直接被外部类给实例化,不需要使用外部类对象
Outer.Inner inner = new Outer.Inner();
  1. 静态内部类中可以声明静态方法和静态变量,但是非静态内部类中就不可以声明静态方法和静态变量

11.4 局部内部类

局部内部类是在一个方法内部声明的一个类

局部内部类中可以访问外部类的成员变量及方法

局部内部类中如果要访问该内部类所在方法中的局部变量,那么这个局部变量就必须是final修饰的

public class Outer {
    private int id;
    //在method01方法中有一个Inner内部类,这个内部类就称为局部内部类
    public void method01(){class Inner{
    		public void in(){
    			System.out.println("这是局部内部类");
    		}
    	}
    }
}
  • 局部内部类一般的作用跟在成员内部类中总结的差不多,但是有两个要注意的地方:
  1. 在局部内部类中,如果要访问局部变量,那么该局部变量要用final修饰
    • 为什么需要使用final?
      • final修饰变量:变为常量,会在常量池中放着,逆向思维想这个问题,如果不实用final修饰,当局部内部类被实例化后,方法弹栈,局部变量随着跟着消失,这个时候局部内部类对象在想去调用该局部变量,就会报错,因为该局部变量已经没了,当局部变量用final修饰后,就会将其加入常量池中,即使方法弹栈了,该局部变量还在常量池中呆着,局部内部类也就是够调用。所以局部内部类想要调用局部变量时,需要使用final修饰,不使用,编译度通不过。
public class Outer{
    private int id;
    public void method1(){
        //这个就是局部变量cid。要让局部内部类使用,就得变为final并且赋值,如果不使用final修饰,就会报错
        final int cid = 3;
        class Inner{
            public void in(){
                System.out.println("这是局部内部类");
            }
            //内部类中的使用局部变量cid的方法
            public void useCid(){
                System.out.println(cid);
            }
        }
    }
}
  1. 局部内部类不能通过外部类对象直接实例化,而是在方法中实例化出自己来,然后通过内部类对象调用自己类中的方法。
public class Outer{
    private int id;

    public void out(){
        System.out.println("外部方法");
    }
    public void method1(){
        class Inner{
            public void in(){
                System.out.println("这是局部内部类");
            }
        }
        //关键在这里,如需要在method01方法中自己创建内部类实例,然后调用内部类中的方法,等待外部类调用method01方法,就可以执行到内部类中的方法了。
        Inner In = new Inner();
        In.in();
    }
}
  • 使用局部内部类需要注意的地方就刚才上面说的:
  1. 在局部内部类中,如果要访问局部变量,那么该局部变量要用final修饰
  2. 如何调用局部内部类方法。
public class LocalInnerClassTest {
    private String name;
    private static int age;
    public void run(){}
    public static void go(){}
    //局部内部类要定义在方法中
    public void test(){
        final String myname="";
        class LocalInnerClass{
            private String name;
            // private static int age;不能定义静态属性
            public void test(String name){
                System.out.println(name);
                System.out.println(this.name);
                System.out.println(myname);
                System.out.println(LocalInnerClassTest.this.name);
                LocalInnerClassTest.this.run();
                LocalInnerClassTest.go();
            }
    	}
    	//局部内部类只能在自己的方法中用,因为局部内部类相当于一个局部变量,除了方法就找不到了。
        LocalInnerClass lic = new LocalInnerClass();
        lic.name="tom";
        lic.test("test");
    }
}

11.5 匿名内部类

在这四种内部类中,以后的工作可能遇到最多的是匿名内部类,所以说匿名内部类是最常用的一种内部类。

如果一个对象只要使用一次,那么我们就是需要new Object().method()。 就可以了,而不需要给这个实例保存到该类型变量中去。这就是匿名对象。

public class Test {
    public static void main(String[] args) {
    	//讲new出来的Apple实例赋给apple变量保存起来,但是我们只需要用一次,就可以这样写
    	Apple apple = new Apple();
    	apple.eat();
    	//这种就叫做匿名对象的使用,不把实例保存到变量中。
    	new Apple().eat();
    }
}
class Apple{
    public void eat(){
    	System.out.println("我要被吃了");
    }
}
  • 匿名内部类跟匿名对象是一个道理:

    • 匿名对象:我只需要用一次,那么我就不用声明一个该类型变量来保存对象了,
    • 匿名内部类:我也只需要用一次,那我就不需要在类中先定义一个内部类,而是等待需要用的时候,我就在临时实现这个内部类,因为用次数少,可能就这一次,那么这样写内部类,更方便。不然先写出一个内部类的全部实现来,然后就调用它一次,岂不是用完之后就一直将其放在那,那就没必要那样。
  • 匿名内部类需要依托于其他类或者接口来创建

    • 如果依托的是类,那么创建出来的匿名内部类就默认是这个类的子类
    • 如果依托的是接口,那么创建出来的匿名内部类就默认是这个接口的实现类。
  • 匿名内部类的声明必须是在使用new关键字的时候

    • 匿名内部类的声明及创建对象必须一气呵成,并且之后不能反复使用,因为没有名字。
main:

A a = new A(){
    //实现A中的抽象方法
    //或者重写A中的普通方法
};
注:这个大括号里面其实就是这个内部类的代码,只不过是声明该内部类的同时就是要new创建了其对象,
并且不能反复使用,因为没有名字。


例如:B是一个接口,依托于B接口创建一个匿名内部类对象
B b = new B(){
	//实现B中的抽象方法
};
  1. 匿名内部类除了依托的类或接口之外,不能指定继承或者实现其他类或接口,同时也不能被其他类所继承,因为没有名字。

  2. 匿名内部中,我们不能写出其构造器,因为没有名字。

  3. 匿名内部中,除了重写上面的方法外,一般不会再写其他独有的方法,因为从外部不能直接调用到。(间接是调用到的)

public interface Work{
	void doWork();
}

public class AnonymousOutterClass{
    private String name;
    private static int age;
    public void say(){}
    public static void go(){}
    public void test(){
    	final int i = 90;
    	Work w = new Work(){
    		public void doWork(){
                System.out.println(AnonymousOutterClass.this.name);
                System.out.println(AnonymousOutterClass.age);
                AnonymousOutterClass.this.say();
                AnonymousOutterClass.go();
                System.out.println(i);
            }
        };
   		w.doWork();
	}
}

【不使用匿名内部类】

public class Test {
	public static void main(String[] args) {
	//如果我们需要使用接口中的方法,我们就需要走3步,1、实现接口 2、创建实现接口类的实例对象 3、通过对象调用方法
	//第二步
	Test02 test = new Test02();
    //第三步
    test.method();
    }
}
//接口Test1
interface Test01{
	public void method();
}
//第一步、实现Test01接口
class Test02 implements Test01{
    @Override
    public void method() {
    	System.out.println("实现了Test接口的方法");
    }
}

【使用匿名内部类】

public class Test {
	public static void main(String[] args) {
	//如果我们需要使用接口中的方法,我们只需要走一步,就是使用匿名内部类,直接将其类的对象创建出来。
    	new Test1(){
    		public void method(){
    			System.out.println("实现了Test接口的方法");
    		}
    	}.method();
	}
}
interface Test1{
	public void method();
}

标签:06,Student,void,class,面向对象编程,JavaSE,方法,public,out
From: https://www.cnblogs.com/haining-study/p/17067402.html

相关文章

  • POJ--3069 Saruman's Army(贪心)
    记录0:332023-1-24http://poj.org/problem?id=3069reference:《挑战程序设计竞赛(第2版)》2.2.4p45DescriptionSarumantheWhitemustleadhisarmyalongastra......
  • 力扣---1306. 跳跃游戏 III
    这里有一个非负整数数组 arr,你最开始位于该数组的起始下标 start 处。当你位于下标 i 处时,你可以跳到 i+arr[i]或者i-arr[i]。请你判断自己是否能够跳到对......
  • 06_标识符
    """_*_coding:utf-8_*_@Time:2023/1/1621:52@Author:软柠柠吖@Description:标识符python中,标识符命名的规则主要有3类:①内容限定......
  • 刷刷刷 Day 18 | 106. 从中序与后序遍历序列构造二叉树
    106.从中序与后序遍历序列构造二叉树LeetCode题目要求给定两个整数数组inorder和postorder,其中inorder是二叉树的中序遍历,postorder是同一棵树的后序遍历,请你构......
  • 06. 类
    一、什么是类  使用Object创建对象时,我们无法区分不同类型的对象,也不方便批量创建对象。在JS中,我们可以通过类(class)来解决这个问题。类是对象模板,我们可以将对象中......
  • 代码随想录 | Day3 | LC 203移除链表元素、206反转链表
    [203.移除链表元素classSolution{publicListNoderemoveElements(ListNodehead,intval){......
  • 06 递归
    递归能不用递归就不用递归递归,处理的数目不能太大,否则会造成栈的溢出就执行不了程序packagecom.zhan.base03Method;publicclassTest06{publicstaticvoid......
  • 【动画消消乐】HTML+CSS 自定义加载动画:清新折叠方块效果 063(附源码及原理详解)
    前言Hello!小伙伴!非常感谢您阅读海轰的文章,倘若文中有错误的地方,欢迎您指出~自我介绍ଘ(੭ˊᵕˋ)੭昵称:海轰标签:程序猿|C++选手|学生简介:因C语言结识编程,随后转入计算机专......
  • 【动画消消乐】HTML+CSS 自定义加载动画:怦然心跳 066
    前言Hello!小伙伴!非常感谢您阅读海轰的文章,倘若文中有错误的地方,欢迎您指出~ 自我介绍ଘ(੭ˊᵕˋ)੭昵称:海轰标签:程序猿|C++选手|学生简介:因C语言结识编程,随后转入计算......
  • Day06-字符串操作
    一、字符串的下标(索引)#获取正负索引数据sub_str=str_data[1] #y#[正索引]0开始取索引的格式 下标 获取单个数据print(sub_str)sub_str=str_data[-2] #o......