一文瞭解 Java 中的構造器

来源:https://www.cnblogs.com/huaweiyun/archive/2022/09/14/16693709.html
-Advertisement-
Play Games

摘要:Java 也採用了構造器,並且還提供了一個垃圾收集器(garbage collector),當不再使用記憶體資源的時候,垃圾收集器會自動將其釋放。 本文分享自華為雲社區《一文帶你瞭解 Java 中的構造器》,作者: 宇宙之一粟 。 C ++ 引入了構造器(constructor,也叫構造函數)的 ...


摘要:Java 也採用了構造器,並且還提供了一個垃圾收集器(garbage collector),當不再使用記憶體資源的時候,垃圾收集器會自動將其釋放。

本文分享自華為雲社區《一文帶你瞭解 Java 中的構造器》,作者: 宇宙之一粟 。

C ++ 引入了構造器(constructor,也叫構造函數)的概念,它是在創建對象時被自動調用的特殊方法

Java 也採用了構造器,並且還提供了一個垃圾收集器(garbage collector),當不再使用記憶體資源的時候,垃圾收集器會自動將其釋放。

構造器定義

在 Java 中,可以通過編寫構造器來確保每個對象的初始化。但是這裡有兩個問題:

  1. 這個構造器使用的任何名字都有可能與類里某個成員相衝突;
  2. 編譯器負責調用構造器,所以它必須始終知道應該調用哪個方法。

C++ 語言採用的方案就是將構造器和類的名字定義相同,Java 也採用了這個方案。

構造器的作用是用來建立一個新的類的實例,當一個對象被創建時,JVM 使用一個構造函數,併為其分配記憶體空間。

語法結構

class ClassName {
 ClassName() {
 }
}

例如,在下麵的示例中,我們創建了一個名為 ReLearnConstructor 的構造函數。在構造函數內部,我們正在初始化 hello 變數的值。:

public class ReLearnConstructor {

String hello; // 屬性
 // 構造器
public ReLearnConstructor() {
hello = "Hello, Constructor!";
}

public static void main(String[] args) {

ReLearnConstructor rc = new ReLearnConstructor();
System.out.println(rc.hello);
}
}

註意創建 ReLearnConstructor 類的對象的語句:ReLearnConstructor rc = new ReLearnConstructor();

在這裡,當創建對象時,調用 ReLearnConstructor 構造函數。並且,hello 變數的值被初始化。

因此列印的 hello 的值為:

構造器目的

構造函數的目的是初始化對象的狀態,為所有聲明的屬性賦值。如果我們沒有自定義構造函數,JVM 就會為這些屬性分配預設值。

原始類型的預設值:

  • 整數類型是 0
  • 浮點類型是 0.0
  • 布爾類型是 false

對於其他 Java 引用類型,預設值是null,這意味著引用類型的屬性沒有被分配任何值。

後面可以用代碼查看這些預設值。

構造器分類

在 Java 中,有三種類型的構造器:

  1. 無參構造器
  2. 有參構造器
  3. 預設構造器

無參構造器

與方法類似,Java 構造函數可能有參數,也可能沒有任何參數。如果構造函數不接受任何參數,則稱為無參數構造器。例如上述代碼中 ReLearnConstructor 構造器就是:

// 無參構造器
public ReLearnConstructor() {
hello = "Hello, Constructor!";
}

有參構造器

字面理解,具有參數的構造函數稱為有參數構造器。那為什麼需要使用有參構造器?

有參構造器可用於為不同對象提供不同初始化的值。 例如:

public class ReLearnConstructor {

String languages;

// 接受單個參數的構造器
public ReLearnConstructor(String lang) {
languages = lang;
System.out.println("我在學習 " + languages + " 語言!");
}

public static void main(String[] args) {
// 向構造器中傳入不同的值
ReLearnConstructor rc1 = new ReLearnConstructor("Java");
ReLearnConstructor rc2 = new ReLearnConstructor("Go");
ReLearnConstructor rc3 = new ReLearnConstructor("Python");
}
}

運行結果:

預設構造器

如果我們不創建任何構造函數,Java 編譯器會在程式執行期間自動創建一個無參數構造函數。這個構造函數稱為預設構造函數。來看一個例子;

