Java中创建(实例化)对象的五种方式

Java中创建(实例化)对象的五种方式
1、用new语句创建对象,这是最常见的创建对象的方法。

2、通过工厂方法返回对象,如:String str = String.valueOf(23); 
3、运用反射手段,调用java.lang.Class或者java.lang.reflect.Constructor类的newInstance()实例方法。如:Object obj = Class.forName("java.lang.Object").newInstance(); 
4、调用对象的clone()方法。

5、通过I/O流(包括反序列化),如运用反序列化手段,调用java.io.ObjectInputStream对象的 readObject()方法。

package org.whb.test.demo724;
/*
 *测试Cloneable接口的使用
 *包含第一种和第三种方法clone()
 *不过要注意在clone()中深复制和潜复制的理解
 *实例化对象 
 */
class Person implements Cloneable{
    private String name;
    private int age;
 
  public Person( String name,int age) {
    this.name = name; 
    this.age = age;
  }
 
  public int getAge() {
   return age;
  }
  
  public void setAge(int age) {
   this.age = age;
  }
 
 public String getName() {
  return name;
 }
 
 public void setName(String name){
      this.name =name;
    }
 
@Override
 public Object clone() throws CloneNotSupportedException {
  // TODO Auto-generated method stub
  return super.clone();
 }
  @Override
 public String toString() {
  // TODO Auto-generated method stub
  return "姓名是:"+name+"; 年龄是:"+age;
 }
   
}
public class TestClone{
 public static void main(String[] args){
   Person p1 = new Person("王豪博",25);
   System.out.println(p1);
   Person p2 =null;
   try {
     p2 = (Person)p1.clone();
   } catch (CloneNotSupportedException e) {
    // TODO Auto-generated catch block
     e.printStackTrace();
   }
   p2.setName("春香");
   p2.setAge(24);
   System.out.println(p2);
  }
}

/*
 *通过反射对对象进行初始化
 *注意必须有无参数的Constructor
 *实例化Class类然后调用newInstance()方法
 *
 */
package org.whb.test.demo715;
class Person{
 private int age;
 private String name;
 public int getAge() {
  return age;
 }
 public void setAge(int age) {
  this.age = age;
 }
 public String getName() {
  return name;
 }
 public void setName(String name) {
  this.name = name;
 }
 public String toString(){
  return "年龄是:"+this.age+"  姓名是:"+this.name;
  
 } 
 
}
public class TestClass {
  public static void main(String[] args){
   Class< ?> c1 = null;
   try{
        c1 = Class.forName("org.whb.test.demo715.Person");
   }catch(ClassNotFoundException e){
    e.printStackTrace();
   }   
   Person p1 = null;
   try {
   p1 =(Person)c1.newInstance();
 } catch (InstantiationException e) {
  // TODO Auto-generated catch block
  e.printStackTrace();
 } catch (IllegalAccessException e) {
  // TODO Auto-generated catch block
  e.printStackTrace();
 }
 
 
   p1.setAge(12);
   p1.setName("haobo");
   System.out.println(p1);
  }
}

package org.whb.test.demo724;
import java.io.*; 
import java.util.Date;
/** 
* 对象的序列化和反序列化测试类. 
*1、序列化是干什么的?
简单说就是为了保存在内存中的各种对象的状态(也就是实例变量,不是方法),并且可以把保存的对象状态再读出来。虽然你可以用你自 己的各种各样的方法来保存object states,但是Java给你提供一种应该比你自己好的保存对象状态的机制,那就是序列化。
*2、什么情况下需要序列化 
a)当你想把的内存中的对象状态保存到一个文件中或者数据库中时候;
b)当你想用套接字在网络上传送对象的时候;
c)当你想通过RMI传输对象的时候;
*
*3、相关注意事项
a)序列化时,只对对象的状态进行保存,而不管对象的方法;
b)当一个父类实现序列化,子类自动实现序列化,不需要显式实现Serializable接口;
c)当一个对象的实例变量引用其他对象,序列化该对象时也把引用对象进行序列化;
d)并非所有的对象都可以序列化,,至于为什么不可以,有很多原因了,比如:
1.安全方面的原因,比如一个对象拥有private,public等field,对于一个要传输的对象,比如写到文件,或者进行rmi传输 等等,在序列化进行传输的过程中,这个对象的private等域是不受保护的。
2. 资源分配方面的原因,比如socket,thread类,如果可以序列化,进行传输或者保存,也无法对他们进行重新的资源分 配,而且,也是没有必要这样实现。
*/
public class ObjectSaver { 
public static void main(String[] args) throws Exception { 
 ObjectOutputStream out = new ObjectOutputStream (new FileOutputStream("D:/objectFile.swf")); 
 //序列化对象 
 Customer customer = new Customer("haobo", 24); 
 out.writeObject("你好!"); 
 out.writeObject(new Date()); 
 out.writeObject(customer); 
 out.writeInt(123); //写入基本类型数据 
 out.close(); 
 //反序列化对象 
 ObjectInputStream in = new ObjectInputStream(new FileInputStream("D:/objectFile.swf")); 
 System.out.println("obj1=" + (String) in.readObject()); 
 System.out.println("obj2=" + (Date) in.readObject()); 
 Customer obj3 = (Customer) in.readObject(); 
 System.out.println("obj3=" + obj3); 
 int obj4 = in.readInt(); 
 System.out.println("obj4=" + obj4); 
 in.close(); 
 } 
} 
class Customer implements Serializable { 
 /** 
 * 
 */ 
 private static final long serialVersionUID = -88175599799432325L; 
 private String name; 
 private int age; 
 public Customer(String name, int age) { 
  this.name = name; 
  this.age = age; 
 } 
  public String toString() { 
   return "name=" + name + ", age=" + age; 
  } 
 } 
