- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
好吧,让我看看是否可以解释一下。
我有一些代码将 Java 迭代器(碰巧来自 Hadoop)包装在 Scala Stream 中,以便我无法直接控制的客户端代码可以多次读取它。此 Stream 完成的最后一件事是 reduce() 操作。 Stream 会记住它已经看到的所有项目。不幸的是,在某些情况下迭代器会非常大,因此在其中存储所有项目将导致内存不足错误。但是,一般来说,客户端代码需要多次迭代功能的情况与内存消耗迭代器的情况不同,如果确实存在这种情况,那不是我的问题。
我想确保的是,我可以为需要它的代码提供内存功能,但不能为不需要它的代码提供内存功能(特别是对于根本不查看流的代码)。
Stream 中的 reduce() 代码表示,它的编写方式允许在减少时对 Stream 已访问的部分进行 GC。因此,如果我能确保这确实发生,那就没问题了。但在实践中我怎样才能确保这种情况发生呢?特别是,如果函数 A 创建流并将其传递给函数 B,函数 B 将流传递给函数 C,然后函数 C 调用 reduce(),那么函数 A、B 和 C 中对流的引用仍然存在吗? ?在所有这些情况下,这三个函数中的任何一个都不会进一步使用流,尽管调用不一定是尾递归的。 JVM是否足够聪明,可以确保在调用reduce()时函数A、B和C的引用计数为0,以便GC发生?本质上,这意味着 JVM 在函数 A 中注意到它对该项所做的最后一件事是调用函数 B,因此它在调用 B 的同时消除了自己的句柄,同样对于 B 到 C,以及 C 到 reduce( )。
如果这可以正常工作,那么如果 A、B 或 C 有一个保存该项目的局部变量,它也可以工作吗? (同样,之后不会使用它。)这是因为在不使用本地变量的情况下正确编码会更加棘手。
最佳答案
一个在作用域内但永远不会被读取的变量是死。出于垃圾收集的目的,JVM 可以自由地忽略死变量;仅由死变量指向的对象是无法访问的,并且可以被收集。 JLS 的相关位很模糊地是 §12.6.1 Implementing Finalization ,其中表示:
A reachable object is any object that can be accessed in any potential continuing computation from any live thread.
并解释说:
Optimizing transformations of a program can be designed that reduce the number of objects that are reachable to be less than those which would naively be considered reachable. For example, a Java compiler or code generator may choose to set a variable or parameter that will no longer be used to null to cause the storage for such an object to be potentially reclaimable sooner.
Another example of this occurs if the values in an object's fields are stored in registers. The program may then access the registers instead of the object, and never access the object again. This would imply that the object is garbage. Note that this sort of optimization is only allowed if references are on the stack, not stored in the heap.
如果你的方法 A 只有引用流的死变量,那么它不会阻碍它的收集。
但请注意,这意味着局部变量:如果您有引用流的字段(包括来自封闭嵌套类的方法的封闭局部变量),则这不适用;我不认为 JVM 可以将它们视为已死的。换句话说,这里:public Callable<String> foo(final Object o) {
return new Callable<String>() {
public String call() throws InterruptedException {
String s = o.toString();
Thread.sleep(1000000);
return s;
}
};
}
在收集匿名 Callable
之前,无法收集对象 o
,即使它在 toString
调用之后从未使用过,因为是在 Callable 中引用它的合成字段。
关于java - 在长 Scala 列表或 Stream 上进行 reduce() 操作期间,Java JVM 关于 GC 的智能程度如何?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12571252/
我有一个问题,但由于 this question 部分正在解决,但我想知道如何计算给定间隔之间的天数。 这是一个计算员工休假天数的查询。所以给定(或不给定)一个日期范围,我想计算给定间隔之间有多少假期
变量dateSubtract结果是 16,但我想找到这 2 天之间的总天数,应该是 165。没有 JODA TIME 我该如何做到这一点? String date = "06/17/2014"; Da
我想选择创建日期介于给定月份的第一天和最后一天之间的记录。我通过以下方式计算开始日期和结束日期的月份: 日期“月份”只是时间范围内的随机日期 Calendar cal = Calendar.getIn
我有一个对你们大多数人来说可能微不足道的问题。我尝试了很多,没有找到解决方案,所以如果有人能给我提示,我会很高兴。起点是每周 xts -时间序列。 月周值(value)目标 2011 年 12 月 W
我有一个 Facebook 应用程序,它将用户生日作为 varchar 存储在 mysql 数据库中。我正在尝试获取所有用户的生日 1周后推出,如果是在本周如果生日是上周。 在我的 php 中,我获取
我正在使用以下代码来获取年、月、日中的两个日期之间的差异 tenAppDTO.getTAP_PROPOSED_START_DATE()=2009-11-01 tenAppDTO.getTAP_PRO
我想检查当前时间(在 C++ 中)是否在一个时间范围内。 我想从元组 ("12:00", "17:30") 构造时间范围,即 (string, string) 并检查时间 now() 是否介于两者之间
gitlab 有一个功能,如果我在提交消息中放入票号,那么提交将与 gitlab.com 上的票相关联。 这在进行代码审查时非常方便。不幸的是,开发人员有时会忘记这样做。 我想指定 git hooks
我正在尝试制作使用SQLite数据库的简单注册/登录应用程序,到目前为止我得到了这段代码。这是我的“注册” Activity ,我猜它应该在按下注册按钮后将用户名和 pin(密码)实现到数据库,遗憾的
我正在尝试打开、关闭和写入文件。每当我尝试打开一个文件时,如果我提供的路径中不存在该文件,程序就会告诉我。如果存在,程序将读取其中的内容并显示它。如果用户不想查找文件,可以选择创建文件并用数据填充它。
我想要我的至slideToggle每当发生 react 性变化时,但到目前为止我还无法使其发生。我尝试在 rendered 中使用 JQuery和created模板的事件,但它没有触发。 触发此操作的
我们的 MySQL 遇到了神秘的网络问题。简单的更新查询(使用索引更新单行)通常会立即运行,然后有时(假设 1000 次中有 1 次)因超时而失败。与简单的插入查询相同。数据库没有过载。我们怀疑网络问
我正在使用 actionbarsherlock 的 ActionBar,第一次以横向或水平方向运行应用程序时,选项卡以 Tabs Mode 显示。将方向更改为纵向后,导航模式仍在 Tabs 中。第二次
每天晚上(太平洋标准时间晚上 8 点)我都会对生产数据库(innoDB 引擎)进行全局备份。 这是 mysqldump 命令: mysqldump -u$MYSQLUSER -p$MYSQLPWD -
当我的应用程序第一次启动时,它应该显示用户协议(protocol),这是一个 59kb 的 txt 文件。由于读取文件并将其附加到 TextView 需要一些时间,因此我决定在异步任务中执行此操作并在
如何只允许一个“.”在按键期间的javascript中? 我这里有一个代码: function allowOneDot(txt) { if ((txt.value.split(".")
我已经创建了像主页和用户这样的标题图标。在桌面 View 中,如果我单击用户图像,它会显示相应的重定向页面。如果我在选项卡或移动 View 中将其最小化, 它什么都不显示。此问题仅发生在用户图像上,而
下面的代码在 Release模式下工作,并且仅在 Debug模式下在 g_ItemList.push_back() 引发错误,我浏览了一些 SO 帖子和论坛。有人提到 "You can't itera
我遇到了一个我似乎无法解决的 mmap 问题。下面是设置:我使用 malloc 将一个巨大的多维数组分配到内存中,用我的值填充它,然后我想将它保存在一个文件中。该数组包含 3200000000 个字节
尝试加载共享库: handle = dlopen( "libaaa.so.2.5", RTLD_NOW ); if ( !handle ) { printf("Failed t
我是一名优秀的程序员,十分优秀!