【Java探索之旅】解密构造方法 对象初始化的关键一步

简介: 【Java探索之旅】解密构造方法 对象初始化的关键一步

📑前言

Java编程中,对象的构造和初始化是非常重要的概念。通过构造方法,我们可以为对象的成员变量赋予初始值,并确保对象在创建时处于一个合适的状态。本文将深入探讨构造方法的特性、作用以及与对象初始化相关的重要知识点,帮助您更好地理解和运用Java中对象的构造和初始化过程。  

一、对象的构造及初始化

在Java方法内部定义一个局部变量时,必须要初始化,否则会编译失败。对于对象的成员在默认的情况下是有初始值的,但是如果要给对象中的成员变量赋值,就要调用创建好的方法然后传值。想要给成员变量赋值这样就显得很麻烦,有没有在实例化对象的时候就可以直接对对象赋值呢?答案是有的。

1.1 构造方法

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

class Data{
    public int year;
    public int month;
    public int day;
 
    public Data(int year,int month,int day){
        this.year = year;
        this.month = month;
        this.day = day;
        System.out.println("调用了三个参数的构造方法");
    }
 
    public void setData(Data this,int year,int month,int day) {
        this.year = year;
        this.month = month;
        this.day = day;
        this.printData();
    }
    public void printData(){
        System.out.println("年:"+year+"月:"+month+"日:"+day);
    }
}
    public static void main(String[] args) {
        Data data1 = new Data(1888,9,1);
    }

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

1.2 构造方法的特性

  1. 名字必须与类名相同
  2. 没有返回值类型,设置为void也不行
  3. 创建对象时由编译器自动调用,并且在对象的生命周期内只调用一次(相当于人的出生,每个人只能出生一次)
  4. 构造方法可以重载(用户根据自己的需求提供不同参数的构造方法)
  5. 如果用户没有显式定义,编译器会生成一份默认的构造方法,生成的默认构造方法一定是无参的,一旦用户定义,编译器则不再生成
  6. 构造方法中,可以通过this调用其他构造方法来简化代码,但是this(…)必须是构造方法中第一条语句,并且不能形成环
  7. 绝大多数情况下使用public来修饰,特殊场景下会被private修饰

public Data(int year,int month,int day){
    this.year = year;
    this.month = month;
    this.day = day;
    System.out.println("调用了三个参数的构造方法");
}
public Data() {
    this(2000,8,1);
}
//名字相同,参数列表不同,构成了重载

public Date(){
    this(1900,1,1);
}
public Date(int year, int month, int day) {
    this();
}
//无参构造器调用三个参数的构造器,而三个参数构造器有调用无参的构造器,形成构造器的递归调用,形成了闭环

1.3 默认初始化

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

其实这都和我们的实例化对象new有关,要搞清楚这个过程,就需要知道 new 关键字背后所发生的一些事情。

Data data = new Data(1800,5,8);

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

  1. 检测对象对应的类是否加载了,如果没有加载则加载  
  2. 为对象分配内存空间  
  3. 处理并发安全问题  
  4. 比如:多个线程同时申请对象,JVM要保证给对象分配的空间不冲突  
  5. 初始化所分配的空间  
  6. 即:对象空间被申请好之后,对象中包含的成员已经设置好了初始值  
  7. 设置对象头信息  
  8. 调用构造方法,给对象中各个成员赋值  

据类型

默认值

byte

0

char

‘\u0000’

short

0

int

0

long

0L

boolean

false

float

0.0f

double

0.0

reference

null

1.4 就地初始化

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

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();
    }
}

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

🌤️全篇总结

看到这里,您已经了解了构造方法在Java中的重要性和作用。构造方法是对象初始化的关键步骤,负责为对象的成员变量赋初值,确保对象在创建时具备所需的属性。同时,我们也探讨了构造方法的特性、重载以及默认初始化的相关内容,希望这些知识能帮助您更好地理解和应用对象的构造和初始化过程。

33dcadccda9a4327a9541c4fd95c09fd.png

目录
相关文章
|
17天前
|
安全 Java 编译器
Java对象一定分配在堆上吗?
本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。
Java对象一定分配在堆上吗?
|
21天前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
42 17
|
13天前
|
Java 程序员 数据库连接
Java执行顺序大揭秘:静态块、非静态块和构造方法谁先谁后?
本文详细介绍了Java中的初始化块,包括静态初始化块和非静态初始化块的概念、执行顺序和实际应用场景。通过具体示例,帮助读者理解这两种初始化块的区别和使用场景,让面试官对你刮目相看。
25 0
Java执行顺序大揭秘:静态块、非静态块和构造方法谁先谁后?
|
20天前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
18天前
|
Java
Java 静态变量的初始化顺序
【10月更文挑战第15天】了解 Java 静态变量的初始化顺序对于正确编写和维护代码至关重要。通过深入理解初始化顺序的原理和细节,我们可以更好地避免潜在的问题,并提高代码的质量和可靠性。
|
29天前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第7天】Java零基础教学篇,手把手实践教学!
24 6
|
20天前
|
存储 缓存 NoSQL
一篇搞懂!Java对象序列化与反序列化的底层逻辑
本文介绍了Java中的序列化与反序列化,包括基本概念、应用场景、实现方式及注意事项。序列化是将对象转换为字节流,便于存储和传输;反序列化则是将字节流还原为对象。文中详细讲解了实现序列化的步骤,以及常见的反序列化失败原因和最佳实践。通过实例和代码示例,帮助读者更好地理解和应用这一重要技术。
18 0
java202303java学习笔记第四十天反射获取构造方法1
java202303java学习笔记第四十天反射获取构造方法1
58 0
|
存储 Java Spring
java笔记之反射机制(方法&构造方法)
如何获取Class 1.Class中静态方法 forName("类全名") 2.类.class关键字 3.对象引用.getClass()方法 Class中的常用方法 int = getModifiers(); 获取类的修饰符(权限+特征) String = getName(); 获取类的全名(包名.类名) String = getSimpleName(); 获取类简单名(只有类名 缺少包) Class = getSuperClass(); 获取当前类父类的对应Class Class[] = getInterfaces(); 获取当前类父亲接口 Package
117 0