Java 애플리케이션을 다시 시작하려면 어떻게해야합니까?


94

Java AWT 애플리케이션을 어떻게 다시 시작할 수 있습니까? 이벤트 핸들러를 첨부 한 버튼이 있습니다. 응용 프로그램을 다시 시작하려면 어떤 코드를 사용해야합니까?

Application.Restart()C # 애플리케이션에서하는 것과 똑같은 일을하고 싶습니다 .


2
질문을 이해하지 못할 수도 있습니다. 애플리케이션에 애플리케이션을 다시 시작하는 버튼이 있기를 원하십니까? 그렇다면 앱이 더 이상 실행되지 않으면 자동으로 다시 시작할 수 있어야합니까? 그것은 불가능한 것 같습니다.
Jay

JVM이 중지 된 후에는 메인 자바 프레임을 어떻게 다시 생성 할 수 있는지 묻지 않습니다.
Azfar Niaz

2
불가능하지 않다. 이클립스 워크 벤치가 자주 다시 시작되는 것을 봅니다. 윈도우도 업데이트 후에이 트릭을 수행합니다. 잘못된 가정은 응용 프로그램이 그 아래에 아무것도없이 실행되는 유일한 것입니다. 다시 시작할 수있는 런처가 필요합니다. 거북이는 끝까지 내려갑니다.
whatnick

C # 응용 프로그램과 동일한 방식으로 System.restart ()를 작성할 수 있습니까?
Azfar Niaz

@aniaz 그런 다음 프레임을 표시 / 숨기기를 원한다는 점을 지적하도록 질문을 업데이트해야합니다. 응용 프로그램은 프레임이 아닙니다.
whatnick

답변:


105

물론 Java 애플리케이션을 다시 시작할 수 있습니다.

다음 방법은 Java 애플리케이션을 다시 시작하는 방법을 보여줍니다.

public void restartApplication()
{
  final String javaBin = System.getProperty("java.home") + File.separator + "bin" + File.separator + "java";
  final File currentJar = new File(MyClassInTheJar.class.getProtectionDomain().getCodeSource().getLocation().toURI());

  /* is it a jar file? */
  if(!currentJar.getName().endsWith(".jar"))
    return;

  /* Build command: java -jar application.jar */
  final ArrayList<String> command = new ArrayList<String>();
  command.add(javaBin);
  command.add("-jar");
  command.add(currentJar.getPath());

  final ProcessBuilder builder = new ProcessBuilder(command);
  builder.start();
  System.exit(0);
}

기본적으로 다음을 수행합니다.

  1. Java 실행 파일을 찾습니다 (여기서 Java 바이너리를 사용했지만 요구 사항에 따라 다름).
  2. 응용 프로그램을 찾습니다 (제 경우에는 항아리, MyClassInTheJar클래스를 사용 하여 항아리 위치 자체를 찾습니다)
  3. jar를 다시 시작하는 명령을 빌드합니다 (이 경우 Java 바이너리 사용).
  4. 실행 해! (따라서 현재 응용 프로그램을 종료하고 다시 시작)

5
동일한 앱의 두 버전이 동시에 실행되는 짧은 시간 프레임이 있지 않나요?
Monir

5
System.exit (0)이 자식 프로세스를 종료하지 않습니까?
Horcrux7 2015

16
@Veger System.exit(0)자식 프로세스를 종료 할지 여부에 대한 질문은이 답변이 실제로 작동하는지 그리고 그 이유와 동일한 대답을 갖습니다. 대답과 함께 현명한 설명을 할 수 없다면 잘못된 일을 한 것입니다. 답변보다 더 많은 질문을 제공하는 답변은 철저한 답변의 예가 아닙니다. 좋은 답변은 코드를 보여줄뿐만 아니라 작동 방식과 이유, 단점 및 대안이 무엇인지 설명합니다. 당신은 이러한 것들을 덮 으려하지도 않았습니다.
Tomáš Zato-Monica 복원

