- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
解决了
我试图了解为什么我的单元测试之一会消耗这么多的内存。我所做的第一件事就是使用VisualVM运行该测试并进行测量:
最初的平坦线归因于测试开始时的Thread.sleep()
,从而为VisualVM提供了启动时间。
测试(和设置方法)非常简单:
@BeforeClass
private void setup() throws Exception {
mockedDatawireConfig = mock(DatawireConfig.class);
when(mockedDatawireConfig.getUrl()).thenReturn(new URL("http://example.domain.fake/"));
when(mockedDatawireConfig.getTid()).thenReturn("0000000");
when(mockedDatawireConfig.getMid()).thenReturn("0000000");
when(mockedDatawireConfig.getDid()).thenReturn("0000000");
when(mockedDatawireConfig.getAppName()).thenReturn("XXXXXXXXXXXXXXX");
when(mockedDatawireConfig.getNodeId()).thenReturn("t");
mockedVersionConfig = mock(VersionConfig.class);
when(mockedVersionConfig.getDatawireVersion()).thenReturn("000031");
defaultCRM = new ClientRefManager();
defaultCRM.setVersionConfig(mockedVersionConfig);
defaultCRM.setDatawireConfig(mockedDatawireConfig);
}
@Test
public void transactionCounterTest() throws Exception {
Thread.sleep(15000L);
String appInstanceID = "";
for (Long i = 0L; i < 100000L; i++) {
if (i % 1000 == 0) {
Assert.assertNotEquals(defaultCRM.getAppInstanceID(), appInstanceID);
appInstanceID = defaultCRM.getAppInstanceID();
}
ReqClientID r = defaultCRM.getReqClientID(); // This call is where memory use explodes.
Assert.assertEquals(getNum(r.getClientRef()), new Long(i % 1000));
Assert.assertEquals(r.getClientRef().length(), 14);
}
Thread.sleep(10000L);
}
defaultCRM.getReqClientID()
生成具有有效计数器(介于000-999之间)的正确ReqClientID对象,并确保随机化前缀在翻转时正确更改。
defaultCRM.getReqClientID()
是发生内存问题的地方。让我们来看看:
public ReqClientID getReqClientID() {
ReqClientID req = new ReqClientID();
req.setDID(datawireConfig.getDid()); // #1
req.setApp(String.format("%s&%s", datawireConfig.getAppName(), versionConfig.toString())); // #2
req.setAuth(String.format("%s|%s", datawireConfig.getMid(), datawireConfig.getTid())); // #3
Long c = counter.getAndIncrement();
String appID = appInstanceID;
if(c >= 999L) {
LOGGER.warn("Counter exceeds 3-digits. Resetting appInstanceID and counter.");
resetAppInstanceID();
counter.set(0L);
}
req.setClientRef(String.format("%s%s%03dV%s", datawireConfig.getNodeId(), appID, c, versionConfig.getDatawireVersion())); // #4
return req;
}
String
设置器,计算一个递增计数器,并在翻转时添加随机前缀。
+
进行简单的字符串连接的。我改为
String.format()
,但是没有任何效果。我也尝试了
StringBuilder
和
append()
无效。
-XX:+UseG1GC
,
-XX:InitiatingHeapOccupancyPercent=35
和
-Xms1g -Xmx1g
(请注意,在我的构建从机上1g仍然不合理,我希望将其最大长度降至256m附近)。这是图形:
-Xms25m -Xmx256m
会导致OutOfMemoryError。
+
/
String.format()
/
StringBuilder
之外,我看不到其他替代方法,它们似乎都产生相同的结果。我是否缺少一些神奇的方式来构建琴弦?
System.gc()
,因此我尝试每隔1K循环执行一次。这对内存使用有显着影响,而对性能则有严重影响:
Thread.sleep()
。几个问题:
System.gc()
(
i % 1000 == 0
)。实际上,这导致未使用的堆空间减少。为什么在第一次调用后总堆空间从未减少?
System.gc()
在保持使用的堆空间不变方面没有更有效。
mockedDatawireConfig
向我指出了正确的方向。这实际上是Spring @ConfigurationProperties类(即Spring将数据从Yaml加载到实例中并在需要的地方连接该实例)。在单元测试中,我没有使用任何与Spring相关的东西(单元测试,而不是集成测试)。在这种情况下,它只是带有getter和setter的POJO,但类中没有逻辑。
setup()
中看到它。我决定切换到对象的真实实例,而不是模拟对象。这就彻底解决了问题! Mockito似乎是固有的问题,或者可能是因为我似乎正在使用2.0.2-
beta 。如果确实存在未知问题,我将作进一步调查,并与Mockito开发人员联系。
最佳答案
好吧,这取决于JVM如何分配堆空间。它只是看到内存消耗大幅度(且快速!)增加,因此分配了足够的堆空间以免发生OutOfMemoryException。
您已经看到,可以通过使用参数来更改此行为。您还可以看到,一旦使用率保持不变,堆就不会进一步增长(它停止在〜3G而不是直到〜8G为止)。
要真正看到正在发生的事情,您不应该进行一些printf调试(这意味着注释掉某些东西并查看会发生什么),而应该使用IDE或其他工具来检查正在使用什么内存。
这样做将向您显示(例如):120k个String实例将2GiB或1.5GiB垃圾和500MiB用作字符串。
然后,您清楚地知道它是只是一个惰性集合(因为一个集合有开销)还是您仍然有一些引用在徘徊(我想说不,因为增长停止了)。
作为一种肮脏的解决方法,您还可以在循环中添加System.gc()
调用以强制执行Garbage-Collection,以查看其是否可以提高堆使用率(当然会浪费CPU时间)。
关于java - Java字符串垃圾回收:或者为什么这会消耗这么多内存,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37843795/
我正在构建一个我想要的 FragmentPagerAdapter(带有小图标的那个); 到屏幕中间(水平)的第一个项目 滑动它并使它们向左/向右移动屏幕的五分之一 实现此目标的最佳方法是什么? 这是一
我正在使用 GalleryView 和大约 40 张图片,而且速度很慢,因为没有回收... 任何人都可以向我展示在 getView 方法上对 GalleryView 的基本回收。 public cla
我收到警告 This FragmentManager should be recycled after use with #recycle() 我正在尝试修复它。有什么建议吗? date.setOnC
已结束。此问题不符合 Stack Overflow guidelines .它目前不接受答案。 这个问题似乎与 a specific programming problem, a software a
是否可以在 global.asax 中捕获回收事件? 我知道 Application_End 会被触发,但有没有办法知道它是由应用程序池的回收触发的? 谢谢,Lieven Cardoen 又名 Joh
当我第一次启动应用程序时,我以正确的方式获得了抽屉导航。当我向下滚动抽屉时,问题就出现了。元素开始消失,其中一些甚至交换了位置。我试图找出问题所在,但找不到。希望你能帮助我。 我已经使用 getIte
我正在尝试构建一个 Android 应用程序,在其中使用 RecyclerView。 在 RecyclerView 的项目中,我使用一个 TextView 和四个 CheckBox 来选择任何人。 现
回收 IIS7 应用程序池是否会终止任何当前正在执行的请求?还是等待所有请求完成(如排水管停止)? 我不希望回收规则导致我的 WCF 站点出现间歇性错误。 谢谢 最佳答案 不。 By default,
我动态创建 div,并希望通过检查 div 的 ID 以正确的顺序放置新的 div。 创建新数据时,我对数组进行排序并创建一个新的 div 容器。第一次构建 DOM 时,它工作正常,因为我先创建数据,
IIS 中 Web 应用程序的 .net Recycle 的 Java 等价物是什么。 这是在 IIS 之外的 Linux 机器上使用 Java 时的情况。 只是停止和启动应用程序吗? 最佳答案 并非
问题: 假设您的传输速度高达 10 MB/s,那么回收 DatagramPacket 对象(而不是每次发送数据包时创建一个新对象)是否是一个好主意? 故事: 我正在创建一个 LAN 文件同步应用程序,
我已经使用适配器创建了一个 ListView,但唯一的问题是我们使用了不同的布局,因为它是一个消息传递应用程序。我想通过使用 View 回收来提高性能,但不知道如何回收 View 。 是否可以更改 V
我是 Android 初学者,我不明白为什么会这样。 Activity 截图: 一切正常,除非我向下滚动(因此我认为它与回收有关)...所以当我向上滚动并尝试撤消第一篇文章中的投票(红色箭头)时,它认
我在 RecyclerView 中使用 FlexboxLayoutManager,我需要阻止 RV 回收。我尝试将 itemViewCacheSize() 设置为项目总数,但没有帮助。我也尝试将 ma
在我使用 SearchView 搜索项目后尝试从 RecyclerView 中删除项目时遇到问题,而如果我不使用 SearchView 并正常删除该项目,它工作正常。 这是一个示例,我正在删除该项目,
我正在尝试动态更改 RecycleView 中的图像。它将成功更改但是当我滚动 RecycleView ImageView 时将更改 这是我的适配器类代码: public class Recycler
我有一个表格如下 |GroupID | UserID | -------------------- |1 | 1 | |1 | 2 | |1 |
大家好,在我的 Activity 布局中,我使用这个 XML 来获取按钮数组
我有一个动画时钟,目前有内存泄漏。我目前似乎能够将东西放在图像之上,但不能将其取走或必须清除它(除了重新绘制整个时钟以使秒针不在两个不同的位置之外)。 用于清除时钟组件的代码是: for(UIView
我正在尝试创建一个简单的动画,在 JavaFX 2.x 中将圆向左移动一个像素,该动画有效,但每当我尝试第二次启动时,它就不再起作用了。我尝试了几种方法,代码如下: public void handl
我是一名优秀的程序员,十分优秀!