【Java中创建对象的方式有哪些?】

在这里插入图片描述

✅Java中创建对象的方式有哪些?

  • ✅使用New关键字
  • ✅使用反射机制
  • ✅使用clone方法
  • ✅使用反序列化
  • ✅使用方法句柄
  • ✅ 使用Unsafe分配内存

✅使用New关键字


这是我们最常见的也是最简单的创建对象的方式,通过这种方式我们还可以调用任意的构造函数 (无参的和有参的) 。


User user = new User();


✅使用反射机制


运用反射手段,调用Java.lang.Class或者Java.lang.reflect.Constructor类的newlnstance()实例方法。

1.使用Class类的newlnstance()方法


可以使用Class类的newlnstance方法创建对象。这个newlnstance方法调用无参的构造函数创建对象。


User user = (User)Class.forlame("xxx.xxx.User").newInstance();
User user = User.class .newInstance();

2.使用Constructor类的newlnstance方法


和Class类的newInstance方法很像,java.lang.reflect.Constructor类里也有一个newinstance()方法可以创建对象。我们可以通过这个newlnstance()方法调用有参数的和私有的构造函数。


Constructor constructor = User.class.getConstructor();
User user = constructor.newInstance();

这两种newInstance()方法就是大家所说的反射。事实上Class的newlnstance()方法内部调用Constructor的newlnstance()方法。


✅使用clone方法


无论何时我们调用一个对象的clone方法,jvm就会创建一个新的对象,将前面对象的内容全部拷贝进去。用clone方法创建对象并不会调用任何构造函数。


要使用clone方法,我们需要先实现Cloneable接口并实现其定义的clone方法。如果只实现了Cloneable接口,并没有重写clone方法的话,会默认使用Obiect类中的clone方法,这是一个native的方法。


public class CloneTest implements Cloneable {
	private String name;
	private int age;
	public String getName() {
		return name;
	}
	public void setName(string name)  {
		this .name = name;
	}
	public int getAge() {
		return age;
	}
	public void setAge(int age) {
		this .age = age;
	}
	public CloneTest(String name, int age) {
		super();
		this .name = name ;
		this .age = age;
	}
	public static void main(Stringl] args) {
		try {
			CloneTest cloneTest = new CloneTest("wangql",18);
			CloneTest copyClone = (CloneTest) cloneTest.clone();
			System.out.println("newclone:"+cloneTest.getName());
			System.out.println("copyClone:"+copyClone.getName());
			
		} catch (CloneNotSupportedException e) {
			e.printstackTrace();
		}
	}
}

✅使用反序列化


当我们序列化和反序列化一个对象,jvm会给我们创建一个单独的对象。其实反序列化也是基于反射实现的。


public static void main(Stringl] args) {
	//Initializes The Object
	User1 user = new User1();
	user.setName("Java");
	user.setAge(23);
	System.out.println(user);


	//Write Obj to File
	ObjectOutputStream oos = null;
	try {
		oos = new ObjectOutputStream(new FileOutputStream("tempFile"));
		oos.writeObject(user);
	} catch (IOException e) {
		e.printStackTrace();
	} finally {
		IOUtils.closeQuietly(oos);
	}


	//Read Obj from File
	File file = new File("tempFile");
	ObjectInputStream ois = null;
	try {
		ois = new ObjectInputStream(new FileInputStream(file));
		User1 newUser = (User1) ois.readObject();
		System.out.println(newUser);
	} catch (IOException e)  {
		e.printstackTrace();
	} catch (ClassNotFoundException e)  {
		e.printstackTrace();
	}  finally  {
		IOUtils.closeQuietly(ois);
		try {
			FileUtils.forceDelete(file);
		} catch (IOException e) {
			e.printstackTrace();
		}
	}
}

✅使用方法句柄


通过使用方法句柄,可以间接地调用构造函数来创建对象


public static void main(String [] args) throws Throwable  {
	//定义构造函数的方法句柄类型为void类型,无参数
	MethodType constructorType = MethodType.methodType(void.class);


	//获取构造函数的方法句柄
	MethodHandles.Lookup lookup = MethodHandles.lookup();
	MethodHandle constructorHandle = lookup.findConstructor(User,class, constructorType);


	// 使用方法句柄调用构造函数创建对象
	User obj = (User) constructorHandle.invoke();
}

使用了MethodHandles.lookup().findConstructor()方法获取构造函数的方法句柄,然后通过invoke()方法调用构造函数来创建对象。


✅ 使用Unsafe分配内存


在Java中,可以使用sun.misc.Unsate类来进行直接的内存提作,包括内存分配和对象实例化。然而,需要注意的是,sun.misc.Unsafe类是Java的内部API,它并不属于Java标准库的一部分,也不建议直接在生产环境中使用。


public static void main(String[] args) throws Exception {
	
	Field field = Unsafe.class.getDeclaredField("theUnsafe");
	field.setAccessible(true);
	Unsafe unsafe = field.get(null);
	
	//获取User类的字段偏移量
	long nameOffset = unsafe.objectFieldOffset(User.class.getDeclaredField("name"));
	long ageOffset = unsafe.objectField0ffset(User.class.getDeclaredField("age"));
	
	//使用allocateInstance方法创建对象,不会调用构造函数
	User user = (User) unsafe.allocateInstance(User.class);

	// 使用putObject方法设置字段的值
	unsafe.putObject(user, nameOffset,"Java");
	unsafe.putInt(user, ageOffset, 30);
}

这种方式有以下几个缺点:


1 . 不可移植性:Unsafe类的行为在不同的Java版本和不同的JVM实现中可能会有差异,因此代码在不同的环境可能会出现不可移植的问题。
2 . 安全性问题:Unsafe类的功能是非常强大和危险的,可以绕过Java的安全机制,可能会导致内存泄漏、非法访问、数据损坏等安全问题。
3 . 不符合面向对象的原则:Java是一门面向对象的语言,鼓励使用构造函数和工厂方法来创建对象,以确保对象的正确初始化和维护对象的不变性。

你可能感兴趣的:(Java基础,java,开发语言)