JAVA语言-类和对象详解(2万字超详解!!!)

目录

1.面向对象的初步认识

1.1什么是面向对象

1.2 面向对象与面向过程

2. 类定义和使用

2.1 简单认识类

2.2 类的定义格式

2.3 一些例子

2.3.1 定义一个狗类

2.3.2 定义一个学生类

3. 类的实例化

3.1 什么是实例化

3.2 类和对象的说明

4. this引用

4.1 为什么要有this引用

4.2 什么是this引用

4.3 this引用的特性

5. 对象的构造及初始化

5.1 如何初始化对象

5.2 构造方法

5.2.1 概念

5.2.2 特性

5.3 默认初始化

5.4 就地初始化

6. 封装

6.1 封装的概念

6.2 访问限定符

6.3 封装扩展之包

6.3.1 包的概念

6.3.2 导入包中的类

6.3.3 自定义包

6.3.4 常见的包

7. static成员

7.1 再谈学生类

7.2 static修饰成员变量

7.3 static修饰成员方法

7.4 static成员变量初始化

8. 代码块

8.1 代码块概念以及分类

8.2 普通代码块

8.3 构造代码块

8.4 静态代码块

9. 内部类

9.1 内部类的分类

9.2 成员内部类

9.2.1 静态内部类

9.2.2 实例内部类

9.3 局部内部类

9.4 匿名内部类

10. 对象的打印



1.面向对象的初步认识

1.1什么是面向对象

Java 是一门纯面向对象的语言 (Object Oriented Program ,简称 OOP) ,在面向对象的世界里,一切皆为对象。 向对象是解决问题的一种思想,主要依靠对象之间的交互完成一件事情 。用面向对象的思想来涉及程序,更符合人 们对事物的认知,对于大型程序的设计、扩展以及维护都非常友好。

1.2 面向对象与面向过程

面向过程编程:

    public static void main(String[] args) {
        int[] arr=new int[]{1,2,3,4,5,6,7,8,9,10};
        System.out.print("[");
        for (int i = 0; i < arr.length; i++) {
            System.out.print(arr[i]);
            if(i<arr.length-1){
                System.out.print(", ");
            }
        }
        System.out.println("]");
    }

就如同上面的示例代码:面向过程编程需要完整地把每个步骤写出来以实现某种功能。

而对于面向对象编程来说:

    public static void main(String[] args) {
        int[] arr=new int[]{1,2,3,4,5,6,7,8,9,10};
        System.out.println(Arrays.toString(arr));
    }

例如上面这段代码,总共有两个对象:一个 是数组,一个是库函数toString

我只需要将数组传入toString函数中,它就会自动实现将数组转为字符串的功能

至于这个函数是如何实现的,我们作为使用者则不需要关注。

结论:在面向对象过程中,我只需要关注对象与对象之间的交互,并不需要知道它的功能是如何实现的

相较于面向过程,使用面向对象思想,我只需要调用库函数,仅一行代码就能实现相同的功能。

因此可以看出相较于面对过程编程,面对对象编程能够大幅提高写代码的效率。

2. 类定义和使用

面相对象程序设计关注的是对象,而对象是现实生活中的实体,比如:洗衣机。但是洗衣机计算机并不认识,需要开发人员告诉给计算机什么是洗衣机。

上图左侧就是 对洗衣机简单的描述,该过程称为对洗衣机对象 ( 实体 ) 进行抽象 ( 对一个复杂事物的重新认知 ) ,但是这些简化的抽象结果计算机也不能识别,开发人员可以采用某种面相对象的编程语言来进行描述,比如:Java 语言。

2.1 简单认识类

类是用来对一个实体 ( 对象 ) 来进行描述的 ,主要描述该实体 ( 对象 ) 具有哪些属性 ( 外观尺寸等 ) ,哪些功能 ( 用来干啥) ,描述完成后计算机就可以识别了。
比如:洗衣机,它是一个品牌,在 Java 中可以将其看成是一个类别。
属性:产品品牌,型号,产品重量,外观尺寸,颜色 ...
功能:洗衣,烘干、定时 ....

那么在Java语言中,如何对上述的洗衣机类来进行定义呢?

2.2 类的定义格式

java中定义类时需要用到class关键字,具体语法如下

// 创建类
class ClassName {
fifield ; // 字段 ( 属性 ) 或者 成员变量
method ; // 行为 或者 成员方法
}

其中,class定义类的关键字,ClassName为类的名字,{}中为类的主体。

