JAVA-类和对象 (超详细)

目录

一、面向对象的初步认知

1.什么是面向对象

2.面向对象和面向过程的区别

2.1 面向过程

2.2 面向对象 

3.类和对象的区别

二、类和对象

1.类的定义格式

 2.定义一个狗类​编辑

2.1 通过工具改类名

3. 类的实例化

三、this的用法

1.解决命名冲突

 2.调用类中的方法

3.用当前对象

4.返回当前对象

5.构造器中调用其他构造器

6.this引用的特性

四、对象的构造及初始化

1.如何初始化对象 

 2.构造方法

3.特性

 4.默认初始化

5.就地初始化

五、static

1.static修饰成员变量

 2.static修饰成员方法

 3.static成员变量初始化

六、代码块

1.普通代码块

2.构造代码块

3.静态代码块

七、执行顺序

一、面向对象的初步认知

1.什么是面向对象

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

2.面向对象和面向过程的区别

2.1 面向过程

代表语言:C语言、……

面向过程强调的是步骤,每一步做什么。是通过事件触发的。拿洗衣服举列子:

传统的步骤不易更改,而且少了一步都不成,也不易扩展

2.2 面向对象 

代表语言:JAVA、C++、……

面向对象,更多的是对对象之间的交互。是通过对象触发的,不同对象触发的,处理方式也会不一样。想要了解这一现象可以看我的这篇博客:JAVA-多态 和 重写 (彻底给你讲明白)_重写和多态的区别-CSDN博客

3.类和对象的区别

  • 类:用于描述对象的,里面具有属性和行为。
  • 对象:被类实例化人/物品/动物……,是在内存中已经实际存在的

二、类和对象

1.类的定义格式

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

//创建类
class Dog {
  //成员属性或者成员变量
  //行为 或者 成员方法
}

 class 后面接着的是类名 可以根据实际需求取

采用Java语言将狗类在计算机中定义完成,经过javac编译之后形成.class文件,在JVM的基础上计算机就可以识别了。

通过项目文件夹out文件夹点下去就可以找到class文件

 2.定义一个狗类

注意事项:

  • 类名注意采用大驼峰定义
  • public哪里是来控制访问权限的,是封装的内容
  • 一般一个文件当中只定义一个类
  • main方法所在的类一般要使用public修饰(注意:Eclipse默认会在public修饰的类中找main方法)
  • public修饰的类必须要和文件名相同,如果要在一个文件写多个类,那么其他类不用加public

2.1 通过工具改类名

不要轻易去修改public修饰的类的名称,如果要修改,通过开发工具修改

3. 类的实例化

定义了一个类,就相当于在计算机中定义了一种新的类型,与int,double类似,只不过int和double是java语言自 带的内置类型,而类是用户自定义了一个新的类型。

用类类型创建对象的过程,称为类的实例化,在java中采用new关键字,配合类名来实例化对象。

class Dog {
    //成员属性或者成员变量
    public String name;
    public String color;

    //成员方法或者行为
    public void eat() {
        System.out.println(name + "正在吃狗粮");
    }
}

public class main {
    public static void main(String[] args) {
        Dog dog = new Dog();//实例化对象

        //对象的访问
        dog.name = "旺财";
        dog.color = "黄色";
        dog.eat();
    }
}

结果: 

 注意事项

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

 代码解析:

内存分析: 

三、this的用法

1.解决命名冲突

class Date {
    public int year;
    public int month;
    public int day;
    public void setDay(int year, int month, int day) {
        year = year;
        month = month;
        day = day;
    }
    public void printDate(){
        System.out.println(this.year + "/" + this.month + "/" + this.day);
    }
}

public class main {
    public static void main(String[] args) {
        Date date = new Date();
        date.setDay(2024,8,25);
        System.out.println(date.year);
        System.out.println(date.month);
        System.out.println(date.day);
    }
}

 结果:

可以加上 this.成员变量名

 2.调用类中的方法

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;
        
        this.printDate();//调用当前类方法
    }
    public void printDate(){
        System.out.println(this.year + "/" + this.month + "/" + this.day);
    }
}

3.用当前对象

其实pritDate已经用到了,在属性名前加上this.

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

4.返回当前对象

在方法中可以返回当前对象的引用,通常用于实现方法链

class Date {
    
  public Date MyClass() {//返回值是当前类
      return this;
  }
}

单个的this代表当对象,这间接说明了为什么上述两个案列,可以通过this.访问

5.构造器中调用其他构造器

class Student {
    public String name;
    public int age;

    public Student() {
        this("lisi",20);//调用其他构造方法
    }
    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }
}

这里有用到重载的内容,想要了解的可以看这篇博客:JAVA-方法的重载-CSDN博客