public class ReLearnConstructor {

String languages;
int a;
boolean b;
float c;
public static void main(String[] args) {
ReLearnConstructor rc = new ReLearnConstructor();
System.out.println("預設值:");
System.out.println("languages:" + rc.languages);
System.out.println("a:" + rc.a);
System.out.println("b:" + rc.b);
System.out.println("c:" + rc.c);
}
}

運行結果:

預設值:
languages:null
a:0
b:false
c:0.0

可以看到,我們還沒有創建任何構造函數。因此,Java 編譯器會自動創建預設構造函數。上述表格得以印證。

原生方法和構造器的區別

  • 構造函數必須與在 Java 中定義的類具有相同的名稱
  • 當方法沒有返回任何值時,構造函數不會返回任何類型,而方法則具有返回類型或 void
  • 在對象創建時,僅調用構造函數一次,而方法可以被調用任何次數

如果我們不用構造器來給屬性賦值的話,可以先使用 new 運算符獲取類的實例,並使用類的 setter 方法設置值,如下:

import java.util.Arrays;
class Person
{
 private String name;
 private int age;
    @Override
 public String toString() {
 return Arrays.asList(name, String.valueOf(age)).toString();
 }
 public void setName(String name) {
 this.name = name;
 }
 public void setAge(int age) {
 this.age = age;
 }
 // getters
}
// Initialize an object in Java
class Main
{
 public static void main(String[] args)
 {
        Person person = new Person();
 person.setName("Yuzhou1su");
 person.setAge(22);
 System.out.println(person);
 }
}

通過構造器進行初始化就可以省去我們的 setter 方法。

如下的例子:

import java.util.Arrays;
class Person {
 private String name;
 private int age;
 // 構造器
 public Person(String name, int age) {
 this.name = name;
 this.age = age;
 }
 public String toString() {
 return Arrays.asList(name, String.valueOf(age)).toString();
 }
}
class SimpleConstructor {
 public static void main(String[] args) {
        Person person = new Person("Yuzhou1su", 22);
 System.out.println(person);
 }
}

運行結果:

[Yuzhou1su, 22]

構造器重載

與 Java 方法重載類似,我們也可以創建兩個或多個具有不同參數的構造函數。這稱為構造函數重載。

public class ReLearnConstructor {

String language;

public ReLearnConstructor() {
this.language = "Java";
}

// 構造器
public ReLearnConstructor(String language) {
this.language = language;
}

public void getName() {
System.out.println("編程語言:" + this.language);
}

public static void main(String[] args) {
ReLearnConstructor rc1 = new ReLearnConstructor();

ReLearnConstructor rc2 = new ReLearnConstructor("Python");

rc1.getName();
rc2.getName();
}
}

在上面的例子中,我們有兩個構造函數:ReLearnConstructor() 和 ReLearnConstructor(String language)。在這裡,兩個構造函數都用不同的值初始化變數語言的值。根據創建對象時傳遞的參數,調用不同的構造函數,分配不同的值。

運行結果:

編程語言:Java
編程語言:Python

拷貝構造器

Java 中的拷貝構造方法是一種使用該類的一個對象構造另外一個對象的構造方法。

複製構造函數是一種特殊構造函數,用於將新對象創建為現有對象的副本。它只需要一個參數,它將是同一類的另一個實例。我們可以使用 this() 語句從複製構造函數中顯式調用另一個構造函數:

public class ReLearnConstructor {

private String language;

// 構造器
public ReLearnConstructor(String language) {
this.language = language;
}

// 拷貝構造器
public ReLearnConstructor(ReLearnConstructor rc) {
this.language = rc.language;
}

public void getName() {
System.out.println("編程語言:" + this.language);
}

public static void main(String[] args) {
ReLearnConstructor rc = new ReLearnConstructor("Python");

ReLearnConstructor copyOfrc = new ReLearnConstructor(rc);

rc.getName();
copyOfrc.getName();
}
}

運行結果:

編程語言:Python
編程語言:Python

當需要拷貝一個帶有多個成員變數的複雜對象或者想構造已存在對象的深拷貝對象時非常有用。

匿名內部類

除了上文介紹的使用構造器的方法,另一種初始化對象的方法是使用“雙大括弧初始化”。這將創建一個匿名內部類,其中只有一個實例初始化程式。建議不要使用這種方法。