类中包含的内容称为类的成员。
属性主要是用来描述类的称之为类的 成员属性 或者类 成员变量
方法主要说明类具有哪些功能,称为类的 成员方法
对于上面的洗衣机例子,在JAVA中我们可以这样描述:
class WashMachine {
public String brand ; // 品牌
public String type ; // 型号
public double weight ; // 重量
public double length ; //
public double width ; //
public double height ; //
public String color ; // 颜色
public void washClothes (){ // 洗衣服
  System . out . println ( " 洗衣功能 " );
 }
public void dryClothes (){ // 脱水
  System . out . println ( " 脱水功能 " );
 }
public void setTime (){ // 定时
  System . out . println ( " 定时功能 " );
 }
}
采用 Java 语言将洗衣机类在计算机中定义完成,经过 javac 编译之后形成 .class 文件,在 JVM 的基础上计算机就可以识别了。
注意:类名要使用大驼峰定义

2.3 一些例子

2.3.1 定义一个狗类

class PetDog {
public String name ; // 名字
public String color ; // 颜色
// 狗的属性
public void barks () {
System . out . println ( name + ": 旺旺旺 ~~~" );
}
// 狗的行为
public void wag () {
System . out . println ( name + ": 摇尾巴 ~~~" );
}
}

2.3.2 定义一个学生类

public class Student {
public String name ;
public String gender ;
public short age ;
public double score ;
public void DoClass (){}
public void DoHomework (){}
public void Exam (){}
}

【注意事项】
1. 一般一个文件当中只定义一个类
2. main 方法所在的类一般要使用 public 修饰 ( 注意: Eclipse 默认会在 public 修饰的类中找 main 方法 )
3. public 修饰的类必须要和文件名相同
4. 不要轻易去修改 public 修饰的类的名称,如果要修改,通过开发工具修改(下面是修改的演示)

右键类名->Refactor->Rename

单击Rename后会弹出这样一个窗口:

在框选出来的位置进行重命名,然后点Refactor按钮确认。

确认之后,文件名和类名会自动更改成新名字。

3. 类的实例化

3.1 什么是实例化

定义了一个类,就相当于在计算机中定义了一种新的类型 ,与 int double 类似,只不过 int double java 语言自带的内置类型,而类是用户自定义了一个新的类型,比如上述的:PetDog 类和 Student 类。它们都是类 ( 一种新定义的类型) 有了这些自定义的类型之后,就可以使用这些类来定义实例 ( 或者称为对象 )
用类类型创建对象的过程,称为类的实例化 ,在 java 中采用 new 关键字,配合类名来实例化对象。

例如:

class PetDog{
    public String name;
    public String color;

    public void barks(){
        System.out.println(name+":汪汪汪");
    }

    public void wag(){
        System.out.println(name+":正在摇尾巴~");
    }
}
public class Main {
public static void main ( String [] args ) {
PetDog dogh = new PetDog (); // 通过 new 实例化对象
dogh . name = " 阿黄 " ;
dogh . color = " 黑黄 " ;
dogh . barks ();
dogh . wag ();
PetDog dogs = new PetDog (); // 通过 new 实例化对象
dogs . name = " 阿黄 " ;
dogs . color = " 黑黄 " ;
dogs . barks ();
dogs . wag ();
}
}

输出:

【注意事项】
new 关键字用于创建一个对象的实例 .
使用 . 来访问对象中的属性和方法 .
同一个类可以创建多个实例 .

3.2 类和对象的说明

1. 类只是一个模型一样的东西,用来对一个实体进行描述,限定了类有哪些成员.

2. 类是一种自定义的类型,可以用来定义变量.

3. 一个类可以实例化出多个对象, 实例化出的对象 占用实际的物理空间,存储类成员变量
4. 做个比方。 类实例化出对象就像现实中使用建筑设计图建造出房子,类就像是设计图 ,只设计出需要什么东西,但是并没有实体的建筑存在,同样类也只是一个设计,实例化出的对象才能实际存储数据,占用物理空间

4. this引用

4.1 为什么要有this引用

先来看一个例子:

public class Date {
public int year ;
public int month ;
public int day ;
public void setDay ( int y , int m , int d ){
year = y ;
month = m ;
day = d ;
}
public void printDate (){
System . out . println ( year + "/" + month + "/" + day );
}
public static void main ( String [] args ) {
// 构造三个日期类型的对象 d1 d2 d3
Date d1 = new Date ();
Date d2 = new Date ();
Date d3 = new Date ();
// d1 d2 d3 的日期设置
d1 . setDay ( 2020 , 9 , 15 );
d2 . setDay ( 2020 , 9 , 16 );
d3 . setDay ( 2020 , 9 , 17 );
// 打印日期中的内容
d1 . printDate ();
d2 . printDate ();
d3 . printDate ();
}
}

以上代码定义了一个日期类,然后 main 方法中创建了三个对象,并通过 Date 类中的成员方法对对象进行设置和打印,代码整体逻辑非常简单,没有任何问题。
但是细思之下有以下两个疑问
1. 形参名不小心与成员变量名相同
public void setDay ( int year , int month , int day ){
year = year ;
month = month ;
day = day ;
}
运行结果会是什么样呢?
进行日期设置后,所有的日期都变成了0/0/0,怎么回事?
原因:局部变量的优先级大于成员变量。
因此,在setDay方法中,实现的是局部变量=局部变量。并没有改变成员变量。
所以打印成员变量的时候,成员变量为初始值0。
那么这种情况要怎么办呢?
2. 三个对象都在调用 setDate printDate 函数,但是这两个函数中没有任何有关对象的说明, setDate printDate 函数如何知道打印的是那个对象的数据呢
一切让this引用来揭开这层神秘的面纱。

