在 JVM 运行时在 Windows 上创建符号链接

Creating a symlink on Windows during JVM runtime

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

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. 符号链接有何帮助?
    • 我已经进行了测试以确保符号链接允许我 "bypass" 260 个字符的限制。

这是我的问题:

  1. 是否有另一种方法可以在 Java 中创建符号链接,这样 Windows 将在命令 超过 260 个字符限制时起作用?
  2. 可以用SET代替mklink吗?
  3. 即使命令超过 260 个字符,也可以使用 java.nio.file 吗?

同样,我知道这是一个奇怪的问题。如有不妥,请说明。

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

一个实施说明:: 不要使用del删除符号链接,否则目标目录中的所有文件都将被删除。使用 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

Eddie B 的解决方案是在正确的轨道上,但是当 Java 尝试 运行 命令时,我一直收到错误。这是我的作品:

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 {

    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, flag, link, target;
    private List<String> commands = Arrays.asList(D_LINK, H_LINK, J_LINK, REM_LINK), symlinks;

    public WindowsSymlinkUtility() {
        this.command = this.flag = this.link = this.target = "";
        this.symlinks = new ArrayList<>();
    }

    /**
     * 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, 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) {
        if(!this.commands.contains(flag)) {
            System.err.printf("%s is not a valid command\n", flag);
            return false;
        }

        this.command = "mklink";
        this.flag = flag;
        this.link = link;
        this.target = target;

        if(this.runner() == 0) {
            this.symlinks.add(this.link);
            return true;
        }

        return false;
    }

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

        try {
            ProcessBuilder pb = new ProcessBuilder("cmd.exe", "/C", this.command, this.flag, this.link, this.target);
            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();
            System.out.println(message);
        } catch(Exception e) {
            e.printStackTrace();
            System.err.println(e.getMessage());
            System.err.println(message);
        }

        return exitVal;
    }

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

}