import java.util.Arrays;
class Person
{
 private String name;
 private int age;
    @Override
 public String toString() {
 return Arrays.asList(name, String.valueOf(age)).toString();
 }
 public void setName(String name) {
 this.name = name;
 }
 public void setAge(int age) {
 this.age = age;
 }
 // getters
}
// Initialize an object in Java
class Main
{
 public static void main(String[] args)
 {
 // Anonymous class
        Person person = new Person() {{
 // Initializer block
 setName("Yuzhou1su");
 setAge(22);
 }};
 System.out.println(person);
 }
}

總結

  • 實例化對象時會隱式調用構造函數。
  • 創建構造函數的兩條規則是:構造函數的名稱應與類相同。Java 構造函數不能有返回類型。
  • 如果一個類沒有構造函數,Java 編譯器會在運行時自動創建一個預設構造函數。預設構造函數使用預設值初始化實例變數。例如 int 變數將被初始化為 0
  • 構造函數類型:
  • 無參構造器 - 不接受任何參數的構造函數參數化構造函數
  • 接受參數的構造器 - 接受參數的構造函數
  • 預設構造器 - 如果沒有明確定義,Java 編譯器會自動創建一個構造函數。
  • 構造函數不能被 abstract、static 或 final 修飾

編譯器會報如下錯誤:

Illegal modifier for the constructor in type ReLearnConstructor; only public, protected & private are permitted
  • 構造函數可以重載但不能被覆蓋

 

點擊關註,第一時間瞭解華為雲新鮮技術~


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、什麼是使用註解開發 使用註解開發就是無需再配置Mapper.xml文件,直接在介面中利用註解實現SQL語句。 二、為什麼要使用註解開發 正如官方文檔所說: 使用註解來映射簡單語句會使代碼顯得更加簡潔。 但對於稍微複雜一點的語句,Java 註解不僅力不從心,還會讓你本就複雜的 SQL 語句更加混亂 ...
  • Java坦克大戰06 8.IO流應用01 坦克大戰6.0版 增加功能: 防止敵人坦克重疊運動 記錄玩家的成績(累計擊毀坦克數),存檔退出 記錄當時的敵人坦克坐標,存檔退出 玩游戲時,可以選擇是開新游戲還是繼續上局游戲 8.1防止敵人坦克重疊運動 8.1.1思路分析 按照目標坦克的向右下左四種情況分析 ...
  • 第一篇:電腦的基礎知識 編程語言的介紹 電腦介紹和五大組成 平臺與軟體跨平臺介紹 CS、BS架構和網路通信協議 操作系統的介紹 cpu詳解 存儲器詳解 操作系統啟動流程和BIOS介紹 第二篇:python環境的搭建 python介紹和解釋器的安裝(暫略) python程式的運行方式和步驟 集成開 ...
  • 首先,先看一下intern方法(JDK1.8)的官方文檔: 全是英文,閱讀起來有點困難怎麼辦?沒關係,博主對此做了翻譯: 返回字元串對象的規範表示形式。 最初為空的字元串池由類字元串私人維護。 調用intern方法時,如果池中已包含一個字元串,該字元串等於由equals(object)方法確定的該字 ...
  • emmm~ 起因呢,這昨晚女同桌跟我說電腦有點卡,喊我去宿舍給她裝個新系統,裝系統就裝系統吧,結果又說新系統表情包都沒保存~ 我當時就有點生氣,真當我是萬能的呢? 於是我直接就用Python給她爬了幾十個G,完事扭頭就走,任她怎麼喊我也沒用! 一、準備工作 使用的環境 python3.8 | Ana ...
  • 模塊 三種方法: import from 模塊 import 成員,成員 from 模塊 import * *代表所有的成員 隱藏成員: 模塊中以下劃線_開頭的屬性 隱藏成員不會被from 模塊 import * 導入 導入模塊時會將模塊的代碼全部執行 as 取別名 from module01 im ...
  • 兩種方式Docker和Docker Compose部署web項目,相對於Go語言里說,不管是使用docker部署還是直接伺服器部署都相當方便,比python要簡單很多。 1、Dockerfile結構解析 From 我們正在使用基礎鏡像golang:alpine來創建我們的鏡像。這和我們要創建的鏡像一 ...
  • 首先說明一下,這種涉及了在MyBatis(二)中說的那個第二種老方法,所以一般不推薦使用。 上一篇我們利用SQL的limit實現了分頁,是在SQL層面的,那麼這次我們利用java代碼RowBounds來實現。直接上操作。 一、RowBounds實現分頁 1.在UserMapper介面中聲明一個新的方 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...