java - 在 JVM 运行时在 Windows 上创建符号链接(symbolic link)

标签 java windows windows-7 cmd

有点奇怪的问题,但我在使用 mklink 创建符号链接(symbolic link)时遇到了问题在 Windows 7 上。由于使用 cmd.exe 时存在 260 个字符的限制,我正在做一些奇怪的事情。通过使用 Process 在我的 Java 源代码中创建符号链接(symbolic link).由于我不能完全解释它,这里是代码:

import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Arrays;

public class WindowsSymlinkUtility {

    private List<String> command, currentSymlinks;

    public WindowsSymlinkUtility() {
        this.command = this.currentSymlinks = new ArrayList<String>();
        this.command.add("cmd.exe");
        this.command.add("/C");
    }

    /**
     * Automatically creates a directory junction
     * @param String link - the path and name of the symlink
     * @param String target - the directory to point the symlink to
     * @return boolean
     * @see http://ss64.com/nt/mklink.html
     */
    public boolean createSymlink(String link, String target) {
        return createSymlink("\\J", link, target);
    }

    /**
     *
     * @param String flag - the flag for mklink
     * @param String link - the path and name of the symlink
     * @param String target - the directory to point the symlink to
     * @return boolean
     * @see http://ss64.com/nt/mklink.html
     */
    public boolean createSymlink(String flag, String link, String target) {
        this.command.clear();
        this.command.add("mklink");
        this.command.add(flag);
        this.command.add(link);
        this.command.add(target);
        this.currentSymlinks.add(link);

        return this.runner() == 0;
    }

    public boolean removeSymlink(String link) {
        this.command.clear();
        this.command.add("RD");
        this.command.add(link);

        if(this.runner() != 0) {
            this.command.clear();
            this.command.add("DEL");
            this.command.add(link);
        } else {
            return true;
        }

        return this.runner() == 0;
    }

    public boolean removeAllSymlinks() {
        for(String link : this.currentSymlinks) {
            if(!this.removeSymlink(link)) {
                return false;
            }
        }

        return true;
    }

    /**
     * Leave for debugging purposes
     * @return String
     */
    public String getCurrentCommand() {
        String cmd = "";
        for(String part : this.command) {
            cmd += part + " ";
        }

        return cmd;
    }

    private int runner() {
        Process process = null;
        String message = null;
        BufferedInputStream bis = null;
        int exitVal = -1;
        StringBuilder strBuff = new StringBuilder();

        try {
            if(this.command.size() < 1) throw new Exception("Length of Windows command cannot be zero");

            ProcessBuilder pb = new ProcessBuilder(this.command);
            Map<String, String> envVars = pb.environment();

            pb.directory();
            pb.redirectErrorStream(true);
            process = pb.start();
            bis = new BufferedInputStream(process.getInputStream());
            byte[] bArr = new byte[2048];
            while (bis.read(bArr) != -1) {
                strBuff.append(new String(bArr).trim());
                bArr = new byte[2048];
            }

            exitVal = process.waitFor();
            message = strBuff.toString();
        } catch(Exception e) {
            e.printStackTrace();
            System.err.println(e.getMessage());
            System.err.println(message);
        }

        return exitVal;
    }

    public static void main(String[] args) {
        WindowsSymlinkUtility foo = new WindowsSymlinkUtility();
        foo.createSymlink("%TEMP%\\foo", "C:\\Users\\djthomps\\Downloads");
    }

}

我得到的错误:

java.io.IOException: Cannot run program "mklink": CreateProcess error=2, The system cannot find the file specified
        at java.lang.ProcessBuilder.start(Unknown Source)
        at WindowsSymlinkUtility.runner(WindowsSymlinkUtility.java:113)
        at WindowsSymlinkUtility.createSymlink(WindowsSymlinkUtility.java:56)
        at WindowsSymlinkUtility.createSymlink(WindowsSymlinkUtility.java:37)
        at WindowsSymlinkUtility.main(WindowsSymlinkUtility.java:134)
Caused by: java.io.IOException: CreateProcess error=2, The system cannot find the file specified
        at java.lang.ProcessImpl.create(Native Method)
        at java.lang.ProcessImpl.<init>(Unknown Source)
        at java.lang.ProcessImpl.start(Unknown Source)
        ... 5 more