8
@ Horcrux7의 질문에 답할 것인지에 대한 의견이 너무 많습니다. 너희들은 처음부터 그에게 대답을 말할 수 있었다. 글쎄, 나는 계속해서 그것을 할 것입니다 (좀 늦게 알고 있습니다) : 아닙니다. 그곳에.
Voldemort

10
내 질문에 스스로 대답합니다. 샘플이 작동하지 않습니다 !!! System.exit (0)은 클라이언트 프로세스를 즉시 종료합니다.
Horcrux7

35
import java.io.File;
import java.io.IOException;
import java.lang.management.ManagementFactory;

public class Main {
    public static void main(String[] args) throws IOException, InterruptedException {
        StringBuilder cmd = new StringBuilder();
        cmd.append(System.getProperty("java.home") + File.separator + "bin" + File.separator + "java ");
        for (String jvmArg : ManagementFactory.getRuntimeMXBean().getInputArguments()) {
            cmd.append(jvmArg + " ");
        }
        cmd.append("-cp ").append(ManagementFactory.getRuntimeMXBean().getClassPath()).append(" ");
        cmd.append(Main.class.getName()).append(" ");
        for (String arg : args) {
            cmd.append(arg).append(" ");
        }
        Runtime.getRuntime().exec(cmd.toString());
        System.exit(0);
    }
}

불가능하다고 말하는 모든 사람들에게 바칩니다.

이 프로그램은 원래 명령 줄을 재구성하는 데 사용할 수있는 모든 정보를 수집합니다. 그런 다음 시작하고 동일한 명령이므로 응용 프로그램이 두 번째로 시작됩니다. 그런 다음 원래 프로그램을 종료하면 자식 프로그램이 계속 실행되고 (리눅스에서도) 동일한 작업을 수행합니다.

경고 : 이것을 실행하면 포크 폭탄 과 유사한 새로운 프로세스 생성이 끝나지 않는다는 점에 유의하십시오 .


가능한 개선 ManagementFactory.getRuntimeMXBean().getInputArguments() 은 JVM에 전달 된 입력 인수 만 제공합니다. 애플리케이션에 전달 된 매개 변수가 누락되었습니다. 예 : java -jar start.jar -MISSED_PARAM=true. Oracle jvm에서는 System.getProperty("sun.java.command").
Chris2M

1
하위 VM과 상위 VM이 파이프로 서로 연결되지 않으면 상위 VM이 종료 될 수 있으며, 이는 하위 VM이 시작되는 방식입니다. 사용하여 ProcessBuilder하고 inheritIO(), 아이를 VM 부모 VM이 끝날 것이라고 방식으로 시작할 수 있습니다.
기독교 Hujer

1
나는 이것의 버전을 얻었다. 이 주석은 중지하는 방법을 알려줍니다. java.exe가 포함 된 경로에서 이름을 바꿉니다.
Dale

이것은 엄밀히 말하면 다시 시작하는 것이 아니라 이것과 동일한 인수로 새 JVM을 시작하는 것입니다.
Thorbjørn Ravn Andersen

4
차이점이 뭐야? PC를 다시 시작하는 것과 OS를 종료하고 다시 부팅하는 것 사이에 차이가 있습니까?
Meinersbur 2017-06-20

28

기본적으로 할 수 없습니다. 적어도 신뢰할 수있는 방식은 아닙니다. 그러나 그럴 필요는 없습니다.

수없는 부분

