자바에서 배치파일(.batch)뿐만 아니라 DOS 커맨드, 실행파일(.exe) 파일도 실행 가능하다
1. java에서 배치파일 실행하기 (DOS 명령어, exe 실행파일)
import java.io.BufferedReader;
import java.io.InputStreamReader;
public class ManucalDBBackup{
try {
Process p =Runtime.getRuntime().exec("C:\\Tomcat 5.5\\webapps\\TEST.bat");
BufferedReader br = new BufferedReader(new InputStreamReader(p.getInputStream()));
String line = null;
while ((line = br.readLine()) != null) {
System.out.println(line);
}
} catch (Exception e) {
System.err.println(e);
}
}
자바소스를 실행하면, 일반 명령어나 배치파일의 실행결과가 consol에 뿌려지면서 실행이 된다.ㅋ
2. java에서 ORACLE DB 백업(exp 명령어) 실행하기
[오라클 DB 백업 배치파일 내용]
@echo off
d:
cd D:\BackUp\aBisManager\backupDB
setlocal
set date2=%date:-=%
set time2=%time: =0%
set time3=%time2:~0,2%%time2:~3,2%%time:~6,2%
set DS=%date2%_%time3%
mkdir %DS%
exp userid=userid/userpwd file='D:\BackUp\%DS%\TEST.dmp'
log='D:\BackUp\%DS%\TEST.log' tables=(userid.xx, userid.yy) statistics=none
endlocal
@echo on
로컬에서..
오라클 백업 명령 배치파일을 그냥 실행하면 되는데
웹상(was : 톰캣)에서..
java를 통해 호출하면 계속 hang이 걸려버린다.. -ㅇ-;;
이 상태에서 톰캣을 꺼버리면 바로 백업이 완료되는 시츄에이션이 계속 되고..
톰캣을 안 끄고
걍 시간을 두고 보니 약 한시간 반 뒤에 백업파일이 생성되는 황당함ㅋ
이유가 무엇일까?
많은 국내 사이트를 돌아다녀봤는데.. 모두들 비슷한 문제를 겪고 있었다.
모두들 엉뚱한 소리만 하고,,
눈을 돌려.. 해외로~
해외에도 동일한 문제로 고민을 하는 많은 사람들.
... 하지만 해답을 제시하는 사람 몇몇 발견!!!
찾았다~!! ㅠㅠ
원인은 exp 명령어가 실행되면 output을 만들게 되는데 이를 (표준)출력 해주지 않으면 버퍼에 가득차 버려서 버퍼가 비워지길 기다리게 되면서 hang상태가 되는 것.
결론은 java에서 exp배치파일을 실행하면서 동시에 thread를 이용하여 표준출력을 해주는 것
소스 :
http://www.objects.com.au/java/examples/io/examples/ConsoleExecExample.java
http://www.objects.com.au/java/examples/io/AutoReader.java
출처 - http://blog.daum.net/nsh777/5494230
CmdExecutor.java
출처 - http://blog.daum.net/istae0430/246
아시다시피 위 함수는 process 를 실행시키기 위한 것입니다.
그런데 만약 프로세스의 수행이 끝이 날때까지 기다려야 한다면 어떻게 할까요?
일반적으로 아래와 같이 waitFor() 를 호출해 줍니다.
1 2 | Process process = Runtime.getRuntime().exec(cmd);
process.waitFor();
|
문제는 waitFor() 가 영원히 끝나지 않는 경우가 발생한다는 것입니다.분명히 cmd 에 해당하는 프로세스가 종료되어야 하는데, 이상하게도 끝나지 않는다는 이야깁니다.
원인은 stream 의 버퍼가 비워지지 않았기 때문입니다.
이를 비워줌으로써 문제를 해결할 수 있습니다.
그럼 어떻게 비워주면 될까요?
waitFor() 함수를 호출하는 순간, Thread 는 process 가 종료될때까지 block 이 됩니다.
다시 말해서 해당 stream 을 비워주는 작업을 별도의 thread 에서 수행해야 한다는 이야기죠.
별도 Thread 를 만들고, 이 Thread 에서 process.getErrorStream(); process.getInputStream(); 를 통해 Stream 을 얻어온 뒤, 계속해서 read 해 줌으로써 스트림을 비워야 한다는 것입니다.만약, process 수행결과 얻어지는 출력을 필요로 하지 않는다면 코드는 훨씬 간단해 집니다.
아래처럼 애초에 stream 을 close 해 버리면 되거든요. :)1 2 3 4 5 | Process process = Runtime.getRuntime().exec(cmd);
process.getErrorStream().close();
process.getInputStream().close();
process.getOutputStream().close();
process.waitFor();
|
문제없이 process 가 실행 후, waitFor() 에서 빠져가나는 것을 확인하실 수 있습니다.
출처 - http://crystalcube.co.kr/69
자바에서 Runtime.getRuntime().exec() 를 사용하여 windows 명령어를 실행하는 방법.
여기서 command는 실행 명령어 이다.
예 : move c:\data\source.txt c:\data\target.txt <- 윈도우용
예 : mv /data/source.txt /data/target.txt <- 리눅스용
Process process;
String osName = System.getProperty("os.name");
String[] cmd;
if(osName.toLowerCase().startsWith("window")) {
cmd = new String[] { "cmd.exe", "/y", "/c", command };
} else {
cmd = new String[] { "/bin/sh", "-c", command };
}
process = Runtime.getRuntime().exec(cmd);
return process.waitFor();
출처 - http://blog.daum.net/lkflower/17224498
시스템 명령어 테스트 샘플 소스
------------------------------------------------------------------------
import java.io.*;
public class Test
{
public static void main ( String [] args ) throws IOException, InterruptedException
{
String servlet_path = "C:\resin1.2.3\doc\WEB-INF\classes";
String sourceDir = System.getProperty ( "user.dir" ) + File.separator + "source" + File.separator + "*.java";
String comstr = "javac -d " + servlet_path + " " + sourceDir;
Process proc = Runtime.getRuntime ( ).exec ( comstr );
proc.waitFor ( );
//주의: 윈도우의 경우 servlet_path는 구분문자가 \ 인데, sourceDir는 로 나온다.
//배열을 안쓰고 문자열로 직접하는게 잘될 때가 있다.
}
}
------------------------------------------------------------------------
import java.io.*;
public class Test
{
public static void main ( String [] args )
{
try
{
String comstr = "chmod 777 test.txt";
String [] command = { "/bin/bsh", "-c", comstr };
Process proc = Runtime.getRuntime ().exec ( command );
proc.waitFor ( );
if ( proc.exitValue ( ) != 0 )
out.println ( "에러..." );
proc.destroy ( );
}
catch ( Exception e ) { System.out.println ( e ); }
}
}
------------------------------------------------------------------------
import java.io.*;
public class Test
{
public static void main ( String [] args )
{
try
{
String comstr = "ls -l";
String [] command = { "/bin/bsh", "-c", comstr }; //리눅스에서
//String [] command = { "/bin/csh", "-c", comstr }; // Unix에서
//String [] command = { "command.com", "/c", comstr }; // 윈도우에서
Process proc = Runtime.getRuntime ( ).exec ( command );
proc.waitFor ( );
if ( proc.exitValue ( ) != 0 )
{
BufferedReader err = new BufferedReader ( new InputStreamReader ( proc.getErrorStream ( ) ) );
while ( err.ready ( ) )
System.out.println ( err.readLine ( ) );
err.close ( );
}
else
{
BufferedReader out = new BufferedReader ( new InputStreamReader ( proc.getInputStream ( ) ) );
while ( out.ready ( ) )
System.out.println ( out.readLine ( ) );
out.close ( );
}
proc.destroy ( );
}
catch ( Exception e ) { System.out.println ( e ); }
}
}
------------------------------------------------------------------------
2. 리눅스 시스템 명령어 실행 결과를 받아오기
------------------------------------------------------------------------
String command = "ls";
String data = null;
StringBuffer strbuf = new StringBuffer ( );
BufferedReader br = null;
try
{ Process proc = Runtime.getRuntime ( ).exec ( command );
br = new BufferedReader ( new InputStreamReader ( proc.getInputStream ( ) ) );
String line;
while ( ( line = br.readLine ( ) ) != null )
strbuf.append ( line + " " );
br.close ( );
}
catch ( Exception e ) { out.println ( e ); }
data = strbuf.toString ( );
------------------------------------------------------------------------
3. 상호작용(interactive) 명령어의 실행
프로세스의 표준 출력을 자바 프로그램에서는 p.getInputStream 메소드를 이용하여 얻고,
p.getOutputStream 메소드를 이용하여 프로세스의 표준 입력 장치에 쓰게 됩니다.
이 때, 한 가지 주의할 사항은 표준 출력 스트림에 대답을 쓴(write) 후, 꼭 flush 또는 close 메소드를 이용하여
표준 출력 스트림을 비워(flush) 주어야 합니다.
------------------------------------------------------------------------
import java.io.*;
public class InteractiveProcessTest
{
public static void main ( String [] args)
{
// 반드시 배열로 해야한다. 문자열로 하면 단지 전체를 하나의 프로그램으로 인식한다.
String [] command = { "ping", "203.252.134.126" };
try
{
Process p = Runtime.getRuntime ( ).exec ( command );
byte [] msg = new byte [128];
int len;
while ( ( len = p.getInputStream ().read ( msg ) ) > 0 )
System.out.print ( new String( msg, 0, len ) );
String rs = " ";
byte [] rb = new byte [] { (byte)' ' } ; //rs.getBytes();
OutputStream os = p.getOutputStream ( );
os.write ( rb );
os.close ( );
}
catch ( Exception e ) { e.printStackTrace ( ); }
}
}
------------------------------------------------------------------------
결과 :
C:>java InteractiveProcessTest
Pinging 203.252.134.126 with 32 bytes of data:
Reply from 203.252.134.126: bytes=32 time<10ms TTL=128
Reply from 203.252.134.126: bytes=32 time<10ms TTL=128
4. 명령어 문자열의 안정적인 작성 예
------------------------------------------------------------------------
// Unix에서
String [] command = { "/bin/csh", "-c", "finger | grep -v 639" };
// 윈도우에서
String [] command = { "command.com", "/c", "finger | grep -v 639" };
//getInputStream ( ) 스트림으로 결과값을 받아올 때 형변환해서 받을 수 없다.
String data = "";
BufferedReader out = new BufferedReader ( new InputStreamReader ( proc.getInputStream ( ) ) );
data = out.readLine ( ); //int count = Integer.parseInt ( out.readLine ( ) );는 안된다.
out.close ( );
int count = Integer.parseInt ( data );
------------------------------------------------------------------------
5. 프로세스 함수들
5.1 프로세스의 생성 및 종료
------------------------------------------------------------------------
- “Runtime runtime = Runtime.getRuntime();”: 런타임 객체를 생성합니다.
- “Process p = runtime.exec(“프로그램경로명”);”: exec 메소드를 이용하여 프로세스를 생성합니다.
- “p.waitFor();”: 자식 프로세스가 종료될 때까지 기다립니다.
- “p.destroy();”: 부모 프로세스에서 자식 프로세스를 강제로 종료시킵니다.
- “System.exit(0);”: 부모 프로세스만 종료되고 자식 프로세스는 계속 실행됩니다.
------------------------------------------------------------------------
5.2 Runtime 클래스가 제공해 주는 주요 메소드
------------------------------------------------------------------------
- public static Runtime getRuntime(): 현재 실행되고 있는 자바 애플리케이션과 관련된 런타임 객체를 리턴해 줍니다.
- public void exit(int status): 현재 자바 가상머신을 종료합니다. status 매개변수는 종료시의 상태값을 나타내며,
일반적으로 0 이외의 값은 비정상적으로 종료되었음을 의미합니다.
- public Process exec(String command) throws IOException: 주어진 명령어를 독립된 프로세스로 실행시켜 줍니다.
exec(command, null)와 같이 실행시킨 것과 같습니다.
- public Process exec(String command, String envp[]) throws IOException:
주어진 명령어를 주어진 환경을 갖는 독립된 프로세스로 실행시켜 줍니다.
이 메소드는 명령어 문자열을 토큰으로 나누어 이 토큰들을 포함하고 있는 cmdarray라는 새로운 배열을 생성합니다.
그리고 나서 exec(cmdarray, envp)을 호출합니다.
- public Process exec(String cmdarray[]) throws IOException:
주어진 문자열 배열에 있는 명령어와 매개변수를 이용하여 독립된 프로세스로 실행시켜 줍니다.
exec(cmdarray, null)을 호출합니다.
- public Process exec(String cmdarray[], String envp[]) throws IOException:
주어진 문자열 배열에 있는 명령어와 매개변수를 이용하여 주어진 환경을 갖는 독립된 프로세스로 실행시켜 줍니다.
문자열 배열 cmdarray에는 명령어와 명령행 인자들을 나타내고 있습니다.
- public native long freeMemory(): 시스템에 남아있는 메모리의 양을 얻습니다.
이 값은 항상 totalMemory() 메소드에 의해 얻어지는 값보다 작습니다.
- public native long totalMemory(): 자바 가상머신의 최대 메모리 크기를 얻습니다.
------------------------------------------------------------------------
5.3 Process 클래스가 제공해 주는 주요 메소드
------------------------------------------------------------------------
- public abstract OutputStream getOutputStream(): 자식 프로세스의 출력 스트림을 얻습니다.
- public abstract InputStream getInputStream(): 자식 프로세스의 입력 스트림을 얻습니다.
- public abstract InputStream getErrorStream(): 자식 프로세스의 에러 스트림을 얻습니다.
- public abstract int waitFor() throws InterruptedException: 자식 프로세스가 종료될 때까지 기다립니다.
- public abstract int exitValue(): 자식 프로세스가 종료할 때의 상태값을 얻습니다.
- public abstract void destroy(): 자식 프로세스를 강제로 종료시킵니다.
------------------------------------------------------------------------
6. 메모리 관리
------------------------------------------------------------------------
public long totalMemory ( ) : 현재 할당받은 총 시스템 메모리 바이트 갯수.
public long freeMemory ( ) : 현재 남아 있는 자유 메모리 바이트 갯수(근사치).
Runtime rt = Runtime.getRuntime ( );
System.out.println ( rt.totalMemory ( ) );
System.out.println ( rt.freeMemory ( ) );
------------------------------------------------------------------------
나. 시스템 정보
static PrintStream err - The "standard" error output stream.
static InputStream in - The "standard" input stream.
static PrintStream out - The "standard" output stream.
static void arraycopy (Object src, int src_position, Object dst, int dst_position, int length)
static long currentTimeMillis () - Returns the current time in milliseconds.
static void exit (int status) - Terminates the currently running Java Virtual Machine.
static void gc () - Runs the garbage collector.
static String getenv (String name) - Deprecated from getProperty method
static Properties getProperties() - Determines the current system properties.
static String getProperty(String key) - Gets the system property indicated by the specified key.
static String getProperty(String key, String def) - Gets the system property indicated by the specified key.
static SecurityManager getSecurityManager() - Gets the system security interface.
static int identityHashCode(Object x) - Returns the same hashcode for the given object as would be returned by the default method hashCode(), whether or not the given object's class overrides hashCode().
static void load(String filename) - Loads a code file with the specified filename from the local file system as a dynamic library.
static void loadLibrary(String libname) - Loads the system library specified by the libname argument.
static String mapLibraryName(String libname) - Maps a library name into a platform-specific string representing a native library.
static void runFinalization() - Runs the finalization methods of any objects pending finalization.
static void runFinalizersOnExit(boolean value) - Deprecated. This method is inherently unsafe. It may result in finalizers being called on live objects while other threads are concurrently manipulating those objects, resulting in erratic behavior or deadlock.
static void setErr(PrintStream err) - Reassigns the "standard" error output stream.
static void setIn(InputStream in) - Reassigns the "standard" input stream.
static void setOut(PrintStream out) - Reassigns the "standard" output stream.
static void setProperties(Properties props) - Sets the system properties to the Properties argument.
static String setProperty(String key, String value) - Sets the system property indicated by the specified key.
static void setSecurityManager(SecurityManager s)
String System.getProperty ( String key ): key에 해당하는 시스템정보를 가져온다.
String System.setProperty ( String key, String value )
Properties System.getProperties ( ): 시스템 정보를 몽땅 갖고 온다.
System.setProperties ( Properties props )
사용법
------------------------------------------------------------------------
String currentDirectory = System.getProperty ( "user.dir" );
------------------------------------------------------------------------
Key값
------------------------------------------------------------------------
// 일반적으로, 보안에 걸리지 않는 표준 시스템 프로퍼티
"java.version", // 자바의 버전 ( "1.0.2", "1.1", "1.1.3", "1.2", ...)
"java.class.version", // 자바 클래스 화일 포맷 버전
"java.vendor", // 자바 가상 머쉰 판매 회사
"java.vendor.url", // 자바 가상 머쉰 판매 회사의 URL
"os.name", // 운영체제 이름 ( "Windows 95", "Windows NT", "MacOS",
// "Linux", "Solaris", "OS/2", "OSF1", ... )
"os.version", // 운영체제 버전
"os.arch", // 컴퓨터 기종 ( "x86", "sparc", ... )
"line.separator", // 시스템에서의 행 분리 문자열 ( "
", " ", " " )
"file.separator", // 화일 경로 이름내의 디렉토리 구분 문자 ("", "/")
"path.separator", // 화일 경로 이름 리스트의 구분 문자 (";", ":")
// 일반적으로, 보안에 의한 접근 제한을 받는 표준 시스템 프로퍼티 (애플릿의 경우)
"java.home", // JDK 설치 디렉토리
"java.class.path", // 패키지가 있는 디렉토리 리스트
// (일반적으로, CLASSPATH 환경변수값에 의해 영향 받음)
"user.name", // 사용자의 등록 이름
"user.home", // 사용자의 홈 디렉토리
"user.dir", // 사용자의 현재 작업 디렉토리(파일을 뺀 전체 경로)
// JDK 1.1에서 비공식적으로 추가된 지역 정보 시스템 프로퍼티
// 일반적으로, 보안에 의한 접근 제한을 받음 (애플릿의 경우)
"file.encoding", // 사용자의 디폴트 문자 인코딩 이름
"user.language", // 사용자의 언어
"user.region", // 사용자의 거주 지역 (보통, 국가)
"user.timezone", // 사용자 지역의 표준 시간대 이름
// 웹 브라우저 정보 비표준 시스템 프로퍼티
"browser", // 웹 브라우저 이름
"browser.version", // 웹 브라우저 버전
"browser.vendor", // 웹 브라우저 회사
------------------------------------------------------------------------
다. 기타
//아래는 자바서비스넷의 이원영님 자료
-----------------------------------------------------------------------------------
import java.io.*;
public class Test
{
public static void main(String[] args) throws Exception
{
Runtime rt = Runtime.getRuntime();
StringBuffer cmd = new StringBuffer();
for(int i=0;i cmd.append(args[i] + " ");
Process p = rt.exec(cmd.toString() );
PrintWriter keyboard = new PrintWriter(p.getOutputStream());
//keyboard.println( ... );
p.waitFor(); // wait for process finishing.
BufferedReader out = new BufferedReader(new InputStreamReader(p.getInputStream()));
while ( out.ready() )
System.out.println(out.readLine());
BufferedReader err = new BufferedReader(new InputStreamReader(p.getErrorStream()));
while ( err.ready() )
System.out.println(err.readLine());
p.destroy(); //confirm
}
}
-----------------------------------------------------------------------------------
javaservice:/home/java/tmp$ javac Test.java
javaservice:/home/java/tmp$ java Test ls -alF /var
그러나 windows환경의 command.com의 경우, "command.com /c dir" 등의 방법으로
될 것으로 예상됐지만, "command.com /c dir" 의 명령이 보통의 DOS창에선 바로 exit으로
이어지나, Runtime.exec("command.com /c dir") 로 호출했을 경우, Process Thread가
끝나질 않는 군요... 결국, p.waitFor()에서 blocking 현상이 일어나네요.
어쨌거나 Unix환경에선 잘 되네요.
//아래는 이원영의 참고자료
-----------------------------------------------------------------------------------
String[] command = {"a.exe"};
Process p = Runtime.getRuntime().exec(command);
OutputStream os = p.getOutputStream();
DataOutputStream dos = new DataOutputStream(os);
dos.writeInt(...);
dos.flush();
os.close();
p.waitFor();
int rc = p.exitValue();
System.out.println("return code: "+ rc);
if(rc!=0)
{
InputStream errStream = p.getErrorStream();
DataInputStream errDataStream = new DataInputStream(errStream);
String s;
s = errDataStream.readLine();
while(s !=null)
{
System.out.println(s);
s = errDataStream.readLine();
}
}
InputStream is = p.getInputStream();
DataInputStream dis = new DataInputStream(is);
dis.readLine();
is.close();
출처 - http://alethe.egloos.com/1853800