提供3000多款全球软件/控件产品
针对软件研发的各个阶段提供专业培训与技术咨询
根据客户需求提供定制化的软件开发服务
全球知名设计软件,显著提升设计质量
打造以经营为中心,实现生产过程透明化管理
帮助企业合理产能分配,提高资源利用率
快速打造数字化生产线,实现全流程追溯
生产过程精准追溯,满足企业合规要求
以六西格玛为理论基础,实现产品质量全数字化管理
通过大屏电子看板,实现车间透明化管理
对设备进行全生命周期管理,提高设备综合利用率
实现设备数据的实时采集与监控
利用数字化技术提升油气勘探的效率和成功率
钻井计划优化、实时监控和风险评估
提供业务洞察与决策支持实现数据驱动决策
原创|行业资讯|编辑:龚雪|2015-11-26 09:54:02.000|阅读 2640 次
概述:下面是一个简单的例子,给大家讲讲如何使用Junit测试多线程Java代码。
#慧都22周年庆大促·界面/图表报表/文档/IDE/IOT/测试等千款热门软控件火热促销中>>
下面是一个简单的例子,给大家讲讲如何使用Junit测试多线程Java代码。假设我们创建一个可以同时使用的计数器(counter)。因此,我们同时使用class Counter和JUnit测试TestCounter:
public class Counter {
private int count=0;
public void addOne()
{
count++;
}
public int getCount()
{
return count;
}
}
import static org.junit.Assert.assertEquals;
import org.junit.After;
import org.junit.Test;
import org.junit.runner.RunWith;
import com.anarsoft.vmlens.concurrent.junit.ConcurrentTestRunner;
@RunWith(ConcurrentTestRunner.class)
public class TestCounter {
private Counter counter = new Counter();
@Test
public void addOne()
{
counter.addOne();
}
@After
public void testCount()
{
assertEquals("4 Threads running addOne in parallel should lead to 4" , 4 , counter);
}
}
通过使用RunWith注释,JUnit测试将运行在一个特殊的ConcurrentTestRunner。这个被注释为“Test”的测试运行器,有4个线程并行运行。

如果我们运行测试用例就会如下图:

我们有一个race condition可以访问field count。为了解决这个问题,我们将count声明为volatile,然后再次运行。
private volatile int count=0;
现在测试用例成功执行了。大部分时候都会运行成功。如果你运行测试用例很多次,会偶尔出现错误提示。想要知道怎么回事,我们可以在vmlens中启用“延迟同步单元测试(Delay synchronization for unit tests)”。
现在,你会看到以下异常:
java.lang.AssertionError: 4 Threads running addOne in parallel should lead to 4 expected:<4> but was:<3>
at org.junit.Assert.fail(Assert.java:88)
at org.junit.Assert.failNotEquals(Assert.java:834)
at org.junit.Assert.assertEquals(Assert.java:645)
at TestCounter.testCount(TestCounter.java:21)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
at java.lang.reflect.Method.invoke(Unknown Source)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
at com.anarsoft.vmlens.concurrent.junit.internal.InvokeListOfMethods.evaluate(InvokeListOfMethods.java:23)
at com.anarsoft.vmlens.concurrent.junit.internal.ConcurrentStatement.evaluateStatement(ConcurrentStatement.java:12)
at com.anarsoft.vmlens.concurrent.junit.ConcurrentTestRunner.evaluateStatement(ConcurrentTestRunner.java:212)
at com.anarsoft.vmlens.concurrent.junit.ConcurrentTestRunner.runChildrenConcurrently(ConcurrentTestRunner.java:172)
at com.anarsoft.vmlens.concurrent.junit.ConcurrentTestRunner.access$0(ConcurrentTestRunner.java:78)
at com.anarsoft.vmlens.concurrent.junit.ConcurrentTestRunner$1.evaluate(ConcurrentTestRunner.java:72)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:86)
at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:459)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:675)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:382)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:192)
count++并不是一个操作,而是一个6字节的代码操作。对于field count,包含一个读和一个写:
ALOAD 0: this DUP GETFIELD Counter.count : int ICONST_1 IADD PUTFIELD Counter.count : int
通过这三个操作之间的延迟,我们要确保两个线程并行执行这些操作。在并行执行时,计数器将一直小于4,有时是3,有时只有2。
为了解决这个问题,我们使用atomic方法。可以通过使用java.util.concurrent.atomic.AtomicInteger来完成:
import java.util.concurrent.atomic.AtomicInteger;
public class Counter {
private final AtomicInteger count= new AtomicInteger();
public void addOne()
{
count.incrementAndGet();
}
public int getCount()
{
return count.get();
}
}
现在测试用例成功了。在这个测试中,Test runner使用的concurrent-junit,race condition catcher使用的vmlens。
本站文章除注明转载外,均为本站原创或翻译。欢迎任何形式的转载,但请务必注明出处、不得修改原文相关链接,如果存在内容上的异议请邮件反馈至chenjj@hmdbvip.cn




近日,欧洲知名企业级前端框架 Vaadin 正式发布了 官方 MCP(Model Context Protocol)服务端,这意味着开发者的 AI 编码助手(如 GitHub Copilot、Claude、Cursor 等)可以实时访问最新的 Vaadin 官方文档,实现精准代码生成与智能指导。
RTF 格式广泛用于文本格式化、字体样式和图片插入,而 PDF(Portable Document Format)则是跨平台文档存储和传输的标准格式,具有高度的兼容性和安全性。本教程将深入介绍如何使用 Spire.Doc for Java 库在 Java 中将 RTF 文件转换为 PDF。
模型驱动工程(MDE)作为一种以模型为核心的软件开发范式,通过提升抽象级别和自动化代码生成,为应对这些挑战提供了系统化的解决方案。Sparx Systems Enterprise Architect作为基于UML的综合性建模平台,为企业实施MDE提供了完整的技术支撑,实现了从业务需求到可执行代码的自动化转换,显著提升了软件开发的质量和效率。
在API的全生命周期管理中,负载测试通过不同场景的模拟,帮助团队发现潜在瓶颈。本文将介绍五种常见的负载测试策略——基线测试、压力测试、浸泡测试、峰值测试与尖峰测试,并结合专业工具的使用方式,展示如何高效验证API的性能极限。
服务电话
重庆/ 023-68661681
华东/ 13452821722
华南/ 18100878085
华北/ 17347785263
客户支持
技术支持咨询服务
服务热线:400-700-1020
邮箱:sales@hmdbvip.cn
关注我们
地址 : 重庆市九龙坡区火炬大道69号6幢
永利最大(官方)网站