4.2 什么是this引用

this 引用指向当前对象 ( 成员方法运行时调用该成员方法的对象 ) 在成员方法中所有成员变量的操作,都是通过该引用去访问 。只不过所有的操作对用户是透明的,即用户不需要来传递, 编译器自动完成
class Date {
    public int year;
    public int month;
    public int day;
    public void setDay(int year, int month, int day){
        this.year = year;
        this.month = month;
        this.day = day;
    }
    public void printDate(){
        System.out.println(this.year + "/" + this.month + "/" + this.day);
    }
}

public class Test {
    public static void main(String[] args) {
        Date d1 = new Date();
        Date d2 = new Date();
        Date d3 = new Date();

        d1.setDay(2020,9,15);
        d2.setDay(2020,9,16);
        d3.setDay(2020,9,17);

        d1.printDate();
        d2.printDate();
        d3.printDate();
    }

更改代码:在所有成员变量前加一个 this.  看看结果会是如何?

注意:this引用的是调用成员方法的对象

4.3 this引用的特性

1. this 的类型:对应类类型引用, 即哪个对象调用就是哪个对象的引用类型
2. this 只能在 "成员方法" 中使用
3. 在"成员方法"中,this只能引用当前对象,不能再引用其他对象
4. this成员方法第一个隐藏的参数,编译器会自动传递,在成员方法执行时,编译器会负责将调用成员方法对象的引用传递给该成员方法,this负责来接收

对于第④点,我用如下代码来证明:

我在所有成员方法的第一个位置添加了参数:Date this

而添加了参数的代码并没有报任何错误。这说明了Date this参数是本来就存在的,只是被隐藏了。        

5. 对象的构造及初始化

5.1 如何初始化对象

我们一般是怎么给对象初始化值的:

    public static void main(String[] args) {
        Date d = new Date();
        d.printDate();
        d.setDate(2021,6,9);
        d.printDate();
    }
运行结果:
需要调用之前写的 SetDate 方法才可以将具体的日期设置到对象中。 通过上述例子发现两个问题:
1. 每次对象创建好后调用 SetDate 方法设置具体日期,比较麻烦,那对象该如何初始化?
2. 局部变量必须要初始化才能使用,为什么字段声明之后没有给值依然可以使用?(为什么它会自动初始化为0?)
这些问题会在学习了 构造方法 后得到解决。

5.2 构造方法

5.2.1 概念

构造方法 ( 也称为构造器 ) 是一个特殊的成员方法, 名字必须与类名相同,在创建对象时,由编译器 自动调用 ,并且 在整个对象的生命周期内只调用一次
构造方法:
名字与类名相同, 没有返回值类型 ,设置为 void 也不行
一般情况下使用 public 修饰
创建对象时由编译器自动调用 ,并且在对象的生命周期内只调用一次
class Date {
    public int year;
    public int month;
    public int day;

    public Date(int year, int month, int day){
        this.year = year;
        this.month = month;
        this.day = day;
        System.out.println("Date(int,int,int)方法被调用了");
    }
    public void printDate(){
        System.out.println(this.year + "/" + this.month + "/" + this.day);
    }
}

public class Test {
    public static void main(String[] args) {
        // 此处创建了一个Date类型的对象,并没有显式调用构造方法
        Date d = new Date(2021,6,9); // 输出Date(int,int,int)方法被调用了
        d.printDate(); // 2021-6-9
    }

注意:构造方法的作用就是对对象中的成员进行初始化,并不负责给对象开辟空间。

5.2.2 特性

1. 名字必须与类名相同
2. 没有返回值类型,设置为 void 也不行
3. 创建对象时由编译器自动调用,并且在对象的生命周期内只调用一次 ( 相当于人的出生,每个人只能出生一次 )
4. 构造方法可以重载(用户根据自己的需求提供不同参数的构造方法)

class Date {
    public int year;
    public int month;
    public int day; // 无参构造方法
    public Date(){
        this.year = 1900;
        this.month = 1;
        this.day = 1;
    }
    // 带有三个参数的构造方法
    public Date(int year, int month, int day) {
        this.year = year;
        this.month = month;
        this.day = day;
    }
    public void printDate(){
        System.out.println(year + "-" + month + "-" + day);
    }
}

public class Test {
    public static void main(String[] args) {
        Date d1 = new Date();
        d1.printDate();
        Date d2= new Date(2024,5,18);
        d2.printDate();
    }

上述两个构造方法:名字相同,参数列表不同,因此构成了方法重载
5. 如果用户没有显式定义,编译器会生成一份默认的构造方法,生成的默认构造方法一定是无参的。
class Date {
    public int year;
    public int month;
    public int day; 

