5

我想使用 Java 的 Runtime.exec 方法执行一些 sql 脚本。我打算调用 mysql.exe / mysql.sh 并将脚本文件重定向到这个进程。从命令提示符我可以运行命令

<mysqInstallDir\/bin\mysql.exe -u <userName> -p <password> < scripts\create_tables.sql

我可以使用 Runtime.exec 调用 mysql.exe 但如何将数据从 sql 文件重定向到 mysql.exe ?


我在http://www.javaworld.com/javaworld/jw-12-2000/jw-1229-traps.html?page=4中阅读了这篇文章,并使用了 StreamGobbler 机制来获取错误和输出流。那里没问题。问题在于使用 BufferedReader 读取文件“scripts\create_tables.sql”并将内容传递给进程的输出流。我期待进程将数据传递给 mysql.exe。但我看到只有第一行是从这个 sql 文件中读取的。

OutputStream outputstream = proc.getOutputStream();
OutputStreamWriter outputstreamwriter = new OutputStreamWriter(outputstream);
BufferedWriter bufferedwriter = new BufferedWriter(outputstreamwriter);
  while ( (line = br.readLine()) != null)
  {
bufferedwriter.write(line);
bufferedwriter.flush();
System.out.println(line);
  }
  bufferedwriter.flush();
  bufferedwriter.close();
  proc.waitFor() 

当我这样做时,我看到只有 create_tables.sql 中的第一行被执行。该进程的退出代码为 0 并且没有其他错误或输出。

4

2 回答 2

6

Exec 向您返回一个 Process 对象。

Process 具有 getInputStream 和 getOutputStream 方法。

只需使用它们来获取输入流并开始将字节推入其中。不要忘记读取输出流,否则进程可能会阻塞。

于 2010-04-01T20:25:54.797 回答
1

重定向是 OS shell/cmd 环境的一项功能。为了正确调用它们,我们应该使用 Runtime.exec(String[]) 而不是 Runtime.exec(String)。这是代码。

public Result executeCmd(String[] cmds, boolean waitForResult)
{
    Result result = new Result();
    result.output = "";
    try
    {
        for(int i=0;i<cmds.length;i++)
        {
            System.out.println("CMD["+i+"]::"+cmds[i]);
        }
        System.out.println("");
        Process process = null;
        if(cmds.length > 1)
            process=Runtime.getRuntime().exec(cmds);
        else
            process=Runtime.getRuntime().exec(cmds[0]);
        if (waitForResult)
        {
            StreamGobbler errordataReader = new StreamGobbler(process
                    .getErrorStream(), "ERROR");

            StreamGobbler outputdataReader = new StreamGobbler(process
                    .getInputStream(), "OUTPUT");

            errordataReader.start();
            outputdataReader.start();

            int exitVal = process.waitFor();
            errordataReader.join();
            outputdataReader.join();
            result.returnCode = exitVal;
            result.output = outputdataReader.output;
            result.error = errordataReader.output;
        }
    }
    catch (Exception exp)
    {
        result.exp = exp;
        result.returnCode = -1;
    }
    return result;
}

并使用调用此方法

Result result = executeCmd(cmds, true);

在哪里

CMD[0]::cmd
CMD[1]::/c
CMD[2]::.\mysql\bin\mysql --host=<hostname> --port=<portNum> -u <userName>  < .\scripts\create_tables.sql
于 2010-05-06T21:46:43.307 回答