生成对象的四种方式
(1)通过new生成对象。这是我们最常用的方式,生成的对象置于内存中的堆空间中,堆空间的构成如下图所示
一个old区,一个eden区,两个survivor区。通常生成的对象会置于Eden区中,但是当生成的对象过大,超过jvm设置的一个值的时候,也会将该对象直接置于old区中。具体的关于创建对象时,jvm对于内存分配以及内存回收的相关知识,这里也就不再累述了。
(2)利用clone复制对象,完成生成对象。
利用clone,在内存中进行数据块的拷贝,复制已有的对象,也是生成对象的一种方式。前提是类实现Cloneable接口,Cloneable接口没有任何方法,是一个空接口,也可以称这样的接口为标志接口,只有实现了该接口,才会支持clone操作。有的人也许会问了,java中的对象都有一个默认的父类Object。Object中有一个clone方法,为什么还必须要实现Cloneable接口呢,这就是cloneable接口这个标志接口的意义,只有实现了这个接口才能实现复制操作,因为jvm在复制对象的时候,会检查对象的类是否实现了Cloneable这个接口,如果没有实现,则会报CloneNotSupportedException异常。类似这样的接口还有Serializable接口、RandomAccess接口等。还有值得一提的是在执行clone操作的时候,不会调用构造函数。还有clone操作还会面临深拷贝和浅拷贝的问题。关于这方面的问题,网上有很多的相关知识了,不再累述了。由于通过复制操作得到对象不需要调用构造函数,只是内存中的数据块的拷贝,那是不是拷贝对象的效率是不是一定会比new的时候的快。答案:不是。显然jvm的开发者也意识到通过new方式来生成对象占据了开发者生成对象的绝大部分,所以对于利用new操作生成对象进行了优化。例如
public class Person implements Cloneable { private String name; private String address; public Person(String name,String address){ this.name=name; this.address=address; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getAddress() { return address; } public void setAddress(String address) { this.address = address; } public Object clone(){ try { return super.clone(); } catch (CloneNotSupportedException e) { throw new Error(); } } public static void main(String[] args) throws Throwable { final int maxLoops=100000; long start=System.nanoTime(); Person person = new Person("iteye", "iteye"); for(int i=0;i<maxLoops;i++){ person.clone(); } long mid=System.nanoTime(); System.out.println("利用clone生成10万个对象花的时间是:"+(mid-start)+"ns"); for(int i=0;i<maxLoops;i++){ new Person("iteye", "iteye"); } System.out.println("利用clone生成10万个对象花的时间是:"+(System.nanoTime()-mid)+"ns"); } }
得到的结果是:
利用clone生成10万个对象花的时间是:16695411ns
利用new生成10万个对象花的时间是:1668425ns
很显然还是new的效率高一些。但是这也并不是说复制的方式没有用。在构建一些复杂对象的时候,或者通过构造函数构造对象比较麻烦的时候,可以考虑这种方式。
(3)通过序列化生成对象
通过序列化方式生成对象是通过在内存中,将对象写入字节流,再从字节流中将其读取出来,这样就可以重建一个新对象,通过这种方式生成的对象与母对象之间不存在引用共享的问题,也就是相当于深拷贝了一个对象。
public class CloneUtils{ public static <T extends Serializable> T clone(T obj){ T cloneObj=null; try{ ByteArrayOutputStream baos= new ByteArrayOutputStream(); ObjectOutputStream oos= new ObjectOutputStream(baos); oos.writeObject(obj); oos.close(); ByteArrayInputStream bais= new ByteArrayInputStream(baos.toByteArray()); ObjectInputStream ois= new ObjectInputStream(bais); cloneObj=(T)ois.readObject(); ois.close(); } catch (Exception e) { e.printStackTrace(); } return cloneObj; } }
以上代码提供了一个可以利用序列化方式生成对象的工具类。当然序列化本身的知识还有很多,读者请自行查阅相关资料。(注:Serializable也是接口,在第二种生成对象的方式中提到过)
(4)可以通过反射,利用class对象生成对象。
相关推荐
比nero更完美的刻录CD软件---CLONE_CD
语音克隆模型
Jlink-clone解决办法,替换文件
matlab开发-MinesweeperClone。用于Matlab的Microsoft Windows XP扫雷器克隆
SpringMVC精品资源--Reddit clone built using Spring Boot, Sprin
(实战)[re:Invent 2018]-001:赛道分析-(致敬1024) 配套 train and evaluation log
matlab开发-MinesweeperClone.zip
JLinkV8V9OB解决j-link-is-defective,j-link clone
grafana-dashboards 20210224 clone
Facebook-login-page-clone-main-源码.rar
3D-clash-royale-clone.zip,一个统一的克隆超级细胞的氏族冲突衍生出移动游戏冲突皇家可能有些曲折。,3D建模使用专门的软件来创建物理对象的数字模型。它是3D计算机图形的一个方面,用于视频游戏,3D打印和VR,以及...
type-racer-clone-new-main
生成被最小化,并且文件名包括哈希值。 您的应用已准备好进行部署! 有关更多信息,请参见关于的部分。 yarn eject 注意:这是单向操作。 eject ,您将无法返回! 如果您对构建工具和配置选择不满意,则可以随时...
twitter-clone, 一种基于 Ruby on Rails的Twitter 开源克隆 twitter克隆使用 Ruby on Rails http://twitter-clone-rails.herokuapp.com/的Twitter的开源克隆。正在启动git clone https://github.co
前端开源库-better-clone更好的克隆,创造价值的克隆
darts-clone-java 用Java编写的Double-ARray Trie System克隆。 该库基于称为“快速高效”库的 。入门设置要使用Maven添加依赖项,请使用以下命令: < dependency> < groupId>...
jquery-clone-tableheader table 头部抬头固定 用于 多数据出现滚动条,看不到标题
git-pull-or-clone 确保磁盘上存在一个git repo,并且它是最新的 安装 npm install git-pull-or-clone 用法 const gitPullOrClone = require ( 'git-pull-or-clone' ) gitPullOrClone ( 'git@github....
matlab 鱼群算法 亲测保证可以运行 鱼群算法 鱼群算法
固件更新可解决 J-link v8 重新烧录解决不识别 usb,灯不亮的问题,同时也可通过更多操作排除 J-link 报 FW Update、Clone 和直接闪退 MDK,详情请阅读附件教程。