Cannot run program "mklink": CreateProcess error=2, The system cannot find the file specified
null

您可能有的一些问题:

  1. 你为什么要这样做?
    • 因为相关文件和文件夹深深嵌套在文件系统中,所以完整命令的长度超过 260 个字符。
  2. 符号链接(symbolic link)有何帮助?
    • 我已进行测试以确保符号链接(symbolic link)允许我“绕过”260 个字符的限制。

这是我的问题:

  1. 是否有另一种方法可以在 Java 中创建符号链接(symbolic link),以便 Windows 在命令超过 260 个字符限制时执行?
  2. 可以SET用于代替 mklink
  3. 是否可以使用 java.nio.file即使命令运行超过 260 个字符也能做到这一点?

同样,我知道这是一个奇怪的问题。如果有什么不对,请要求澄清。

最佳答案

我对你的程序做了一些修改,只是为了提供一个工作示例……本质上,问题是你没有连接变量并将它们作为一个参数传递给 cmd。 .

一个实现说明::不要使用del删除符号链接(symbolic link),否则目标目录中的所有文件都将被删除。使用 rmdir ,这是我为后代添加的。

/**
 * @author Edward Beckett :: <Edward@EdwardBeckett.com>
 * @since :: 7/21/2015
 */
import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.Arrays;
import java.util.List;

public class WindowsSymlinkUtility {

    public static final String D_LINK = "/D";
    public static final String H_LINK = "/H";
    public static final String J_LINK = "/J";
    public static final String REM_LINK = "rmdir";
    private String command = "";
    private String link = "";
    private String target = "";

    private List<String> commands = Arrays.asList( D_LINK, H_LINK, J_LINK, REM_LINK );

    public void createSymlink( String command, String link, String target ) {
        this.command = command;
        this.link = link;
        this.target = target;

        if( !commands.contains( command ) ) {
            System.out.println( command + " Is not a valid command \n " );
            return;
        }
        runner();
    }


    private void runner() {

        try {

            String[] values = { "CMD", "/C", "mklink", this.command, this.link, this.target };
            ProcessBuilder builder = new ProcessBuilder( values );
            builder.directory( new File( this.link ) );
            Process process = builder.start();
            InputStream is = process.getInputStream();
            InputStreamReader isr = new InputStreamReader( is );
            BufferedReader br = new BufferedReader( isr );
            String line;
            System.out.printf( "Output of running %s is:\n",
                Arrays.toString( values ) );
            while( ( line = br.readLine() ) != null ) {
                System.out.println( line );
                int exitValue = process.waitFor();
                System.out.println( "\n\nExit Value is " + exitValue );
            }
        } catch( InterruptedException | IOException e ) {
            e.printStackTrace();
        }
    }
        public static void main( String[] args ) {
        ( new WindowsSymlinkUtility() ).createSymlink( J_LINK, "C:\\Foo", "C:\\Temp" );
    }

}

output

Output of running [CMD, /C, mklink, /J, C:\Foo, C:\Temp] is:
Junction created for C:\Foo <<===>> C:\Temp
Exit Value is 0

关于java - 在 JVM 运行时在 Windows 上创建符号链接(symbolic link),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31551498/

相关文章:

java - JAXB 将 1 个 XML 标记映射到 2 个变量

php - 如何提高 IIS PHP 服务器上的读取文件速度?

delphi - (保存对话框) 如何在 Vista/Win7 中更改文件过滤器时自动更改文件扩展名?

python - 在 win 7(64 位)上安装 scrapy

apache - 添加虚拟主机失败: Access Forbidden Error 403 (XAMPP) (Windows 7)

java - Selenium WebDriver wicket FindElement 挑战

java - Android - 试图实例化一个不是 fragment 的类

Java RMI : exception: java. security.AccessControlException:访问被拒绝 ("java.net.SocketPermission" "127.0.0.1:1099" "connect,resolve")

windows - 是否可以用代理 dll 替换系统 dll?

c++ - ofstream operator<< 访问冲突错误