    public void printDate(){
        System.out.println(year + "-" + month + "-" + day);
    }
}

public class Test {
    public static void main(String[] args) {
        Date d1 = new Date();
        d1.printDate();
    }

上述Date类中,没有定义任何构造方法,编译器会默认生成一个不带参数的构造方法。

注意:一旦用户定义了构造方法,编译器则不再默认生成。
class Date {
    public int year;
    public int month;
    public int day;
    //定义了一个带参数的构造方法
    public Date(int year, int month, int day) { 
        this.year = year; 
        this.month = month; 
        this.day = day; 
    }
    public void printDate(){
        System.out.println(year + "-" + month + "-" + day);
    }
}
public class Test {
    public static void main(String[] args) {
        Date d1 = new Date();
        d1.printDate();
    }

编译器生成的构造函数必然是无参的,当我们实例化对象时调用无参的构造函数,编译器却报错了

这说明编译器并没有生成一个无参的构造函数,原因是我们已经在类中定义了一个构造函数了

6. 构造方法中,可以通过this调用其他构造方法来简化代码
例如:
class Date {
    public int year;
    public int month;
    public int day;
    public Date(){
        //此处在无参的构造函数中调用了带有三个参数的构造函数
        this(1900,1,1);
    }
    public Date(int year, int month, int day) {
        this.year = year;
        this.month = month;
        this.day = day;
    }
    public void printDate(){
        System.out.println(year + "-" + month + "-" + day);
    }
}
public class Test {
    public static void main(String[] args) {
        Date d1 = new Date();//调用无参构造函数
        d1.printDate();
    }

注意:
        ①this(...)必须是构造方法中第一条语句
        ②不能成环
public Date (){
this ( 1900 , 1 , 1 );
}
public Date ( int year , int month , int day ) {
this ();
}
/*
无参构造器调用三个参数的构造器,而三个参数构造器有调用无参的构造器,形成构造器的递归调用
编译报错: Error:(19, 12) java: 递归构造器调用
*/
7. 绝大多数情况下使用 public 来修饰,特殊场景下会被 private 修饰 ( 后序讲单例模式时会遇到 )

5.3 默认初始化

在上文中提出的第二个问题:为什么局部变量在使用时必须要初始化,而成员变量可以不用呢?

class Date {
    public int year;
    public int month;
    public int day;
    public Date(int year, int month, int day) {
        //此处并没有给成员变量赋值,而是直接打印成员变量
        System.out.print(year+" ");
        System.out.print(month+" ");
        System.out.print(day+" ");
    }
    public void printDate(){
        System.out.println(year + "-" + month + "-" + day);
    }
}
public class Test {
    public static void main(String[] args) {
        Date d1 = new Date(2024,5,18);
    }

 

要搞清楚这个过程,就需要知道 new 关键字背后所发生的一些事情:
Date d = new Date ( 2021 , 6 , 9 );
在程序层面只是简单的一条语句,在 JVM 层面需要做好多事情,下面简单介绍下:
        1. 检测对象对应的类是否加载了,如果没有加载则加载
        2. 为对象分配内存空间
        3. 处理并发安全问题
                比如:多个线程同时申请对象,JVM要保证给对象分配的空间不冲突
        4. 初始化所分配的空间

              即:对象空间被申请好之后,对象中包含的成员已经设置好了初始值,比如:

                数据类型                        默认值
                byte                                                    0
                char                                '\u0000'
                short                                0
                int                                0
               long                                0L
                boolean                                false
                float                                0.0f
               double                                          0.0
               reference                                null
        5. 设置对象头信息 ( 关于对象内存模型后面会介绍 )

        6. 调用构造方法,给对象中各个成员赋值

5.4 就地初始化

在声明成员变量时,就直接给出了初始值。

class Date {
    public int year=1900;
    public int month=1;
    public int day=1;