/*
 * obj1=你好!
obj2=Sat Jul 24 21:18:19 CST 2010
obj3=name=haobo, age=24
obj4=123
 */

  Java中,类的实例化方法有四种途径:

1)使用new操作符

2)调用Class对象的newInstance()方法

3)调用clone()方法,对现有实例的拷贝

4)通过ObjectInputStream的readObject()方法反序列化类

1.ClassInstance.java

复制代码
 1 import java.io.*;
 2 
 3 class ClassInstance implements Cloneable, Serializable {
 4     private String str = "测试...";
 5     public void fun(){
 6         System.out.println(str);
 7     }
 8     public ClassInstance(String str){
 9         System.out.println("有参类的实例化");
10         this.str += str; 
11     }
12     public ClassInstance(){
13         System.out.println("无参类的实例化");
14     }
15     public Object clone(){
16         return this;
17     }
18 }
复制代码

2.ClassInstanceTest.java

复制代码
 1 import java.io.*;
 2 import java.lang.reflect.*;
 3 
 4 public class ClassInstanceTest{
 5     public static void main(String[] args) throws ClassNotFoundException, InstantiationException,
 6            IllegalAccessException, IOException,InvocationTargetException, NoSuchMethodException{
 7         //第一种类的实例化方式
 8         ClassInstance ci01 = new ClassInstance("01");
 9         ci01.fun();
10 
11         //第二种类的实例化方式
12         ClassInstance ci02 = (ClassInstance) Class.forName("ClassInstance").newInstance();
13         ci02.fun();
14 
15         //第三种类的实例化方式
16         ClassInstance ci03 = (ClassInstance) ci01.clone();
17         ci03.fun();
18 
19         //第四种类的实例化方式
20         FileOutputStream fos = new FileOutputStream("ci.tmp");
21         ObjectOutputStream oos = new ObjectOutputStream(fos);
22         oos.writeObject(ci01);
23         oos.close();
24         fos.close();
25         
26         FileInputStream fis = new FileInputStream("ci.tmp");
27         ObjectInputStream ois = new ObjectInputStream(fis);
28         
29         ClassInstance ci04  = (ClassInstance) ois.readObject();
30         ois.close();
31         fis.close();
32         
33         ci04.fun();
34         System.out.println("--------------------额外测试--------------------");
35         ClassInstance ci05 = null;
36         //额外的思考 在第二种类实例化的方式中有没有一种方法实现有参数的构造方式
37         //获得类的构造信息
38         Constructor[] ctor = Class.forName("ClassInstance").getDeclaredConstructors();
39         //找到我们需要的构造方法
40         for(int i=0;i<ctor.length;i++ ){
41             Class[] cl = ctor[i].getParameterTypes();
42             if(cl.length == 1){
43                 //实例化对象
44                 ci05 = (ClassInstance) Class.forName("ClassInstance").getConstructor(cl).newInstance(new Object[]{"05"});
45             }
46         }
47         ci05.fun();
48     }
49 }
复制代码

 3.输出结果

复制代码
1 有参类的实例化
2 测试...01
3 无参类的实例化
4 测试...
5 测试...01
6 测试...01
7 -------------------额外测试--------------------
8 有参类的实例化
9 测试...05
复制代码

 除了这几种情况可以实例化一个Java类对象外,隐式调用实例化也是利用了已上集中情况。例如常见的方法:

复制代码
1 public class ClassInstance{
2   public ClassInstance(){
3   }
4     public ClassInstance getInstance(){
5     return new ClassInstance();
6   }
7 }
复制代码

  

  通过观察结果,我们发现无论哪一种方式必须经过的一步---调用构造方法。无论怎样构造函数作为初始化类的意义怎样都不会改变。


  • 1
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值