在Java中快速实现端口转发

Dan*_*iel 7 java portforwarding

我构建了一个打开ServerSocket的简单应用程序,在连接时,它将自身连接到远程计算机上的另一个服务器套接字.为了实现端口转发,我使用两个线程,一个从本地输入流读取并流到远程套接字输出流,反之亦然.

实现感觉有点无法实现,所以我问你是否知道更好的实现策略,或者甚至有一些代码可以以高效的方式实现.

PS:我知道我可以在Linux上使用IPTables,但这必须在Windows上运行.

PPS:如果你发布这个简单任务的实现,我将创建一个基准来测试所有给定的实现.对于许多小型(~100字节)包和稳定数据流,解决方案应该是快速的.

我当前的实现是这样的(在每个方向的两个线程中执行):

public static void route(InputStream inputStream, OutputStream outputStream) throws IOException {
    byte[] buffer = new byte[65536];
    while( true ) {
        // Read one byte to block
        int b = inputStream.read();
        if( b == - 1 ) {
            log.info("No data available anymore. Closing stream.");
            inputStream.close();
            outputStream.close();
            return;
        }
        buffer[0] = (byte)b;
        // Read remaining available bytes
        b = inputStream.read(buffer, 1, Math.min(inputStream.available(), 65535));
        if( b == - 1 ) {
            log.info("No data available anymore. Closing stream.");
            inputStream.close();
            outputStream.close();
            return;
        }
        outputStream.write(buffer, 0, b+1);
    }
}
Run Code Online (Sandbox Code Playgroud)

Boz*_*zho 9

看看tcpmon.其目的是监视tcp数据,但它也转发到不同的主机/端口.

这里是一些代码从拍摄进行端口转发(这不是英文,所以我贴的代码,而不是给一个链接到这本书的电子版):

  • 提供的链接似乎被打破了.在谷歌找到Github链接代码https://github.com/cartmanez/port-forward/blob/master/src/main/java/ed/port/forward/或http://www.nakov.com/books /inetjava/source-code-html/Chapter-1-Sockets/1.4-TCP-Sockets/TCPForwardServer.java.html (2认同)

Ste*_*n C 7

几点意见:

  • 在循环开始时读取的一个字节对提高性能没有任何作用.事实上可能相反.

  • 呼叫inputStream.available()是不必要的.您应该尝试读取"缓冲区大小"字符.readSocket流上的A 将返回当前可用的字符数,但在缓冲区已满之前不会阻塞.(我在javadocs中找不到任何说明这一点的内容,但我确信情况确实如此.很多事情都会表现不佳......或者如果read在缓冲区满了之前被阻止......

  • 正如@ user479257指出的那样,你应该通过使用java.nio和读写ByteBuffers来获得更好的吞吐量.这将减少JVM中发生的数据复制量.

  • 如果读取,写入或关闭操作引发异常,则您的方法将泄漏套接字流.您应该使用try ... finally如下方法来确保无论发生什么情况都始终关闭流.


public static void route(InputStream inputStream, OutputStream outputStream) 
throws IOException {
    byte[] buffer = new byte[65536];
    try {
        while( true ) {
            ...
            b = inputStream.read(...);
            if( b == - 1 ) {
                log.info("No data available anymore. Closing stream.");
                return;
            }
            outputStream.write(buffer, 0, b+1);
        }
    } finally {
        try { inputStream.close();} catch (IOException ex) { /* ignore */ }
        try { outputStream.close();} catch (IOException ex) { /* ignore */ }
    }
}
Run Code Online (Sandbox Code Playgroud)

  • @Daniel - 找到你所有的bug都不是我的**工作.:-) (4认同)