Java 프로그램을 다시 시작하려면 JVM을 다시 시작해야합니다. JVM을 다시 시작하려면 다음을 수행해야합니다.

  1. java사용 된 실행기를 찾습니다 . 시도해 볼 수는 System.getProperty("java.home")있지만 실제로 응용 프로그램을 시작하는 데 사용 된 실행기를 가리킬 것이라는 보장은 없습니다. (반환 된 값 이 애플리케이션을 시작하는 데 사용 된 JRE를 가리 키지 않거나에서 재정의되었을 수 있습니다 -Djava.home.)

  2. 당신은 아마도 설정 등 (원래 메모리 명예 할 것입니다 -Xmx, -Xms당신이 사용하는 설정이 먼저 JVM을 시작하는 알아낼 필요가 ...) 그래서를. 사용해 볼 수는 ManagementFactory.getRuntimeMXBean().getInputArguments()있지만 이것이 사용 된 설정을 반영한다는 보장은 없습니다. 이것은 해당 방법 의 문서 에도 나와 있습니다 .

    일반적으로 'java'명령에 대한 모든 명령 행 옵션이 JVM (Java Virtual Machine)에 전달되는 것은 아닙니다. 따라서 반환 된 입력 인수에 모든 명령 줄 옵션이 포함되지 않을 수 있습니다.

  3. 프로그램 Standard.in이 원래의 stdin 에서 입력을 읽으면 다시 시작할 때 손실됩니다.

  4. 이러한 트릭과 해킹의 대부분은 SecurityManager.

부분 이 필요하지 않아야합니다.

모든 것을 정리하기 쉽도록 응용 프로그램을 디자인하고 그 후에 "main"클래스의 새 인스턴스를 만드는 것이 좋습니다.

많은 애플리케이션은 메인 메소드에서 인스턴스를 생성하는 것 외에는 아무것도하지 않도록 설계되었습니다.

public class MainClass {
    ...
    public static void main(String[] args) {
        new MainClass().launch();
    }
    ...
}

이 패턴을 사용하면 다음과 같은 작업을 쉽게 수행 할 수 있습니다.

public class MainClass {
    ...
    public static void main(String[] args) {
        boolean restart;
        do {
            restart = new MainClass().launch();
        } while (restart);
    }
    ...
}

및 수 있도록 launch()응용 프로그램 방식으로 종료 된 경우에만 다시 시작해야 함을하는 경우에 true를 돌려줍니다.


3
더 나은 디자인 조언을 위해 +1; 예를 들어 특히 JNI를 사용하는 경우에는 가능하지 않은 경우도 있습니다.
maerics

음, 네이티브 라이브러리는 JNI 인터페이스에서 수정할 수없는 전역 상태를 수정할 수 있으므로 프로세스를 다시 시작하는 것 외에 프로그램 상태를 "다시 시작"할 수있는 방법이 없습니다. 물론 네이티브 라이브러리는 더 잘 설계되어야하지만 때로는 제어 할 수없는 것에 의존합니다.
maerics 2010

좋아,하지만 그 추론으로 일부 내부 정적 변수를 수정하는 순수한 Java 라이브러리를 가질 수 있습니다. 그러나 이것은 디자인 결함이며 잘 작성된 라이브러리에서는 발생하지 않아야합니다.
aioobe 2010

1
Meinersbur와 내 답변에서 볼 수 있듯이 외부 응용 프로그램 / 데몬 없이도 완벽하게 가능하기 때문에 귀하의 대답은 올바르지 않습니다. 자동 업데이트 목적으로 응용 프로그램을 다시 시작하는 것이 좋은 솔루션이므로 실제로 응용 프로그램을 다시 시작해야합니다.
Veger 2010

1
하지만 당신은 외부 프로그램을 사용합니다 java! Java가 프로그램이 아니라 언어 사양이라는 사실을 잊고 있습니다. 예를 들어 kaffe 와 같은 다른 jvm을 사용하여 프로그램을 실행하면 어떻게 됩니까? 어쨌든 내 대답을 업데이트했습니다 :-)
aioobe

10

엄밀히 말하면 Java 프로그램은 자체적으로 재시작 할 수 없습니다. 그렇게하려면 실행중인 JVM을 종료 한 다음 다시 시작해야하지만 JVM이 더 이상 실행되지 않으면 (종료) 조치를 취할 수 없습니다.

AWT 구성 요소를 다시로드, 압축 및 시작하기 위해 사용자 정의 클래스 로더로 몇 가지 트릭을 수행 할 수 있지만 GUI 이벤트 루프와 관련하여 많은 골칫거리가 될 수 있습니다.

