Java 在套件级别并行运行JUnit测试?
我在JUnit测试套件中组织了一系列测试。这些测试极大地利用了selenium来测试web应用程序。因此,对于selenium来说,这些测试的运行时间很长。由于测试数据库中存在重叠,套件中的测试类无法并行运行,因此我希望并行运行套件 JUnit ParallelComputer只能在类或方法级别并行执行测试,JUnit是否有任何标准的方法可以通过套件来执行这些测试 如果我只是将套件类传递给junit运行程序,并将计算机配置为在类级别上并行化,那么它会选择测试类本身,而不是套件 溴Java 在套件级别并行运行JUnit测试?,java,junit,automated-tests,parallel-processing,suite,Java,Junit,Automated Tests,Parallel Processing,Suite,我在JUnit测试套件中组织了一系列测试。这些测试极大地利用了selenium来测试web应用程序。因此,对于selenium来说,这些测试的运行时间很长。由于测试数据库中存在重叠,套件中的测试类无法并行运行,因此我希望并行运行套件 JUnit ParallelComputer只能在类或方法级别并行执行测试,JUnit是否有任何标准的方法可以通过套件来执行这些测试 如果我只是将套件类传递给junit运行程序,并将计算机配置为在类级别上并行化,那么它会选择测试类本身,而不是套件 溴 Frank因为
Frank因为套件用于注释类,所以以
JUnitCore.runClasses(ParallelComputer.classes(),cls)
的方式运行套件注释类cls
是套件注释类
@RunWith(Suite.class)
@Suite.SuiteClasses({
Test1.class,
Test2.class})
public class Suite1 {
}
@RunWith(Suite.class)
@Suite.SuiteClasses({
Test3.class,
Test4.class})
public class Suite2 {
}
...
JUnitCore.runClasses(ParallelComputer.classes(), new Class[]{Suite1.class, Suite2.class})
下面是一些对我有用的代码。这不是我写的。如果您使用
@RunWith(ConcurrentSuite.class)
而不是@RunWith(Suite.class)
,它应该可以工作。下面还需要一个注释
package utilities.runners;
import org.junit.internal.builders.AllDefaultPossibilitiesBuilder;
import org.junit.runner.Runner;
import org.junit.runners.Suite;
import org.junit.runners.model.InitializationError;
import org.junit.runners.model.RunnerBuilder;
import org.junit.runners.model.RunnerScheduler;
import utilities.annotations.Concurrent;
import java.util.Arrays;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;
import java.util.concurrent.CompletionService;
import java.util.concurrent.ExecutorCompletionService;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.atomic.AtomicInteger;
/**
* @author Mathieu Carbou (mathieu.carbou@gmail.com)
*/
public final class ConcurrentSuite extends Suite {
public ConcurrentSuite(final Class<?> klass) throws InitializationError {
super(klass, new AllDefaultPossibilitiesBuilder(true) {
@Override
public Runner runnerForClass(Class<?> testClass) throws Throwable {
List<RunnerBuilder> builders = Arrays.asList(
new RunnerBuilder() {
@Override
public Runner runnerForClass(Class<?> testClass) throws Throwable {
Concurrent annotation = testClass.getAnnotation(Concurrent.class);
if (annotation != null)
return new ConcurrentJunitRunner(testClass);
return null;
}
},
ignoredBuilder(),
annotatedBuilder(),
suiteMethodBuilder(),
junit3Builder(),
junit4Builder());
for (RunnerBuilder each : builders) {
Runner runner = each.safeRunnerForClass(testClass);
if (runner != null)
return runner;
}
return null;
}
});
setScheduler(new RunnerScheduler() {
ExecutorService executorService = Executors.newFixedThreadPool(
klass.isAnnotationPresent(Concurrent.class) ?
klass.getAnnotation(Concurrent.class).threads() :
(int) (Runtime.getRuntime().availableProcessors() * 1.5),
new NamedThreadFactory(klass.getSimpleName()));
CompletionService<Void> completionService = new ExecutorCompletionService<Void>(executorService);
Queue<Future<Void>> tasks = new LinkedList<Future<Void>>();
@Override
public void schedule(Runnable childStatement) {
tasks.offer(completionService.submit(childStatement, null));
}
@Override
public void finished() {
try {
while (!tasks.isEmpty())
tasks.remove(completionService.take());
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
} finally {
while (!tasks.isEmpty())
tasks.poll().cancel(true);
executorService.shutdownNow();
}
}
});
}
static final class NamedThreadFactory implements ThreadFactory {
static final AtomicInteger poolNumber = new AtomicInteger(1);
final AtomicInteger threadNumber = new AtomicInteger(1);
final ThreadGroup group;
NamedThreadFactory(String poolName) {
group = new ThreadGroup(poolName + "-" + poolNumber.getAndIncrement());
}
@Override
public Thread newThread(Runnable r) {
return new Thread(group, r, group.getName() + "-thread-" + threadNumber.getAndIncrement(), 0);
}
}
}
嗨,谢谢你的回答。遗憾的是,这并没有以我预期的方式运行测试。正如我在问题末尾所写的,它在套件中并行运行测试类,而不是套件本身。您的示例确实以以下方式执行:Test1和Test2并行运行,这正是我必须防止的行为。欢迎!如果我的代码不能帮助您,您可以编写多线程来运行套件(如果没有更好的解决方案的话)。我正在考虑这个问题,但如果我这样做,我必须自己聚合junit结果,因为我需要它来生成测试报告。但是谢谢你的快速帮助!看起来这个URL上有这个代码的副本是的,我应该包括这个链接。谢谢。在添加了一个额外的构造函数之后,这就起作用了:public ConcurrentSuite(类klass,RunnerBuilder)抛出初始化错误{This(klass);},因为链接现在已经死了-这是Maven上预编译的jar和源代码:和
package utilities.annotations;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* @author Mathieu Carbou (mathieu.carbou@gmail.com)
*/
@Retention(RetentionPolicy.RUNTIME)
@Target({ ElementType.TYPE })
public @interface Concurrent {
int threads() default 5;
}