    public Date(){

    }
    public Date(int year, int month, int day) {
    }
    public void printDate(){
        System.out.println(year + "-" + month + "-" + day);
    }
}
public class Test {
    public static void main(String[] args) {
        Date d1 = new Date(2024,5,18);
        d1.printDate();
        Date d2=new Date();
        d2.printDate();
    }

注意:代码编译完成后,编译器会将所有给成员初始化的这些语句添加到各个构造函数中

6. 封装

6.1 封装的概念

面向对象程序三大特性:封装、继承、多态 。而类和对象阶段,主要研究的就是封装特性。何为封装呢?简单来说就是套壳屏蔽细节
比如:对于电脑这样一个复杂的设备,提供给用户的就只是:开关机、通过键盘输入,显示器, USB 插孔等,让用户来和计算机进行交互,完成日常事务。但实际上:电脑真正工作的却是CPU 、显卡、内存等一些硬件元件。

对于计算机使用者而言,不用关心内部核心部件,比如主板上线路是如何布局的, CPU 内部是如何设计的等,用户只需要知道,怎么开机、怎么通过键盘和鼠标与计算机进行交互即可。因此计算机厂商在出厂时,在外部套上壳 子,将内部实现细节隐藏起来,仅仅对外提供开关机、鼠标以及键盘插孔等,让用户可以与计算机进行交互即可
封装:将数据和操作数据的方法进行有机结合,隐藏对象的属性和实现细节,仅对外公开接口来和对象进行交互。

6.2 访问限定符

比如:
public :可以理解为一个人的外貌特征,谁都可以看得到
default: 对于自己家族中 ( 同一个包中 ) 不是什么秘密,对于其他人来说就是隐私了
private :只有自己知道,其他人都不知道
说明
        ①protected主要是用在继承中,继承部分详细介绍
        ②default权限指:什么都不写时的默认权限
        ③访问权限除了可以限定类中成员的可见性,也可以控制类的可见性

6.3 封装扩展之包

6.3.1 包的概念

在面向对象体系中,提出了一个软件包的概念,即: 为了更好的管理类,把多个类收集在一起成为一组,称为软件 。有点类似于目录。比如:为了更好的管理电脑中的歌曲,一种好的方式就是将相同属性的歌曲放在相同文件下,也可以对某个文件夹下的音乐进行更详细的分类。
Java 中也引入了包, 包是对类、接口等的封装机制的体现,是一种对类或者接口等的很好的组织方式 ,比如:一个包中的类不想被其他包中的类使用。包还有一个重要的作用:在同一个工程中允许存在相同名称的类,只要处在 不同的包中即可

6.3.2 导入包中的类

Java 中已经提供了很多现成的类供我们使用 . 例如 Date 类:可以使用 java.util.Date 导入 java.util 这个包中的 Date类.
public class Test {
public static void main ( String [] args ) {
java . util . Date date = new java . util . Date ();
// 得到一个毫秒级别的时间戳
System . out . println ( date . getTime ());
}
}

但是这种写法比较麻烦一些, 可以使用 import语句导入包.

import java.util.Date;
public class Test {
public static void main ( String [] args ) {
Date date = new Date ();
// 得到一个毫秒级别的时间戳
System . out . println ( date . getTime ());
}
}
如果需要使用 java.util 中的其他类 , 可以使用 import java.util.*
        
import java . util . * ;
public class Test {
public static void main ( String [] args ) {
Date date = new Date ();
// 得到一个毫秒级别的时间戳
System . out . println ( date . getTime ());
}
}
但是我们 更建议显式的指定要导入的类名 . 否则还是容易出现冲突 的情况 .
import java . util . * ;
import java . sql . * ;
public class Test {
public static void main ( String [] args ) {
// util sql 中都存在一个 Date 这样的类 , 此时就会出现歧义 , 编译出错
Date date = new Date ();
System . out . println ( date . getTime ());
}
}
// 编译出错
Error :( 5 , 9 ) java : Date 的引用不明确
java . sql 中的类 java . sql . Date java . util 中的类 java . util . Date 都匹配
在这种情况下需要使用完整的类名
import java . util . * ;
import java . sql . * ;
public class Test {
public static void main ( String [] args ) {
java . util . Date date = new java . util . Date ();
System . out . println ( date . getTime ());
}
}
可以使用 import static 导入包中静态的方法和字段。
import static java . lang . Math . * ;
public class Test {
public static void main ( String [] args ) {
double x = 30 ;
double y = 40 ;
// 静态导入的方式写起来更方便一些 .
// double result = Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2));
double result = sqrt ( pow ( x , 2 ) + pow ( y , 2 ));
System . out . println ( result );
}
}
注意事项 : import C++ #include 差别很大 . C++ 必须 #include 来引入其他文件内容 , 但是 Java 不需要 .
import 只是为了写代码的时候更方便 . import 更类似于 C++ namespace using

6.3.3 自定义包

基本规则
        在文件的最上方加上一个 package 语句指定该代码在哪个包中 .
        包名需要尽量指定成唯一的名字, 通常会用公司的域名的颠倒形式 ( 例如 com.xxx.demo1 ).
        包名要和代码路径相匹配. 例如创建 com.xxx.demo1 的包, 那么会存在一个对应的路径                      com/xxx/demo1 来存储代码.
        如果一个类没有 package 语句 , 则该类被放到一个默认包中 .

操作步骤
        1. 在 IDEA 中先新建一个包 : 右键 src -> 新建 ->

        2. 在弹出的对话框中输入包名, 例如 com.xxx.demo1