注意:

  • 只能在构造方法中调用其他构造方法
  • 在构造方法中调用其他构造方法,this必须在自身构造方法中的第一排

6.this引用的特性

1. this的类型:对应类类型引用,即哪个对象调用就是哪个对象的引用类型

2. this只能在"成员方法"、"构造方法"中使用

3. 在"成员方法"中,this只能引用当前对象,不能再引用其他对象

4.this是“成员方法”第一个隐藏的参数,编译器会自动传递,在成员方法执行时,编译器会负责将调用成员方法 对象的引用传递给该成员方法,this负责来接收

在代码层面来简单演示--->注意:下图右侧中的Date类也是可以通过编译的

这个图比较重要,在用静态方法时会有帮助。

每个成员方法,都会自己加上一个参数 “Data this” 用来接收是哪个对象调用的

四、对象的构造及初始化

1.如何初始化对象 

通过前面知识点的学习知道,在Java方法内部定义一个局部变量时,必须要初始化,否则会编译失败。

public static void main(String[] args) {
    int a;
    System.out.println(a);
}
// Error:(26, 28) java: 可能尚未初始化变量a

 要让上述代码通过编译,非常简单,只需在正式使用a之前,给a设置一个初始值即可。

如果是对象:

public static void main(String[] args) {
    Date d = new Date();
    d.printDate();
    d.setDate(2021,6,9);
    d.printDate();
}
// 代码可以正常通过编译

 2.构造方法

构造方法(也称为构造器)是一个特殊的成员方法,名字必须与类名相同,在创建对象时,由编译器自动调用,并且 在整个对象的生命周期内只调用一次。

public class Date {
    public int year;
    public int month;
    public int day;
// 构造方法:
// 名字与类名相同,没有返回值类型,设置为void也不行
// 一般情况下使用public修饰
// 在创建对象时由编译器自动调用,并且在对象的生命周期内只调用一次
    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(year + "-" + month + "-" + day);
    }
    public static void main(String[] args) {
// 此处创建了一个Date类型的对象,并没有显式调用构造方法
        Date d = new Date(2021,6,9); // 输出Date(int,int,int)方法被调用了
        d.printDate(); 
    }
}

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

3.特性

1.名字必须与类名相同

2. 没有返回值类型,设置为void也不行

3. 创建对象时由编译器自动调用,并且在对象的生命周期内只调用一次(相当于人的出生,每个人只能出生一次)

4. 构造方法可以重载(用户根据自己的需求提供不同参数的构造方法)

上面写过重载的代码:

class Student {
    public String name;
    public int age;

    public Student() {
        this("lisi",20);//调用其他构造方法
    }
    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }
}

5. 如果用户没有显式定义,编译器会生成一份默认的构造方法,生成的默认构造方法一定是无参的。

注意:一旦用户定义如何形式的构造方法,编译器则不会再生成。 

6. 构造方法中,可以通过this调用其他构造方法来简化代码

public class Date {
    public int year;
    public int month;
    public int day;
// 无参构造方法--内部给各个成员赋值初始值,该部分功能与三个参数的构造方法重复
// 此处可以在无参构造方法中通过this调用带有三个参数的构造方法
// 但是this(1900,1,1);必须是构造方法中第一条语句
    public Date(){
        //System.out.println(year); 注释取消掉,编译会失败
        this(1900, 1, 1);
        //this.year = 1900;
        //this.month = 1;
        //this.day = 1;
    }
}

注意:

  • this(...)必须是构造方法中第一条语句
  • 不能形成环
    class Student {
        public String name;
        public int age;
    
        public Student() {
            this("lisi",20);//调用其他构造方法
        }
        public Student(String name, int age) {
            this();
            this.name = name;
            this.age = age;
        }
    }
    /*
    无参构造器调用三个参数的构造器,而三个参数构造器有调用无参的构造器,
    形成构造器的递归调用
    编译报错:Error:(19, 12) java: 递归构造器调用
    */

 7. 绝大多数情况下使用public来修饰,特殊场景下会被private修饰

 4.默认初始化

为什么局部变量在使用时必须要初始化,而成员变量可以不用呢?

要搞清楚这个过程,就需要知道 new 关键字背后所发生的一些事情:

Date d = new Date(2021,6,9);

在程序层面只是简单的一条语句,在JVM层面需要做好多事情,下面简单介绍下:

1. 检测对象对应的类是否加载了,如果没有加载则加载

2. 为对象分配内存空间

3. 处理并发安全问题

比如:多个线程同时申请对象,JVM要保证给对象分配的空间不冲突

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

5. 设置对象头信息

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

5.就地初始化

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

public 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 static void main(String[] args) {
        Date d1 = new Date(2021,6,9);
        Date d2 = new Date();
    }
}

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

五、static

1.static修饰成员变量

