用Java将整数数组写入文件的最快方法?

如标题所示,我正在寻找最快的方式将整数数组写入文件。数组的大小将有所不同,并且实际上包含2500至25000000 int之间的任何位置。

这是我目前正在使用的代码:

DataOutputStream writer = new DataOutputStream(new BufferedOutputStream(new FileOutputStream(filename)));

for (int d : data)

writer.writeInt(d);

鉴于DataOutputStream有一种写入字节数组的方法,我已经尝试将int数组转换为字节数组,如下所示:

private static byte[] integersToBytes(int[] values) throws IOException {

ByteArrayOutputStream baos = new ByteArrayOutputStream();

DataOutputStream dos = new DataOutputStream(baos);

for (int i = 0; i < values.length; ++i) {

dos.writeInt(values[i]);

}

return baos.toByteArray();

}

像这样:

private static byte[] integersToBytes2(int[] src) {

int srcLength = src.length;

byte[] dst = new byte[srcLength << 2];

for (int i = 0; i < srcLength; i++) {

int x = src[i];

int j = i << 2;

dst[j++] = (byte) ((x >>> 0) & 0xff);

dst[j++] = (byte) ((x >>> 8) & 0xff);

dst[j++] = (byte) ((x >>> 16) & 0xff);

dst[j++] = (byte) ((x >>> 24) & 0xff);

}

return dst;

}

两者似乎都使速度略有提高,约为5%。我没有对它们进行足够严格的测试以确认这一点。

是否有任何技术可以加快此文件的写入操作,或者有关Java IO写入性能的最佳实践的相关指南?

回答:

我看了三个选择:

  1. 使用DataOutputStream;
  2. 使用ObjectOutputStream(对于Serializable对象而言int[]);和
  3. 使用FileChannel

结果是

DataOutputStream wrote 1,000,000 ints in 3,159.716 ms

ObjectOutputStream wrote 1,000,000 ints in 295.602 ms

FileChannel wrote 1,000,000 ints in 110.094 ms

因此NIO版本是最快的。它还具有允许编辑的优点,这意味着您可以轻松更改一个int,而这ObjectOutputStream将需要读取整个数组,对其进行修改并将其写到文件中。

代码如下:

private static final int NUM_INTS = 1000000;

interface IntWriter {

void write(int[] ints);

}

public static void main(String[] args) {

int[] ints = new int[NUM_INTS];

Random r = new Random();

for (int i=0; i<NUM_INTS; i++) {

ints[i] = r.nextInt();

}

time("DataOutputStream", new IntWriter() {

public void write(int[] ints) {

storeDO(ints);

}

}, ints);

time("ObjectOutputStream", new IntWriter() {

public void write(int[] ints) {

storeOO(ints);

}

}, ints);

time("FileChannel", new IntWriter() {

public void write(int[] ints) {

storeFC(ints);

}

}, ints);

}

private static void time(String name, IntWriter writer, int[] ints) {

long start = System.nanoTime();

writer.write(ints);

long end = System.nanoTime();

double ms = (end - start) / 1000000d;

System.out.printf("%s wrote %,d ints in %,.3f ms%n", name, ints.length, ms);

}

private static void storeOO(int[] ints) {

ObjectOutputStream out = null;

try {

out = new ObjectOutputStream(new FileOutputStream("object.out"));

out.writeObject(ints);

} catch (IOException e) {

throw new RuntimeException(e);

} finally {

safeClose(out);

}

}

private static void storeDO(int[] ints) {

DataOutputStream out = null;

try {

out = new DataOutputStream(new FileOutputStream("data.out"));

for (int anInt : ints) {

out.write(anInt);

}

} catch (IOException e) {

throw new RuntimeException(e);

} finally {

safeClose(out);

}

}

private static void storeFC(int[] ints) {

FileOutputStream out = null;

try {

out = new FileOutputStream("fc.out");

FileChannel file = out.getChannel();

ByteBuffer buf = file.map(FileChannel.MapMode.READ_WRITE, 0, 4 * ints.length);

for (int i : ints) {

buf.putInt(i);

}

file.close();

} catch (IOException e) {

throw new RuntimeException(e);

} finally {

safeClose(out);

}

}

private static void safeClose(OutputStream out) {

try {

if (out != null) {

out.close();

}

} catch (IOException e) {

// do nothing

}

}

以上是 用Java将整数数组写入文件的最快方法? 的全部内容, 来源链接: utcz.com/qa/399251.html

回到顶部