        3. 在包中创建类 , 右键包名 -> 新建 -> , 然后输入类名即可 .
         4. 此时可以看到我们的磁盘上的目录结构已经被 IDEA 自动创建出来了
         5. 同时我们也看到了 , 在新创建的 Test.java 文件的最上方 , 就出现了一个 package 语句
         

6.3.4 常见的包

1. java.lang: 系统常用基础类 (String Object), 此包从 JDK1.1 后自动导入。
2. java.lang.reflflect:java 反射编程包 ;
3. java.net: 进行网络编程开发包。
4. java.sql: 进行数据库开发的支持包。
5. java.util: java 提供的工具程序包。 ( 集合类等 ) 非常重要
6. java.io:I/O 编程开发包。

7. static成员

7.1 再谈学生类

这是一个学生类(在 2.3.2 处初次定义):

public class Student{
public String name;
public String gender;
public short age;
public double score;
public void DoClass(){}
public void DoHomework(){}
public void Exam(){}
public Student(String name, String gender, int age, double score) {
    this.name = name;
    this.gender = gender;
    this.age = age;
    this.score = score;
}
public static void main(String[] args) {
Student s1 = new Student("Li leilei", "男", 18, 3.8);
Student s2 = new Student("Han MeiMei", "女", 19, 4.0);
Student s3 = new Student("Jim", "男", 18, 2.6);
}

}

假设三个同学是同一个班的,那么他们上课肯定是在同一个教室,那既然在同一个教室,那能否给类中再加一个成员变量,来保存同学上课时的教室呢?答案是不行的。
之前在 Student 类中定义的成员变量,每个对象中都会包含一份 ( 称之为实例变量 ) ,因为需要使用这些信息来描述具体的学生。而现在要表示学生上课的教室,这个教室的属性并不需要每个学生对象中都存储一份,而是需要让所有的学生来共享。在Java 中,被 static 修饰的成员,称之为静态成员,也可以称为类成员,其不属于某个具体的对 象,是所有对象所共享的

7.2 static修饰成员变量

static 修饰的成员变量,称为静态成员变量 ,静态成员变量最大的特性: 不属于某个具体的对象,是所有对象所共 享的
【静态成员变量特性】
        1. 不属于某个具体的对象,是类的属性,所有对象共享的,不存储在某个对象的空间中
        2. 既可以通过对象访问,也可以通过类名访问,但一般更推荐使用类名访问
        3. 类变量存储在方法区当中
        4. 生命周期伴随类的一生 ( 即:随类的加载而创建,随类的卸载而销毁 )
public class Student {
    public String name;
    public String gender;
    public int age;
    public double score;
    public static String classRoom = "Bit306";

    public Student(String name, String gender, int age, double score) {
        this.name = name;
        this.gender = gender;
        this.age = age;
        this.score = score;
    }

    public static void main(String[] args) {
        Student s1 = new Student("Li leilei", "男", 18, 3.8);
        Student s2 = new Student("Han MeiMei", "女", 19, 4.0);
        Student s3 = new Student("Jim", "男", 18, 2.6);

        System.out.println(s1.classRoom);
        System.out.println(s2.classRoom);
        System.out.println(s3.classRoom);
    }
}

调试上述代码,然后在监视窗口中可以看到,静态成员变量并没有存储到某个具体的对象中。

7.3 static修饰成员方法

一般类中的数据成员都设置为 private ,而成员方法设置为 public ,那设置之后, Student 类中 classRoom 属性如何在类外访问呢?
public class Student {
private String name ;
private String gender ;
private int age ;
private double score ;
private static String classRoom = "Bit306" ;
// ...
}
Public class TestStudent {
public static void main ( String [] args ) {
System . out . println ( Student . classRoom );
}
}
编译失败:
Error :( 10 , 35 ) java : classRoom extend01 . Student 中是 private 访问控制
static 属性应该如何访问呢?
Java 中, static 修饰的成员方法称为静态成员方法,是类的方法,不是某个对象所特有的 。静态成员一般是通过静态方法来访问的。
public class Student {
// ...
private static String classRoom = "Bit306" ;
// ...
public static String getClassRoom (){
return classRoom ;
}
}
public class TestStudent {
public static void main ( String [] args ) {
System . out . println ( Student . getClassRoom ());
}
}
输出:Bit306
静态方法特性
        1. 不属于某个具体的对象,是类方法
        2. 可以通过对象调用,也可以通过类名 . 静态方法名 (...) 方式调用,更推荐使用后者
        3. 不能在静态方法中访问任何非静态成员变量
public static String getClassRoom (){
System . out . println ( this );
return classRoom ;
}
// 编译失败: Error:(35, 28) java: 无法从静态上下文中引用非静态 变量 this
public static String getClassRoom (){
age += 1 ;
return classRoom ;
}
// 编译失败: Error:(35, 9) java: 无法从静态上下文中引用非静态 变量 age

4. 静态方法中不能调用任何非静态方法,因为非静态方法有this参数,在静态方法中调用时候无法传递this引用

public static String getClassRoom (){
doClass ();
return classRoom ;
}
// 编译报错: Error:(35, 9) java: 无法从静态上下文中引用非静态 方法 doClass()

7.4 static成员变量初始化

注意:静态成员变量一般不会放在构造方法中来初始化,构造方法中初始化的是与对象相关的实例
          属性
静态成员变量的初始化分为两种:就地初始化 和 静态代码块初始化
1. 就地初始化
就地初始化指的是:在定义时直接给出初始值
public class Student {
private String name ;
private String gender ;
private int age ;
private double score ;
private static String classRoom = "Bit306" ; //定义时直接给出初始值
// ...
}
2. 静态代码块初始化
直接移步目录看 8.4 静态代码块

8. 代码块

8.1 代码块概念以及分类

使用 {} 定义的一段代码称为代码块 。根据代码块定义的位置以及关键字,又可分为以下四种:
         普通代码块
         构造块
         静态块
         ④ 同步代码块(后续讲解多线程部分再谈)

8.2 普通代码块

普通代码块:定义在方法中的代码块

例如:

public class Main {
public static void main ( String [] args ) {
{ // 直接使用 {} 定义,普通方法块
int x = 10 ;
System . out . println ( "x1 = " + x );
}
int x = 100 ;
System . out . println ( "x2 = " + x );
}
}
// 执行结果
x1 = 10
x2 = 100

观察代码和结果可以看出:普通代码块有规划局部变量作用域的用处功能。

而且这种用法比较少见。

8.3 构造代码块

构造块:

定义在类中的代码块(不加修饰符)。也叫:实例代码块构造代码块一般用于初始化实例成员变量

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

