有时候奇怪的发现往list添加数据的时候,一直被最后一个元素覆盖,首先,我们得明白原理:
在new一个对象的时候,对象的ID是唯一确定的;将对象add入list中时,放入list中的其实是对象的引用;而每次循环只是简单的set对象的属性,set新的属性值,而add进list中的对象还是同一个对象id,也就是同一个对象;
所以每次add之后,list发现对象引用和之前元素一样,就覆盖掉了之前add的对象。所以循环之后list中的对象是重复的对象。
想要避免这个问题只要每次add时保证对象引用都是不同的即可,即每循环一次重新new一个对象。
new的对象应该放在for循环内,每循环一次重新new一个新对象
它的确是在main方法栈中先后创建了10个引用叫obj,然后每个obj指向不同的new出来的对象。
关键是:每次循环体执行完后,循环体内定义的代码块局部变量、对象如果没有被继续引用,就立即被销毁了;即obj变量、new出来的对象都被销毁了。
一般是循环一次一个新对象,所以如非必要,不要在循环里面创建对象。
用内存角度来解释的话,在JVM的方法栈和堆内存这两个内存中,当运行Objectobj=newObject();时,在方法栈的栈顶中放入Objectobj,然后在堆中生成一个Object对象,最后方法栈中的obj指向Object对象,这时完成一次循环的内存动作。接着第二次循环,newObject();又产生一个新的Object对象,然后方法栈中的obj指向新的Object对象,第一次循环的Object对象没有被任务变量引用,成为了垃圾,等JVM垃圾回收器回收。其它循环以此类推了。
在这里我提一下题外话,JVM有个优秀的算法是,JVM的方法栈,一个方法的调用产生一个栈帧,这个栈帧在方法调用时计算出你有几个局部变量,这个计算会过滤作用域的,如你的循环后面再建立一个局部变量,那JVM认为只有一个局部变量,因为你的Objectobj=newObject();在大括号内,出了大括号就没有了,所以只需要一个位置来存放局部变量即可,也就是说运行到大括号时用这个位置来执行代码,出了大括号后下一个局部变量就可以接着使用这个栈的这个位置来执行下面的代码了
再看一个例子:
结果:
解释:
在for循环内部创建的话,每次执行循环都会创建对象,没什么特别明显坏处,只是会消耗内存。所以我们通常在for循环外部实例化对象,因为它执行一次
Objectobj是创建对象引用,引用的实例地址。newObject();是创建对象实例
两者都会占用系统资源。
改进之前for循环中创建了10次引用和10实例,改进后是创建了1次引用,10实例。
在方法执行完后内存资源会被回收
以上就是动力节点Java培训机构小编介绍的“Javase高级教程:for循环创建对象”的内容,希望对大家有帮助,如有疑问,请在线咨询,有专业老师随时为你服务。
你适合学Java吗?4大专业测评方法
代码逻辑 吸收能力 技术学习能力 综合素质
先测评确定适合在学习