2012-07-08 1 views
20

동시성 기능을 사용하는 단위 테스트 (JUnit) 컨트롤러와 을 만들 수 있도록 JavaFX 런타임을 초기화하는 올바른 방법은 무엇입니까?어떻게 JUnit을 사용하여 JavaFX 컨트롤러를 단위 테스트합니까?

@BeforeClass 메서드에서 Application.launch(...)을 호출하면 교착 상태가 발생합니다. Application.launch(...)를 호출 해 있지 않은 경우 다음과 같은 오류가 발생합니다 :


java.lang.IllegalStateException: Toolkit not initialized 
    at com.sun.javafx.application.PlatformImpl.runLater(PlatformImpl.java:121) 
    at com.sun.javafx.application.PlatformImpl.runLater(PlatformImpl.java:116) 
    at javafx.application.Platform.runLater(Platform.java:52) 
    at javafx.concurrent.Task.runLater(Task.java:1042) 
    at javafx.concurrent.Task.updateMessage(Task.java:987) 
    at com.xyz.AudioSegmentExtractor.call(AudioSegmentExtractor.java:64) 
    at com.xyz.CompletionControllerTest.setUp(CompletionControllerTest.java:69) 
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57) 
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) 
    at java.lang.reflect.Method.invoke(Method.java:601) 
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:44) 
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15) 
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41) 
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:27) 
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:76) 
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50) 
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:193) 
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52) 
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191) 
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:42) 
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184) 
    at org.junit.runners.ParentRunner.run(ParentRunner.java:236) 
    at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50) 
    at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38) 
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467) 
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683) 
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390) 
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197) 

추가 답변 :이 내가 @SergeyGrinev에 의해 추천을 기반으로 사용하고 모티브입니다. @BeforeClass에서 launch()를 호출

... // Inside test class 

public static class AsNonApp extends Application { 
    @Override 
    public void start(Stage primaryStage) throws Exception { 
     // noop 
    } 
} 

@BeforeClass 
public static void initJFX() { 
    Thread t = new Thread("JavaFX Init Thread") { 
     public void run() { 
      Application.launch(AsNonApp.class, new String[0]); 
     } 
    }; 
    t.setDaemon(true); 
    t.start(); 
} 
... // controller tests follow... 

답변

14

은 올바른 접근 방법이다. launch()은 호출 코드에 제어권을 반환하지 않습니다. 따라서 이것을 new Thread(...).start()에 입력해야합니다.

7

JavaFX 응용 프로그램 스레드를 시작한 후에 Thread.sleep (500)을 추가 한 후에 만 ​​작동합니다. 아마도 그것은 (내 맥북 프로 망막에 약 200 밀리 초)를하고 준비 FX 환경을 얻기 위해 약간의 시간이 소요

@BeforeClass 
public static void setUpClass() throws InterruptedException { 
    // Initialise Java FX 

    System.out.printf("About to launch FX App\n"); 
    Thread t = new Thread("JavaFX Init Thread") { 
     public void run() { 
      Application.launch(AsNonApp.class, new String[0]); 
     } 
    }; 
    t.setDaemon(true); 
    t.start(); 
    System.out.printf("FX App thread started\n"); 
    Thread.sleep(500); 
} 
+3

[Application.launch (이 두 번 이상 호출 할 수 없습니다해야하며 예외가 발생합니다.) 설명서에 "두 번 이상 호출하면 안되며 예외가 발생합니다."라는 질문에이 대답에서 setUpClass() 메서드를 두 번 이상 실행하려고하면주의해야합니다. – jewelsea

관련 문제