【Java SE】类和对象(下)

简介: 从本期开始,欢迎各位正式开始面向对象编程,Java 一门纯面向对象编程的语言,在它的世界里,一切皆为对象,面向对象和面向过程其实都是一种解决问题的思路,而面向对象主要是各个对象之间互相交互去完成一件事情。

4、this 引用的认识

4.1 为什么需要 this 引用

假设这里我们要对前面的 Student 类进行修改,如果跟上面一样一个个初始化姓名年龄分数等太麻了,能不能在类中写一个方法可以用来一次性初始化里面的成员变量呢?当然是可以的:

public class Student {
    //成员变量【属性】
    public String name; //姓名
    public int age; //年龄
    public float score; //分数
    public String subject; //课程
    //成员方法【行为】
    public void studySubject() {
        System.out.println(name + "正在学习" + subject);
    }
    public void setStudentData(String n, int a, float c, String s) {
        n = name;
        a = age;
        c = score;
        s = subject;
    }
    public static void main(String[] args) {
        Student student1 = new Student();
        student1.setStudentData("张三", 20, 59.9f, "Linux");
        Student student2 = new Student();
        student2.setStudentData("李四", 22, 89.8f, "Java");
        student1.studySubject();
        student2.studySubject();
    }
}

但是这里细心的小伙伴可能就发现了,你这个成员方法里面的形参名取得很随意啊,就不能跟成员变量名一样吗?如果我的形参名跟成员变量名一样了,那方法体中,是谁给谁赋值呢?成员变量给成员变量,还是形参给形参?但是在实际中,如果形参和成员变量名一样,则会是局部变量优先,所以也就是形参给自己赋值!

第二个问题,我们 new 了两个对象,student1和student2分别引用了他们,这两个对象都在调用studySubject方法,这两个方法又是如何知道打印的是哪个对象的数据呢?

如果想弄清楚这几个问题,请往后看:

4.2 什么是 this 引用?

this 引用是指向当前对象的!也就是在方法运行的时候调用该成员方法的对象,我们在成员方法中所有对成员变量的操作,本质是通过 this 引用去访问的!不过这些操作编译器帮我们做了,对于我们来说是透明的,也就是用户不需要传递,编译器自己搞定

所以我们上面的 studySubject() 这个成员方法实际上是这样的:

public void studySubject(Student this) {
        System.out.println(this.name + "正在学习" + this.subject);
    }

这个地方,this 引用的是什么呢?其实就是哪个对象调用了这个成员方法,它引用的就是谁!this 代表了当前对象的引用

如果是 student1.studySubject(),那么 this 接收的就是 student1,也就是说 this 在这里是成员方法第一个隐藏参数,编译器会自动传递,在成员方法执行的时候,编译器会负责将调用成员方法对象的引用传递给该成员方法,this 负责来接收,所以这就是为什么在成员方法内部我们能使用 this.成员变量 的操作了!

现在使用 this 来解决我们上面成员方法命名的问题:

public void setStudentData(String name, int age, float score, String subject) {
        this.name = name;
        this.age = age;
        this.score = score;
        this.subject = subject;
    }

那我们在以后写代码在成员方法内部访问成员变量要不要加上 this 呢,虽然编译器会自动加上,但我们最好加上,不仅语义更明确,也可以避免形参与成员变量重名的情况

所以目前我们知道了 this 引用的三个用法:

  • this 访问当前成员变量 this.
  • this 访问成员方法(在成员方法中,嵌套其他成员方法) this.
  • this 访问构造方法 this();

5、对象的构造和初始化

5.1 如何初始化对象

这个相信大家都在前面见到过了,可以使用 引用.成员变量  进行初始化,同时呢也可以像我们上面一样写一个成员方法来初始化,但是像以上两种做法还是比较麻烦的,因为每次都要实例化对象之后才能进行初始化,能不能做到在创建对象的同时给成员变量初始化呢?

我们还知道一个点,变量不初始化是不能使用的,但是如果是对象的话,不初始化里面的成员变量是会有默认值的!这是为什么呢?

5.2 构造方法

构造方法是一种特殊的成员方法,也被称作为构造器,它特殊在哪呢?

首先它的方法名必须与类名相同!没有返回类型!设置成void也不行,一般用public修饰,并且在创建对象的时候,自动调用,而且在整个对象的生命周期中只能调用一次!

那么我们现在就在我们原有的 Student 类中添加一个构造方法:

public class Student {
    //成员变量【属性】
    public String name; //姓名
    public int age; //年龄
    public float score; //分数
    public String subject; //课程
    //构造方法
    public Student(String name, int age, float score, String subject) {
        this.name = name;
        this.age = age;
        this.score = score;
        this.subject = subject;
    }
    //成员方法【行为】
    public void studySubject() {
        System.out.println(this.name + "正在学习" + this.subject);
    }
    public static void main(String[] args) {
        Student student1 = new Student("李四", 22, 89.8f, "Java");
        student1.studySubject();
    }
}

这里我们通过增加了一个带有参数的构造方法,可以看到,在创建对象的时候,就可以调用构造方法初始化了, 看到这,有的小伙伴大吃一惊,原来我们之前创建方法的时候不都是在调用一个无参的构造方法吗?是的!但是我们之气明明没有写构造方法啊,这是怎么回事?

当我们没有任何构造方法的时候,编译器会给我们提供一个不带参数的构造方法!

那我们现在写了一个构造方法,那如果下次再创建对象的时候,不传参呢?不行!因为只有在没有任何构造方法的时候,编译器才会提供,如果有了,我们在创建的时候不需要参数还得自己写一个无参的构造方法!而且构造方法的作用主要是给对象中成员进行初始化的!不负责给成员开辟空间!

上面这么一解释,我们既然可以有多个构造方法,但是构造方法的方法名必须与类名一样,那是不是说明构造方法支持重载!是的!构造方法支持重载!

构造方法中,可以用 this 调用其他的构造方法,但是必须是第一条语句,而且不能成环调用,也就 this 调用的方法里不能还有 this 调用:

public Student(String name, int age, float score, String subject) {
        this(name);
        this.name = name;
        this.age = age;
        this.score = score;
        this.subject = subject;
    }
    public Student(String name) {
        System.out.println(name + "构造成功!");
    }

所以到现在我们就明白了一个点:当构造方法调用完成之后,对象才正真产生了!

5.3 默认初始化

前面说过,没初始化对象中的成员变量是有默认值的,为啥呢?简单说一下,在JVM层面上,我们new一个对象,首先要检查类是否被加载,然后为对象分配空间,接着处理并发问题,为了保证分配的空间不冲突,再就是初始化所分配的空间,也就是初始化成默认值,再然后就是设置对象头信息这个后期讲,接着才是调用构造方法。

那么他们的默认值是什么呢?这个参考我上期文章,里面有一个表格,写的很清楚。引用类型的默认值是 null。

5.4 就地初始化

就地初始化是什么意思呢?就是在我们定义类里面成员变量的时候就直接给初始值了,比如:

public class Student {
    //成员变量【属性】
    public String name = "张三"; //姓名
    public int age = 20; //年龄
    public float score = 60.0f; //分数
    public String subject = "Linux"; //课程
}

这种会在编译完成后,编译器会将所有的成员初始化的这些语句添加到各个构造函数中,每个构造函数中默认会添加这些语句!

相关文章
|
2月前
|
设计模式 网络协议 数据可视化
Java 设计模式之状态模式:让对象的行为随状态优雅变化
状态模式通过封装对象的状态,使行为随状态变化而改变。以订单为例,将待支付、已支付等状态独立成类,消除冗长条件判断,提升代码可维护性与扩展性,适用于状态多、转换复杂的场景。
342 0
|
2月前
|
存储 Java 索引
用Java语言实现一个自定义的ArrayList类
自定义MyArrayList类模拟Java ArrayList核心功能,支持泛型、动态扩容(1.5倍)、增删改查及越界检查,底层用Object数组实现,适合学习动态数组原理。
114 4
|
2月前
|
IDE JavaScript Java
在Java 11中,如何处理被弃用的类或接口?
在Java 11中,如何处理被弃用的类或接口?
206 5
|
2月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
196 1
|
2月前
|
Java Go 开发工具
【Java】(8)正则表达式的使用与常用类分享
正则表达式定义了字符串的模式。正则表达式并不仅限于某一种语言,但是在每种语言中有细微的差别。
246 1
|
2月前
|
存储 Java 程序员
【Java】(6)全方面带你了解Java里的日期与时间内容,介绍 Calendar、GregorianCalendar、Date类
java.util 包提供了 Date 类来封装当前的日期和时间。Date 类提供两个构造函数来实例化 Date 对象。第一个构造函数使用当前日期和时间来初始化对象。Date( )第二个构造函数接收一个参数,该参数是从1970年1月1日起的毫秒数。
199 1
|
2月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
221 1
|
3月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案
Java 数据库 Spring
170 0
|
3月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
276 16