Java中五種不同方法的創(chuàng)建對(duì)象
前言
作為Java開(kāi)發(fā)者,我們每天都會(huì)創(chuàng)建大量的對(duì)象,但是,我們總是使用管理依賴(lài)系統(tǒng)(如Spring框架)來(lái)創(chuàng)建這些對(duì)象。其實(shí)還有其他方法可以創(chuàng)建對(duì)象,在接下來(lái)的文章中我會(huì)進(jìn)行詳細(xì)介紹。
1.使用new關(guān)鍵字
這是最常見(jiàn)的創(chuàng)建對(duì)象的方法,并且也非常簡(jiǎn)單。通過(guò)使用這種方法我們可以調(diào)用任何我們需要調(diào)用的構(gòu)造函數(shù)。
Employee emp1 = new Employee(); 0: new #19 // class org/programming/mitra/exercises/Employee 3: dup 4: invokespecial #21 // Method org/programming/mitra/exercises/Employee."":()V
2.使用class類(lèi)的newInstance方法
我們也可以使用class類(lèi)的newInstance方法來(lái)創(chuàng)建對(duì)象。此newInstance方法調(diào)用無(wú)參構(gòu)造函數(shù)以創(chuàng)建對(duì)象。
我們可以通過(guò)newInstance() 用以下方式創(chuàng)建對(duì)象:
Employee emp2 = (Employee) Class.forName("org.programming.mitra.exercises.Employee").newInstance();
或者
Employee emp2 = Employee.class.newInstance(); 51: invokevirtual #70 // Method java/lang/Class.newInstance:()Ljava/lang/Object;
3.使用構(gòu)造函數(shù)類(lèi)的 newInstance方法
與使用class類(lèi)的newInstance
方法相似,java.lang.reflect.Constructor
類(lèi)中有一個(gè)可以用來(lái)創(chuàng)建對(duì)象的newInstance()
函數(shù)方法。通過(guò)使用這個(gè)newInstance
方法我們也可以調(diào)用參數(shù)化構(gòu)造函數(shù)和私有構(gòu)造函數(shù)。
Constructor<Employee> constructor = Employee.class.getConstructor(); Employee emp3 = constructor.newInstance(); 111: invokevirtual #80 // Method java/lang/reflect/Constructor.newInstance:([Ljava/lang/Object;)Ljava/lang/Object;
這些 newInstance()
方法被認(rèn)為是創(chuàng)建對(duì)象的反射手段。實(shí)際上,內(nèi)部類(lèi)的newInstance()
方法使用構(gòu)造函數(shù)類(lèi)的 newInstance()
方法。這就是為什么后者是首選并且使用不同的框架如Spring, Hibernate, Struts
等。
4.使用clone方法
實(shí)際上無(wú)論何時(shí)我們調(diào)用clone
方法,JAVA虛擬機(jī)都為我們創(chuàng)建了一個(gè)新的對(duì)象并且復(fù)制了之前對(duì)象的內(nèi)容到這個(gè)新的對(duì)象中。使用 clone
方法創(chuàng)建對(duì)象不會(huì)調(diào)用任何構(gòu)造函數(shù)。
為了在對(duì)象中使用clone()
方法,我們需要在其中實(shí)現(xiàn)可克隆類(lèi)型并定義clone方法。
Employee emp4 = (Employee) emp3.clone(); 162: invokevirtual #87 // Method org/programming/mitra/exercises/Employee.clone ()Ljava/lang/Object;
5.使用反序列化
無(wú)論何時(shí)我們對(duì)一個(gè)對(duì)象進(jìn)行序列化和反序列化,JAVA虛擬機(jī)都會(huì)為我們創(chuàng)建一個(gè)單獨(dú)的對(duì)象。在反序列化中,JAVA虛擬機(jī)不會(huì)使用任何構(gòu)造函數(shù)來(lái)創(chuàng)建對(duì)象。
對(duì)一個(gè)對(duì)象進(jìn)行序列化需要我們?cè)陬?lèi)中實(shí)現(xiàn)可序列化的接口。
ObjectInputStream in = new ObjectInputStream(new FileInputStream("data.obj")); Employee emp5 = (Employee) in.readObject(); 261: invokevirtual #118 // Method java/io/ObjectInputStream.readObject:()Ljava/lang/Object;
正如我們?cè)谝陨系淖止?jié)代碼片段中所看到的,除第一種被轉(zhuǎn)換為一個(gè)新的函數(shù)和一個(gè) invokespecial
指令以外,其它4種方法都被調(diào)用并轉(zhuǎn)換為invokevirtual
。
示例
讓我們來(lái)看看準(zhǔn)備創(chuàng)建對(duì)象的 Employee
類(lèi):
class Employee implements Cloneable, Serializable { private static final long serialVersionUID = 1L; private String name; public Employee() { System.out.println("Employee Constructor Called..."); } public String getName() { return name; } public void setName(String name) { this.name = name; } @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + ((name == null) ? 0 : name.hashCode()); return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; Employee other = (Employee) obj; if (name == null) { if (other.name != null) return false; } else if (!name.equals(other.name)) return false; return true; } @Override public String toString() { return "Employee [name=" + name + "]"; } @Override public Object clone() { Object obj = null; try { obj = super.clone(); } catch (CloneNotSupportedException e) { e.printStackTrace(); } return obj; } }
在下面的Java程序中我們用5種方式來(lái)創(chuàng)建 Employee
對(duì)象。
public class ObjectCreation { public static void main(String... args) throws Exception { // By using new keyword Employee emp1 = new Employee(); emp1.setName("Naresh"); System.out.println(emp1 + ", hashcode : " + emp1.hashCode()); // By using Class class's newInstance() method Employee emp2 = (Employee) Class.forName("org.programming.mitra.exercises.Employee") .newInstance(); // Or we can simply do this // Employee emp2 = Employee.class.newInstance(); emp2.setName("Rishi"); System.out.println(emp2 + ", hashcode : " + emp2.hashCode()); // By using Constructor class's newInstance() method Constructor<Employee> constructor = Employee.class.getConstructor(); Employee emp3 = constructor.newInstance(); emp3.setName("Yogesh"); System.out.println(emp3 + ", hashcode : " + emp3.hashCode()); // By using clone() method Employee emp4 = (Employee) emp3.clone(); emp4.setName("Atul"); System.out.println(emp4 + ", hashcode : " + emp4.hashCode()); // By using Deserialization // Serialization ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("data.obj")); out.writeObject(emp4); out.close(); //Deserialization ObjectInputStream in = new ObjectInputStream(new FileInputStream("data.obj")); Employee emp5 = (Employee) in.readObject(); in.close(); emp5.setName("Akash"); System.out.println(emp5 + ", hashcode : " + emp5.hashCode()); } }
此程序輸出結(jié)果如下:
Employee Constructor Called... Employee [name=Naresh], hashcode : -1968815046 Employee Constructor Called... Employee [name=Rishi], hashcode : 78970652 Employee Constructor Called... Employee [name=Yogesh], hashcode : -1641292792 Employee [name=Atul], hashcode : 2051657 Employee [name=Akash], hashcode : 63313419
以上內(nèi)容是關(guān)于java創(chuàng)建對(duì)象的5種不同方法,希望給大家學(xué)習(xí)java時(shí)有所幫助。也謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
SpringBoot中的WebSocketSession原理詳解
這篇文章主要介紹了SpringBoot中的WebSocketSession原理詳解,傳統(tǒng)的?HTTP?協(xié)議是無(wú)法支持實(shí)時(shí)通信的,因?yàn)樗且环N無(wú)狀態(tài)協(xié)議,每次請(qǐng)求都是獨(dú)立的,無(wú)法保持連接。為了解決這個(gè)問(wèn)題,WebSocket?協(xié)議被引入,需要的朋友可以參考下2023-07-07淺析Java 數(shù)據(jù)結(jié)構(gòu)常用接口與類(lèi)
本篇文章主要介紹了Java中的數(shù)據(jù)結(jié)構(gòu),Java工具包提供了強(qiáng)大的數(shù)據(jù)結(jié)構(gòu)。需要的朋友可以參考下2017-04-04從零開(kāi)始在Centos7上部署SpringBoot項(xiàng)目
本文主要介紹了從零開(kāi)始在Centos7上部署SpringBoot項(xiàng)目,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04SpringAOP+RabbitMQ+WebSocket實(shí)戰(zhàn)詳解
這篇文章主要介紹了SpringAOP+RabbitMQ+WebSocket實(shí)戰(zhàn)詳解,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-11-11Java中final,finally,finalize?有什么區(qū)別
這篇文章主要給大家分享的是?Java中final,finally,finalize?到底有什么區(qū)別,文章圍繞final,finally,finalize的相關(guān)資料展開(kāi)詳細(xì)內(nèi)容,具有一定的參考的價(jià)值,需要的朋友可以參考一下2021-11-11詳解Java如何應(yīng)對(duì)常見(jiàn)的安全威脅和攻擊類(lèi)型
隨著信息技術(shù)的快速發(fā)展,網(wǎng)絡(luò)安全問(wèn)題日益突出,本文將以Java開(kāi)發(fā)語(yǔ)言為例,深入探討網(wǎng)絡(luò)協(xié)議的安全性問(wèn)題,通過(guò)分析常見(jiàn)的安全威脅和攻擊類(lèi)型,設(shè)計(jì)和實(shí)施安全協(xié)議等主題,為讀者提供一些有益的思路和方法,需要的朋友可以參考下2023-11-11淺談SpringSecurity注解與AOP切面執(zhí)行順序
這篇文章主要介紹了淺談SpringSecurity注解與AOP切面執(zhí)行順序,引入Spring Security后,在Controller的方法中會(huì)出現(xiàn)Spring Security的方法注解與AOP同時(shí)存在的問(wèn)題,這是就會(huì)設(shè)計(jì)順序問(wèn)題,需要的朋友可以參考下2023-10-10