    public Student(){
        System.out.println("构造函数");
    }

    {
        this.name="bit";
        this.gender="male";
        this.age=18;
        this.score=100.0;
        System.out.println("实例代码块");
    }

    public void show(){
        System.out.println(name+" "+gender+" "+age+" "+score);
    }

    public static void main(String[] args) {
        Student stu1=new Student();
        stu1.show();
        //Student stu2=new Student();
    }
}

类似于构造函数,每次创建新对象时都会调用一次实例代码块。

8.4 静态代码块

使用static定义的代码块称为静态代码块。一般用于初始化静态成员变量。

public class Student {
    private String name;
    private String gender;
    private int age;
    private double score;
    private static String classRoom;
    public Student(){
        System.out.println("构造函数");
    }

    {
        this.name="bit";
        this.gender="male";
        this.age=18;
        this.score=100.0;
        System.out.println("实例代码块");
    }

    static{
        classRoom="Room306";
        System.out.println("静态代码块");
    }

    public void show(){
        System.out.println(name+" "+gender+" "+age+" "+score+" "+classRoom);
    }

    public static void main(String[] args) {
        Student stu1=new Student();
        stu1.show();
        //Student stu2=new Student();
    }
}
【注意事项】
        静态代码块不管生成多少个对象,其只会执行一次
        静态成员变量是类的属性,因此是在JVM 加载类时开辟空间并初始化的
        如果一个类中包含多个静态代码块,在编译代码时,编译器会按照定义的先后次序依次
           执行(合并 )
        ④实 例代码块只有在创建对象时才会执行

9. 内部类

当一个事物的内部,还有一个部分需要一个完整的结构进行描述,而这个内部的完整的结构又只为外部事物提供服务,那么这个内部的完整结构最好使用内部类。在 Java 中, 可以将一个类定义在另一个类或者一个方法的内部, 前者称为内部类,后者称为外部类 。内部类也是封装的一种体现。

例如:

public class OutClass {
class InnerClass {
    }
}
// OutClass 是外部类
// InnerClass 是内部类
【注意事项】
1. 定义在 class 类名 {} 花括号外部的,即使是在一个文件里,都不能称为内部类
2. 内部类和外部类共用同一个 java 源文件,但是经过编译之后,内部类会形成单独的字节码文件

9.1 内部类的分类

内部类分为四种:

成员内部类:静态内部类        实例内部类

局部内部类       

匿名内部类

9.2 成员内部类

在外部类中,内部类定义位置与外部类成员所处的位置相同,因此称为成员内部类。

9.2.1 静态内部类

静态内部类: 即被static修饰的内部类

静态内部类要怎么定义?要如何使用?让我们直接看下面这个例子...

public class OutClass {
    public int data1;
    private int data2;

    //静态内部类 被static修饰
    static class InnerClass{
        public int data3;
        private int data4;
        public void func(){
            
            System.out.println("hahaha");
        }
    }

    public static void main(String[] args) {
        InnerClass innerClass=new InnerClass();//静态内部类对象的创建
        innerClass.func();//静态内部类的成员访问
    }
}

运行结果:

注意事项
1. 在静态内部类中 只能访问外部类中的静态成员
如果确实想访问非静态成员,我们该如何做?
2. 创建静态内部类对象时,不需要先创建外部类对象

9.2.2 实例内部类

通过类比静态内部类,可以知道:实例内部类其实就是未被static修饰的内部类。

public class OutClass {
    public int data1=1;
    public static int data2=2;

