我最近正在研究释放Java对象占用的内存。这样做的时候,我对如何在Java中复制对象(浅/深)以及如何避免在使用中意外清除/无效化对象感到困惑。
请考虑以下情形:
ArrayList<Object>
作为参数传递给方法。ArrayList<Object>
给要由线程处理的可运行类。ArrayList<Object>
放入HashMap
。现在,在这些情况下,如果我调用list = null;
或list.clear();
,对象将如何处理?在哪种情况下丢失对象,在哪种情况下仅将引用设置为null?
我想这与对象的浅层复制和深层复制有关,但是在哪种情况下会发生浅表复制,而在Java中会在哪种情况下发生深层复制?
首先,您永远不会将 对象 设置为null。这个概念没有任何意义。您可以指定的值null
的 变量
,但你需要非常仔细地“变量”和“对象”的概念区别开来。完成后,您的问题会自动回答:)
现在,就“浅表副本”还是“深表副本”而言,在这里应该避免使用“浅表副本”一词,因为通常浅表副本涉及创建一个新对象,而只是直接复制现有对象的字段。深层副本还将获取这些字段所引用的对象的副本(用于引用类型字段)。这样的简单分配:
ArrayList<String> list1 = new ArrayList<String>();
ArrayList<String> list2 = list1;
......没有做 任何 浅表复制或在这个意义上的深层复制。它只是复制参考。在上面的代码之后,list1
并且list2
是独立变量-
当前它们恰好具有相同的值(引用)。我们可以更改其中一个的值,而不会影响另一个:
list1 = null;
System.out.println(list2.size()); // Just prints 0
现在,如果不更改 变量 ,而是对变量值所引用的对象进行更改,那么该更改也将通过另一个变量可见:
list2.add("Foo");
System.out.println(list1.get(0)); // Prints Foo
回到最初的问题-您从不将 实际对象 存储在地图,列表,数组等中。您仅存储 引用
。仅当不再有“实时”代码到达该对象的方式时,该对象才能被垃圾回收。因此,在这种情况下:
List<String> list = new ArrayList<String>();
Map<String, List<String>> map = new HashMap<String, List<String>>();
map.put("Foo", list);
list = null;
…该ArrayList
对象仍然不能被垃圾回收,因为的对象Map
有一个引用它的条目。
null 在这种情况下,如果我调用或,对象会发生什么?在哪种情况下,对象丢失了,在哪种情况下,只有引用被设置为NULL? 我想这与对象的浅层和深层复制有关,但在Java中,在哪些情况下浅层复制发生,在哪些情况下深层复制发生?
问题内容: 我用Java 创建了一个对象。当我这样做时,它显示如下:。结果,我的Excel文件似乎减少了一天(27 feb变为26 feb,依此类推),我认为这一定是因为时间的缘故。如何将其设置为下午5:30? 问题答案: 另请参阅 乔达时间 日历文件
我有两个函数,递增速度和递减速度。它们都使用clearInterval()然后使用setInterval,但是有些地方不正确。 降低速度将我的速度从1000降到5000,但是如果我使用该功能,速度将增加10-20倍,看起来更像是100-500毫秒。另外,如果我使用increaseSpeed将速度提高到50毫秒,然后使用decreaseSpeed,decreaseSpeed将不会有任何效果。 这是设
考虑类家长: 和类儿童: 如果我在应用程序中创建一个新的子对象,并将其放入父对象的children字段中,然后持久化父对象,为什么Hibernate不自动更新子对象的Parent字段(以便设置parentID列)?
如果我这样做会发生什么: 它是从内存中删除对象,还是只清除引用? 更正式地说,考虑守则: 为什么仍然不是?它不应该被从记忆中删除吗?
问题内容: 我有两个线程, 如果jvm首先执行thread1并将obj1设置为null,那么thread2会立即看到该更改,还是会花费一些时间,并且由于obj1还不为null,jvm仍可以运行thread2同步块吗? 问题答案: 这几乎肯定会破坏同步抽象- 我不敢相信会立即看到更改。您永远都不应更改要同步的对象的引用,更不要将其设置为,这将导致任何进一步尝试对其进行同步的尝试。