2014-12-28 6 views
1

이 코드가 있습니다왜 내 threadExecutor가 동일한 콜백을 두 번 호출합니까?

public <T> T executeCodeBlockWithTimeLimit(String criticalBlockTimeOutMilli, Callable<T> callable) throws 
     Exception { 
    ExecutorService service = Executors.newSingleThreadExecutor(); 
    Future<T> future = service.submit(callable); 

    startStopWatch(); 
    T result; 

    if (criticalBlockTimeOutMilli != null) { 
     result = future.get(Long.parseLong(criticalBlockTimeOutMilli), TimeUnit.MILLISECONDS); 
    } else { 
     result = callable.call(); 
    } 
    long timeElapsed = stopStopWatch(); 
    e2eResult.runTime = timeElapsed; 
    service.shutdown(); 
    return result; 
} 

및 외부 방법 :

Callable<Void> callable = new 
      Callable<Void>() { 
       @Override 
       public Void call() throws 
         Exception { 

        System.out.println("22222"); 

        return null; 
       } 
      }; 
    timerUtils.executeCodeBlockWithTimeLimit(criticalBlockTimeOutMilli, callable); 

그리고 나는 콘솔 인쇄

22222 
22222 

왜 두 번이라고 볼 수?

+0

criticalBlockTimeOutMilli가 null로 설정 되었습니까? – laune

답변

2

당신은 submit와 함께 집행자에서 미래을 얻을 때 이 메소드는 Callable 오브젝트 메소드를 호출하고 결과를 저장합니다. 코드에서 먼저 Future을 얻고 변수 criticalBlockTimeOutMillinull이면 Callable 객체에서 call() 메서드를 호출합니다. 이는 호출 가능 개체를 두 번 실행한다는 의미입니다.

미래 개체 (이미 Future#get() method 참조)의 이미 수집 된 결과를 사용하므로 get() 메서드를 사용해야합니다.

+0

하지만'Future.get'은 게으른 실행 프로그램입니다. 아니? 'callable.call'을 호출하면 결코 첫 번째 실행 ('submit'의 실행)을 트리거하지 않습니다. 아니? –

+1

'하지만 Future.get은 게으른 실행 프로그램입니다. 아니요? ' 이것은 _Lazy Executor_가 아닙니다. 사실 ** 호출 가능 ** 객체를 ** Executor **에 제출하면 ** 새 스레드 **에서이 객체가 실행되고 그 결과가 ** Future ** 객체에 저장됩니다 ([Executor # submit (Callable)] (http://docs.oracle.com/javase/7/docs/api/java/util/concurrent/ExecutorService.html#submit (java.util.concurrent.Callable))). Indead _Lazy Executor_는 호출 가능 객체가'Future.get()'메소드가 호출 될 때만 실행된다는 것을 의미합니다. – Nemolovich

+1

'그런 다음 callable.call을 호출하면 첫 번째 실행 (제출에서 발생)이 실행되지 않습니다. 아니요? ' ** Callable.call() 메소드를 실행하는 ** Executor **입니다 (제출시). 당신은 스스로 그것을 시작하지 않아도됩니다.만약 당신이 그것을 ** ** 현재/메인 스레드 **에서 메서드를 실행합니다. – Nemolovich

0

다음 실행 프로그램)은 (일단 호출하고

= callable.call 결과를 호출;

은 두 번째로 호출합니다. criticalBlockTimeOutMilli 가정

(?)

1
if (criticalBlockTimeOutMilli != null) { 
    result = future.get(Long.parseLong(criticalBlockTimeOutMilli), TimeUnit.MILLISECONDS); 
} else { 
    result = future.get(); 
} 
0

는 두 번 호출 방법을 요구하고, null로 설정됩니다.

Future<T> future = service.submit(callable); 

을 그리고 당신은 다음과 같이 호출 방법을 통해 호출 : 다음과 같이 실행 프로그램 서비스를 통해 방법을 제출 한

result = callable.call();