    //实例内部类
    class InnerClass{
        public int data3=3;
        private int data4=4;
        public void func(){
            System.out.println("hahaha");
        }
    }
    public static void main(String[] args) {
        // 要访问实例内部类中成员,必须要创建实例内部类的对象 
        // 而普通内部类定义与外部类成员定义位置相同,因此创建实例内部类对象时必须借助外部类

        //写法 1:直接创建实例内部类对象
        OutClass.InnerClass innerClass1=new OutClass().new InnerClass();
        innerClass1.func();//访问实例内部类成员
        
        System.out.println("==============");//分隔符        

        //写法 2:先创建外部类对象 再创建实例内部类对象
        OutClass outClass=new OutClass();
        OutClass.InnerClass innerClass2= outClass.new InnerClass();
        innerClass2.func();//访问实例内部类成员
    }
}
注意事项
1. 外部类中的任何成员都可以在实例内部类方法中直接访问
2. 实例内部类所处的位置与外部类成员位置相同,因此也受publicprivate等访问限定符的约束
3. 在实例内部类方法中访问同名的成员时,优先访问自己的。
运行结果:
如果要访问外部类同名的成员, 必须:外部类名称.this.同名成员来访问
运行结果:
4. 实例内部类对象必须在先有外部类对象前提下才能创建
5. 实例内部类的非静态方法中包含了一个指向外部类对象的引用
6. 外部类中,不能直接访问实例内部类中的成员,如果要访问必须先要创建内部类的对象。

9.3 局部内部类

局部内部类:定义在方法体内部的类就叫做局部内部类。

public class OutClass {
    int a=10;
    public void func(){
        int b=20;
        // 局部内部类:定义在方法体内部
        //局部内部类不能被static public等关键字修饰
        class InnerClass{
            public void funcInnerClass(){
                System.out.println(a);
                System.out.println(b);
            }
        }
        // 只能在该方法体内部使用,其他位置都不能用
        InnerClass innerClass=new InnerClass();
        innerClass.funcInnerClass();
    }
    public static void main(String[] args) {
        OutClass outClass=new OutClass();
        outClass.func();
    }
}
运行结果:
注意事项
1. 局部内部类只能在所定义的方法体内部使用
2. 不能被 public static 等修饰符修饰
3. 编译器也有自己独立的字节码文件,命名格式:外部类名字 $ 数字内部类名字 .class
4. 几乎不会使用

9.4 匿名内部类

假设有一个类

public class OutClass {
    public void func(){
        System.out.println("hahaha");
    }
}

那么 new OutClass(); 就是一个匿名对象

如果在后面加上一个花括号{} 就变成了匿名内部类,即:new OutClass(){   };

下面是一个简单的匿名内部类例子:

public class OutClass {
    public void func(){
        System.out.println("hahaha");
    }
    public static void main(String[] args) {
        new OutClass(){

        }.func();
    }
}

运行结果:

如果在匿名对象内重写外部类方法,就会发生动态绑定。

public class OutClass {
    public void func(){
        System.out.println("hahaha");
    }
    public static void main(String[] args) {
        new OutClass(){
            @Override
            public void func() {
                System.out.println("heiheihei");
            }
        }.func();
    }
}

10. 对象的打印

当我们每次需要打印对象的成员变量时,都要调用类函数来实现。

那么有没有更简便的方法呢,比如使用println来打印?不妨试试:

public class Person {
String name ;
String gender ;
int age ;
public Person ( String name , String gender , int age ) {
this . name = name ;
this . gender = gender ;
this . age = age ;
}
public static void main ( String [] args ) {
Person person = new Person ( "Jim" , " " , 18 );
System . out . println ( person );
}
}
打印结果:

可以看到,打印出来的并不是类里面的成员变量。

如果想要默认打印对象中的属性该如何处理呢?答案:重写toString方法即可。

public class Person {
String name ;
String gender ;
int age ;
public Person ( String name , String gender , int age ) {
this . name = name ;
this . gender = gender ;
this . age = age ;
}
@Override
public String toString () {
return "[" + name + "," + gender + "," + age + "]" ;
}
public static void main ( String [] args ) {
Person person = new Person ( "Jim" , " " , 18 );
System . out . println ( person );
}
}
// 输出结果:         

当我们在类里面重写了一个toString函数之后,使用println打印类就变成了打印其成员变量。

这是什么原因呢?

这是因为:不重写toString,当我们使用toString方法时,系统会调用默认的toString方法。

                如果我们在类里面重写了一个toString方法,那么系统则会自动调用重写的toString方法


看完了觉得有用的姥爷们求求您们点个免费的赞和关注,这对我真的很重要!非常感谢大家!

  • 33
    点赞
  • 28
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值