tatic修饰的成员变量,称为静态成员变量,静态成员变量最大的特性:不属于某个具体的对象,是所有对象所共享的。

例子:

使用前文中介绍的学生类实例化三个对象s1、s2、s3,每个对象都有自己特有的名字、性别,年龄,学分绩点等成 员信息,这些信息就是对不同学生来进行描述的

public class Student{
// ...
    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修饰的成员,称之为静态成员,也可以称为类成员,其不属于某个具体的对象,是所有对象所共享的

静态成员变量特性

1. 不属于某个具体的对象,是类的属性,所有对象共享的,不存储在某个对象的空间中

2. 既可以通过对象访问,也可以通过类名访问,但一般更推荐使用类名访问

3. 类变量存储在方法区当中

4. 生命周期伴随类的一生(即:随类的加载而创建,随类的卸载而销毁)

public class Student {
    public String name;
    public static String classRoom = "107";

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

    public static void main(String[] args) {
        //无需实例化对象
        System.out.println(Student.classRoom);

        //实例化对象
        Student s1 = new Student("Li leilei");
        Student s2 = new Student("Han MeiMei");
        Student s3 = new Student("Jim");
        //也可以通过对象访问:但是classRoom是三个对象共享的
        System.out.println(s1.classRoom);
        System.out.println(s2.classRoom);
        System.out.println(s3.classRoom);
    }
}

 2.static修饰成员方法

Java中,被static修饰的成员方法称为静态成员方法,是类的方法,不是某个对象所特有的。静态成员一般是通过静态方法来访问的。

public class Student {

    public static String classRoom = "107";

    public static String getClassRoom() {
        return classRoom;
    }

    public static void main(String[] args) {
        System.out.println(Student.getClassRoom());
    }
}

结果:

静态方法特性

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引用

因为,在非静态的成员方法中,第一个参数会传Date this ,这个是用来分便你传递的是哪一个对象的。但是在静态方法中并不需要这个传参,因为是类的方法,并不区分是哪个对象。

5. 静态方法无法重写,不能用来实现多态

 3.static成员变量初始化

注意:静态成员变量一般不会放在构造方法中来初始化,构造方法中初始化的是与对象相关的实例属性

静态成员变量的初始化分为两种:就地初始化 和 静态代码块初始化

1. 就地初始化

就地初始化指的是:在定义时直接给出初始值

public class Student {

    public static String classRoom = "107";

}

2. 静态代码块初始化

一般静态代码块主要用于给,静态成员变量初始化

public class Student {

    public static String classRoom;

    static {
        classRoom = "107";    
    }

}

下面马上讲代码块,会讲到。

六、代码块

使用 { } 定义的一段代码称为代码块。根据代码块定义的位置以及关键字,又可分为以下四种:

  • 普通代码块
  • 构造块
  • 静态块
  • 同步代码块(后续我再来写)

1.普通代码块

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

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

这种用法较少见

2.构造代码块

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

public class Student{
    //实例成员变量
    private String name;
    private String gender;
    private int age;
    private double score;
    public Student() {
        System.out.println("I am Student init()!");
    }
    //实例代码块
    {
        this.name = "bit";
        this.age = 12;
        this.sex = "man";
        System.out.println("I am instance init()!");
    }
    public void show(){
        System.out.println("name: "+name+" age: "+age+" sex: "+sex);
    }
}
public class Main {
    public static void main(String[] args) {
        Student stu = new Student();
        stu.show();
    }
}
// 运行结果
I am instance init()!
I am Student init()!
name: bit age: 12 sex: man

3.静态代码块

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

public class Student{
    private String name;
    private String gender;
    private int age;
    private double score;
    private static String classRoom;
    //实例代码块
    {
        this.name = "bit";
        this.age = 12;
        this.gender = "man";
        System.out.println("I am instance init()!");
    }
    // 静态代码块
    static {
        classRoom = "bit306";
        System.out.println("I am static init()!");
    }
    public Student(){
        System.out.println("I am Student init()!");
    }
    public static void main(String[] args) {
        Student s1 = new Student();
        Student s2 = new Student();
    }
}

注意事项 

  • 静态代码块不管生成多少个对象,其只会执行一次
  • 静态成员变量是类的属性,因此是在JVM加载类时开辟空间并初始化的
  • 如果一个类中包含多个静态代码块,在编译代码时,编译器会按照定义的先后次序依次执行(合并)
  • 实例代码块只有在创建对象时才会执行

 有构造方法、构造代码块、静态代码块,他们执行的顺序是怎么样的呢?

七、执行顺序

测试顺序代码:

public class Test {

    {
        System.out.println("构造代码块");
    }

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

    public Test() {
        System.out.println("构造方法");
    }

    public static void main(String[] args) {
        Test test = new Test();
    }
}

 结果:

那么顺序就是如此。

  • 9
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值