Java在通过Windows远程桌面(tsclient)写入时创建巨大的文件
我们的一个客户端报告了一个非常奇怪的问题,当我们的 Swing 应用程序通过 Windows 远程桌面(应用程序托管在用户连接的终端服务器上)将文件写入用户本地计算机时。
流程为:
- 用户通过远程桌面登录并运行应用程序(将其作为“本地资源”包含在内)
C:\
- 在工作时,他们将数据从数据库导出到文件中
- 用户选择要导出的数据
- 用户在其本地计算机上选择目标文件,如
\\tsclient\C\Temp\TestFile.txt
- 文件可能很大,因此每批从数据库读取 1000 行并写入文件
- 在第二批中,当Java打开文件并再次写入它时,一些非常奇怪的事情开始发生!
- 文件大小迅速增加,并在大约 2 GB 处停止
- 然后数据继续写入文件
我不确定这是核心Java库,远程桌面实现还是组合中的问题。我们的应用程序也通过 Citrix 托管,该功能正常,写入本地磁盘或 UNC 网络路径也运行良好。
我已经创建了一个SSCCE来演示这个问题,连接到带有远程桌面的计算机(确保是“本地资源”),然后运行程序以查看一些非常奇怪的行为!我使用的是 JDK-7u45。C:\
import static java.nio.file.StandardOpenOption.APPEND;
import static java.nio.file.StandardOpenOption.CREATE;
import static java.nio.file.StandardOpenOption.TRUNCATE_EXISTING;
import static java.nio.file.StandardOpenOption.WRITE;
import java.io.BufferedWriter;
import java.io.File;
import java.io.IOException;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.nio.charset.Charset;
import java.nio.charset.CharsetEncoder;
import java.nio.file.Files;
import java.nio.file.OpenOption;
import java.util.Collections;
/**
* Demonstrates weird issue when writing (appending) to a file over TsClient (Microsoft Remote Desktop).
*
* @author Martin
*/
public class WriteOverTsClientDemo
{
private static final File FILE_TO_WRITE = new File("\\\\tsclient\\C\\Temp\\TestFile.txt");
//private static final File FILE_TO_WRITE = new File("C:\\Temp\\TestFile.txt");
private static final String ROW_DATA = "111111111122222222223333333333444444444555555555566666666667777777777888888888899999999990000000000";
public static void main(String[] args) throws IOException
{
if (!FILE_TO_WRITE.getParentFile().exists())
{
throw new RuntimeException("\nPlease create directory C:\\Temp\\ on your local machine and run this application via RemoteDesktop with C:\\ as a 'Local resource'.");
}
FILE_TO_WRITE.delete();
new WriteOverTsClientDemo().execute();
}
private void execute() throws IOException
{
System.out.println("Writing to file: " + FILE_TO_WRITE);
System.out.println();
for (int i = 1; i <= 10; i++)
{
System.out.println("Writing batch " + i + "...");
writeDataToFile(i);
System.out.println("Size of file after batch " + i + ": " + FILE_TO_WRITE.length());
System.out.println();
}
System.out.println("Done!");
}
private void writeDataToFile(int batch) throws IOException
{
Charset charset = Charset.forName("UTF-8");
CharsetEncoder encoder = charset.newEncoder();
try(OutputStream out = Files.newOutputStream(FILE_TO_WRITE.toPath(), CREATE, WRITE, getTruncateOrAppendOption(batch));
BufferedWriter writer = new BufferedWriter(new OutputStreamWriter(out, encoder)))
{
writeData(batch, writer);
}
}
private void writeData(int batch, BufferedWriter writer) throws IOException
{
for (String data : createData())
{
writer.append(Integer.toString(batch));
writer.append(" ");
writer.append(data);
writer.append("\n");
}
}
private Iterable<String> createData()
{
return Collections.nCopies(100, ROW_DATA);
}
/**
* @return option to write from the beginning or from the end of the file
*/
private OpenOption getTruncateOrAppendOption(int batch)
{
return batch == 1 ? TRUNCATE_EXISTING : APPEND;
}
}