没有找到合适的产品?
联系客服协助选型:023-68661681
提供3000多款全球软件/控件产品
针对软件研发的各个阶段提供专业培训与技术咨询
根据客户需求提供定制化的软件开发服务
全球知名设计软件,显著提升设计质量
打造以经营为中心,实现生产过程透明化管理
帮助企业合理产能分配,提高资源利用率
快速打造数字化生产线,实现全流程追溯
生产过程精准追溯,满足企业合规要求
以六西格玛为理论基础,实现产品质量全数字化管理
通过大屏电子看板,实现车间透明化管理
对设备进行全生命周期管理,提高设备综合利用率
实现设备数据的实时采集与监控
利用数字化技术提升油气勘探的效率和成功率
钻井计划优化、实时监控和风险评估
提供业务洞察与决策支持实现数据驱动决策
原创|行业资讯|编辑:龚雪|2015-11-26 09:54:02.000|阅读 2578 次
概述:下面是一个简单的例子,给大家讲讲如何使用Junit测试多线程Java代码。
# 界面/图表报表/文档/IDE等千款热门软控件火热销售中 >>
下面是一个简单的例子,给大家讲讲如何使用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@defefr.cn
CEETRON Envision专为CAE数据工作打造,能够显著简化大规模模拟后处理的可视化开发流程,是搭建桌面与云端CAE平台的理想之选。
微服务架构凭借灵活和可扩展的优势越来越普及,但随之而来的变更也成了开发团队的“心头大患”。服务之间依赖复杂,接口改来改去,不仅让开发更费劲,还容易导致测试用例失效、测试效率下降,甚至埋下系统不稳的隐患。自动化测试利器Parasoft SOAtest精准捕捉接口变更、秒级识别波及范围、自动迁移测试资产,让变更引发的测试混乱与风险,从此快、准、稳地一键修复。
在敏捷和DevOps当道的今天,测试团队需要更灵活、更高效的工具。TestComplete通过直观的拖拽操作降低门槛,让创建测试用例变得简单快速;凭借强大的可复用性,显著减少了重复劳动和维护成本。在各种场景里,它都展现出了强大的适应性和实用价值,帮助测试团队真正把自动化UI测试从“负担”变成提升交付速度和质量的有力“助力”。
HOOPS Visualize 专注于为工程应用打造高性能、高质量的图形可视化体验。
服务电话
重庆/ 023-68661681
华东/ 13452821722
华南/ 18100878085
华北/ 17347785263
客户支持
技术支持咨询服务
服务热线:400-700-1020
邮箱:sales@defefr.cn
关注我们
地址 : 重庆市九龙坡区火炬大道69号6幢