애플리케이션이 시작되는 방법에 따라 JVM이 특정 코드와 함께 종료되는 동안 계속되는 do / while 루프가 포함 된 래퍼 스크립트에서 JVM을 시작할 수 있습니다. 그런 다음 AWT 앱은를 호출해야합니다 System.exit(RESTART_CODE). 예를 들어 스크립팅 의사 코드에서 :

DO
  # Launch the awt program
  EXIT_CODE = # Get the exit code of the last process
WHILE (EXIT_CODE == RESTART_CODE)

AWT 앱은 다시 시작할 필요가없는 "정상"종료시 RESTART_CODE 이외의 다른 항목으로 JVM을 종료해야합니다.


매우 흥미로운 솔루션입니다. OSX의 문제는 일반적으로 Java 앱이 컴파일 된 상태에서 실행 JavaApplicationStub된다는 것입니다. 쉬운 방법이 있는지 확실하지 않습니다.
Dan Rosenstark

7

Eclipse는 일반적으로 플러그인이 설치된 후 다시 시작됩니다. Windows 용 래퍼 eclipse.exe (런처 앱)를 사용하여이를 수행합니다. 이 애플리케이션은 코어 Eclipse 러너 jar를 실행하고 Eclipse Java 애플리케이션이 재실행 코드로 종료되면 eclipse.exe가 워크 벤치를 다시 시작합니다. 다시 시작하기 위해 유사한 네이티브 코드, 쉘 스크립트 또는 다른 Java 코드 래퍼를 빌드 할 수 있습니다.


5

윈도우

public void restartApp(){

    // This launches a new instance of application dirctly, 
    // remember to add some sleep to the start of the cmd file to make sure current instance is
    // completely terminated, otherwise 2 instances of the application can overlap causing strange
    // things:)

    new ProcessBuilder("cmd","/c start /min c:/path/to/script/that/launches/my/application.cmd ^& exit").start();
    System.exit(0);
}

/ min 최소화 된 창에서 스크립트 시작

^ & 종료 후 cmd 창 닫기

샘플 cmd 스크립트는

@echo off
rem add some sleep (e.g. 10 seconds) to allow the preceding application instance to release any open resources (like ports) and exit gracefully, otherwise the new instance could fail to start
sleep 10   
set path=C:\someFolder\application_lib\libs;%path%
java -jar application.jar

수면 10 10 초 동안 수면



4

이 질문은 오래되고 답변되었지만 일부 솔루션에서 문제를 발견하고 제안을 믹스에 추가하기로 결정했습니다.

일부 솔루션의 문제점은 단일 명령 문자열을 빌드한다는 것입니다. 이로 인해 일부 매개 변수에 공백, 특히 java.home이 포함 된 경우 문제가 발생합니다 .

예를 들어, 창문에서는

final String javaBin = System.getProperty("java.home") + File.separator + "bin" + File.separator + "java";

다음과 같은 것을 반환 할 수 있습니다.C:\Program Files\Java\jre7\bin\java

이 문자열은 따옴표로 묶거나 공백으로 인해 이스케이프해야합니다 Program Files. 큰 문제는 아니지만 특히 크로스 플랫폼 애플리케이션에서 다소 성 가시고 오류가 발생하기 쉽습니다.

따라서 내 솔루션은 명령 을 명령 배열 로 빌드 합니다.

public static void restart(String[] args) {

        ArrayList<String> commands = new ArrayList<String>(4 + jvmArgs.size() + args.length);
        List<String> jvmArgs = ManagementFactory.getRuntimeMXBean().getInputArguments();

        // Java
        commands.add(System.getProperty("java.home") + File.separator + "bin" + File.separator + "java");

        // Jvm arguments
        for (String jvmArg : jvmArgs) {
            commands.add(jvmArg);
        }

        // Classpath
        commands.add("-cp");
        commands.add(ManagementFactory.getRuntimeMXBean().getClassPath());

        // Class to be executed
        commands.add(BGAgent.class.getName());

        // Command line arguments
        for (String arg : args) {
            commands.add(arg);
        }

        File workingDir = null; // Null working dir means that the child uses the same working directory

        String[] env = null; // Null env means that the child uses the same environment

        String[] commandArray = new String[commands.size()];
        commandArray = commands.toArray(commandArray);

        try {
            Runtime.getRuntime().exec(commandArray, env, workingDir);
            System.exit(0);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

3

이 질문을 접했을 때 직접 주제를 조사하고있었습니다.

답변이 이미 받아 들여 졌다는 사실과 상관없이, 저는 여전히 완전성을위한 대안적인 접근 방식을 제공하고 싶습니다. 특히 Apache Ant는 매우 유연한 솔루션이었습니다.

기본적으로 모든 것은 Java 코드 ( 여기 참조)에서 호출 된 단일 Java 실행 태스크 ( 여기여기 참조)가있는 Ant 스크립트 파일로 요약됩니다 . 메소드 시작일 수있는이 Java 코드는 다시 시작해야하는 애플리케이션의 일부일 수 있습니다. 애플리케이션에는 Apache Ant 라이브러리 (jar)에 대한 종속성이 있어야합니다.

애플리케이션을 다시 시작해야 할 때마다 메서드 launch를 호출 하고 VM을 종료 해야합니다 . Ant Java 태스크에는 forkspawn 옵션이 있어야 합니다. 이 true로 설정되어 있어야합니다.

다음은 Ant 스크립트의 예입니다.

<project name="applaucher" default="launch" basedir=".">
<target name="launch">
    <java classname="package.MasinClass" fork="true" spawn="true">
        <jvmarg value="-splash:splash.jpg"/>
        <jvmarg value="-D other VM params"/>
        <classpath>
            <pathelement location="lib-1.jar" />
            ...
            <pathelement location="lib-n.jar" />
        </classpath>
    </java>
</target>
</project>

시작 메서드 의 코드는 다음과 같습니다.

public final void launch(final String antScriptFile) {
 /* configure Ant and execute the task */
   final File buildFile = new File(antScriptFile);
   final Project p = new Project();
   p.setUserProperty("ant.file", buildFile.getAbsolutePath());

   final DefaultLogger consoleLogger = new DefaultLogger();
   consoleLogger.setErrorPrintStream(System.err);
   consoleLogger.setOutputPrintStream(System.out);
   consoleLogger.setMessageOutputLevel(Project.MSG_INFO);
   p.addBuildListener(consoleLogger);

   try {
       p.fireBuildStarted();
       p.init();
       final ProjectHelper helper = ProjectHelper.getProjectHelper();
       p.addReference("ant.projectHelper", helper);
       helper.parse(p, buildFile);
       p.executeTarget(p.getDefaultTarget());
       p.fireBuildFinished(null);
   } catch (final BuildException e) {
       p.fireBuildFinished(e);
   }

   /* exit the current VM */
   System.exit(0);

}

여기서 매우 편리한 점은 초기 응용 프로그램 시작 및 다시 시작에 동일한 스크립트가 사용된다는 것입니다.


3

다른 답변에없는 정보를 추가하는 것뿐입니다.

procfs /proc/self/cmdline 를 사용할 수있는 경우

당신이 제공하는 환경에서 실행하는 경우 procfs의를 따라서있다 /proc(이것은 휴대용 해결책이 아니다 의미) 사용 가능한 파일 시스템을, 당신은 자바 읽을 수 있습니다 /proc/self/cmdline위해 다음과 같이 자체를 다시 시작에 :

public static void restart() throws IOException {
    new ProcessBuilder(getMyOwnCmdLine()).inheritIO().start();
}
public static String[] getMyOwnCmdLine() throws IOException {
    return readFirstLine("/proc/self/cmdline").split("\u0000");
}
public static String readFirstLine(final String filename) throws IOException {
    try (final BufferedReader in = new BufferedReader(new FileReader(filename))) {
        return in.readLine();
    }
}

사용 /proc/self/cmdline가능한 시스템 에서 이것은 아마도 Java에서 현재 Java 프로세스를 "다시 시작"하는 가장 우아한 방법 일 것입니다. JNI가 필요하지 않으며 경로와 물건을 추측 할 필요가 없습니다. java바이너리에 전달 된 모든 JVM 옵션도 처리합니다 . 명령 줄은 현재 JVM 프로세스 중 하나와 정확히 동일합니다.

오늘날 GNU / Linux (Android 포함)를 포함한 많은 UNIX 시스템에는 procfs가 있습니다. 그러나 FreeBSD와 같은 일부에서는 더 이상 사용되지 않으며 단계적으로 제거됩니다. Mac OS Xprocfs가 없다는 점에서 예외입니다 . Windows 에는 procfs 도 없습니다 . Cygwin에는 procfs가 있습니다. 있지만 Windows 시스템 호출 대신 Cygwin DLL을 사용하는 응용 프로그램에서만 볼 수 있고 Java는 Cygwin을 인식하지 못하기 때문에 Java에는 표시되지 않습니다.

사용하는 것을 잊지 마세요 ProcessBuilder.inheritIO()

기본값은 시작된 프로세스의 stdin/ stdout/ stderr(Java에서 System.in/ System.out/ 라고 함 System.err)가 현재 실행중인 프로세스가 새로 시작된 프로세스와 통신 할 수 있도록하는 파이프 로 설정되는 것입니다. 현재 프로세스를 다시 시작하려는 경우 원하는 것이 아닐 가능성큽니다 . 대신 stdin/ stdout/ stderr가 현재 VM의 것과 동일 하기를 원할 것 입니다. 이를 상속 이라고 합니다. 인스턴스 를 호출 inheritIO()하여 이를 수행 할 수 있습니다 ProcessBuilder.

Windows의 함정

restart()함수 의 빈번한 사용 사례 는 업데이트 후 애플리케이션을 다시 시작하는 것입니다. 마지막으로 Windows에서 이것을 시도했을 때 문제가있었습니다. 응용 프로그램의 .jar파일을 새 버전으로 덮어 쓰면 응용 프로그램 이 오작동하기 시작하고 .jar파일 에 대한 예외가 발생 합니다. 이것이 귀하의 사용 사례 인 경우를 대비하여 말하고 있습니다. 그때 나는 응용 프로그램을 배치 파일로 래핑하고 배치 파일에서 System.exit()쿼리 한 마법 반환 값을 사용하여 문제를 해결하고 배치 파일이 대신 응용 프로그램을 다시 시작하도록했습니다.


2

오래된 질문과 그 모든 것. 그러나 이것은 몇 가지 장점을 제공하는 또 다른 방법입니다.

Windows에서는 작업 스케줄러에게 앱을 다시 시작하도록 요청할 수 있습니다. 이는 앱이 다시 시작되기 전에 특정 시간을 기다리는 이점이 있습니다. 작업 관리자로 이동하여 작업을 삭제하면 반복이 중지됩니다.

SimpleDateFormat hhmm = new SimpleDateFormat("kk:mm");    
Calendar aCal = Calendar.getInstance(); 
aCal.add(Calendar.SECOND, 65);
String nextMinute = hhmm.format(aCal.getTime()); //Task Scheduler Doesn't accept seconds and won't do current minute.
String[] create = {"c:\\windows\\system32\\schtasks.exe", "/CREATE", "/F", "/TN", "RestartMyProg", "/SC", "ONCE", "/ST", nextMinute, "/TR", "java -jar c:\\my\\dev\\RestartTest.jar"};  
Process proc = Runtime.getRuntime().exec(create, null, null);
System.out.println("Exit Now");
try {Thread.sleep(1000);} catch (Exception e){} // just so you can see it better
System.exit(0);

2

Yoda의 ' 개선 된 '답변과 비슷하지만 추가로 개선되었습니다 (기능, 가독성 및 테스트 가능성 모두). 이제 실행하는 것이 안전하고 주어진 프로그램 인수의 양만큼 다시 시작됩니다.

  • JAVA_TOOL_OPTIONS옵션이 누적되지 않습니다 .
  • 메인 클래스를 자동으로 찾습니다.
  • 현재 stdout / stderr을 상속합니다.

public static void main(String[] args) throws Exception {
    if (args.length == 0)
        return;
    else
        args = Arrays.copyOf(args, args.length - 1);

    List<String> command = new ArrayList<>(32);
    appendJavaExecutable(command);
    appendVMArgs(command);
    appendClassPath(command);
    appendEntryPoint(command);
    appendArgs(command, args);

    System.out.println(command);
    try {
        new ProcessBuilder(command).inheritIO().start();
    } catch (IOException ex) {
        ex.printStackTrace();
    }
}

private static void appendJavaExecutable(List<String> cmd) {
    cmd.add(System.getProperty("java.home") + File.separator + "bin" + File.separator + "java");
}

private static void appendVMArgs(Collection<String> cmd) {
    Collection<String> vmArguments = ManagementFactory.getRuntimeMXBean().getInputArguments();

    String javaToolOptions = System.getenv("JAVA_TOOL_OPTIONS");
    if (javaToolOptions != null) {
        Collection<String> javaToolOptionsList = Arrays.asList(javaToolOptions.split(" "));
        vmArguments = new ArrayList<>(vmArguments);
        vmArguments.removeAll(javaToolOptionsList);
    }

    cmd.addAll(vmArguments);
}

private static void appendClassPath(List<String> cmd) {
    cmd.add("-cp");
    cmd.add(ManagementFactory.getRuntimeMXBean().getClassPath());
}

    private static void appendEntryPoint(List<String> cmd) {
    StackTraceElement[] stackTrace          = new Throwable().getStackTrace();
    StackTraceElement   stackTraceElement   = stackTrace[stackTrace.length - 1];
    String              fullyQualifiedClass = stackTraceElement.getClassName();
    String              entryMethod         = stackTraceElement.getMethodName();
    if (!entryMethod.equals("main"))
        throw new AssertionError("Entry point is not a 'main()': " + fullyQualifiedClass + '.' + entryMethod);

    cmd.add(fullyQualifiedClass);
}

private static void appendArgs(List<String> cmd, String[] args) {
    cmd.addAll(Arrays.asList(args));
}

V1.1 버그 픽스 : JAVA_TOOL_OPTIONS가 설정되지 않은 경우 널 포인터


예:

$ java -cp Temp.jar Temp a b c d e
[/usr/lib/jvm/java-8-openjdk-amd64/jre/bin/java, -cp, Temp.jar, Temp, a, b, c, d]
[/usr/lib/jvm/java-8-openjdk-amd64/jre/bin/java, -cp, Temp.jar, Temp, a, b, c]
[/usr/lib/jvm/java-8-openjdk-amd64/jre/bin/java, -cp, Temp.jar, Temp, a, b]
[/usr/lib/jvm/java-8-openjdk-amd64/jre/bin/java, -cp, Temp.jar, Temp, a]
[/usr/lib/jvm/java-8-openjdk-amd64/jre/bin/java, -cp, Temp.jar, Temp]
$

-13
System.err.println("Someone is Restarting me...");
setVisible(false);
try {
    Thread.sleep(600);
} catch (InterruptedException e1) {
    e1.printStackTrace();
}
setVisible(true);

응용 프로그램을 중지하고 싶지는 않지만 "다시 시작"하는 것이 좋습니다. 이를 위해 이것을 사용하고 잠자기 전과 보이지 않는 창 뒤에 "Reset"을 추가 할 수 있습니다.


4
사용자는 창을 숨기고 표시하는 것이 아니라 응용 프로그램을 다시 시작하도록 요청했습니다.
Amr Lotfy 2014 년
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.