查表法比计算要快多少?

目标与结论

本文通过对sin不同角度的值分别进行计算和查表操作,以比较两者的性能差距。根据实验结果,使用sin进行直接计算每秒能完成1万次,而查表每秒可进行300万次,相差巨大。同时还需要注意一点,sin在计算的时候,如果是在 2 π 2\pi 2π以内的,可以效率要高于其他范围的值,因为其他值还需要进行一个取余操作,使其落在 2 π 2\pi 2π以内,而浮点型的取余计算相对要耗时,所以导致性能会有明显的差异。

过程

36000000 / 3600 = 10000 每秒约1万次 sin 计算,查表每秒可以进行300万次。根据 Java内存性能测试的结果,内存读写大约5百万次左右,与本次测试结果基本一致。

以下是测试数据和代码。

---------------------------------
Table.length = 36000000, sum(table) = -0.1235.
test0: 361 ms, test1: 3944 ms, test2: 371 ms.
Pure time: test1 = 3583 ms, test2 = 10 ms, test1/test2 = 358.30.
---------------------------------
Table.length = 36000000, sum(table) = -0.1235.
test0: 360 ms, test1: 3947 ms, test2: 372 ms.
Pure time: test1 = 3587 ms, test2 = 12 ms, test1/test2 = 298.92.
---------------------------------
Table.length = 36000000, sum(table) = -0.1235.
test0: 362 ms, test1: 3964 ms, test2: 374 ms.
Pure time: test1 = 3602 ms, test2 = 12 ms, test1/test2 = 300.17.
---------------------------------
Table.length = 36000000, sum(table) = -0.1235.
test0: 359 ms, test1: 3959 ms, test2: 373 ms.
Pure time: test1 = 3600 ms, test2 = 14 ms, test1/test2 = 257.14.
---------------------------------
Table.length = 36000000, sum(table) = -0.1235.
test0: 359 ms, test1: 3942 ms, test2: 371 ms.
Pure time: test1 = 3583 ms, test2 = 12 ms, test1/test2 = 298.58.
---------------------------------
Table.length = 36000000, sum(table) = -0.1235.
test0: 363 ms, test1: 3952 ms, test2: 372 ms.
Pure time: test1 = 3589 ms, test2 = 9 ms, test1/test2 = 398.78.
---------------------------------
Table.length = 36000000, sum(table) = -0.1235.
test0: 362 ms, test1: 4019 ms, test2: 402 ms.
Pure time: test1 = 3657 ms, test2 = 40 ms, test1/test2 = 91.43.
---------------------------------
Table.length = 36000000, sum(table) = -0.1235.
test0: 386 ms, test1: 3959 ms, test2: 372 ms.
Pure time: test1 = 3573 ms, test2 = -14 ms, test1/test2 = -255.21.
---------------------------------
Table.length = 36000000, sum(table) = -0.1235.
test0: 359 ms, test1: 4167 ms, test2: 375 ms.
Pure time: test1 = 3808 ms, test2 = 16 ms, test1/test2 = 238.00.
---------------------------------
Table.length = 36000000, sum(table) = -0.1235.
test0: 359 ms, test1: 4068 ms, test2: 387 ms.
Pure time: test1 = 3709 ms, test2 = 28 ms, test1/test2 = 132.46.
public class TrigonometricTest {
	static double delta = 0.00001 * Math.PI / 180;
	static int len = 0;

	public static void main(String[] args) {
		for (int i = 0; i < 10; i++)
			test();
	}

	public static void test() {
		System.out.println("---------------------------------");
		len = (int) (2 * Math.PI / delta);
		var table = generateTable();
		int loops = 10;
		double r = 0;

		// warm up
		for (int i = 0; i < 2; i++) {
			test0();
			r = test1(r);
			r = test2(table, r);
		}

		// start real test
		long t0 = System.nanoTime();
		for (int i = 0; i < loops; i++)
			test0();
		long t1 = System.nanoTime();
		for (int i = 0; i < loops; i++)
			r = test1(r);
		long t2 = System.nanoTime();
		for (int i = 0; i < loops; i++)
			r = test2(table, r);
		long t3 = System.nanoTime();

		long t_empty = (t1 - t0) / 1000000;
		long t_test1 = (t2 - t1) / 1000000;
		long t_test2 = (t3 - t2) / 1000000;

		System.out.printf("Table.length = %d, sum(table) = %.4f.\n", table.length, r);
		System.out.printf("test0: %d ms, test1: %d ms, test2: %d ms.\n", t_empty, t_test1, t_test2);
		System.out.printf("Pure time: test1 = %d ms, test2 = %d ms, test1/test2 = %.2f.\n", t_test1 - t_empty,
				t_test2 - t_empty, 1.0 * (t_test1 - t_empty) / (t_test2 - t_empty));
	}

	public static double test0() {
		double p = 0;
		for (int i = 0; i < len; i++)
			p += delta;
		return p;
	}

	public static double test1(double r) {
		double p = 0;
		for (int i = 0; i < len; i++) {
			r += Math.sin(p);
			p += delta;
		}
		return r;
	}

	public static double test2(double[] table, double r) {
		for (int i = 0; i < table.length; i++)
			r += table[i];
		return r;
	}

	public static double[] generateTable() {
		double[] table = new double[len];
		for (int i = 0; i < table.length; i++)
			table[i] = Math.sin(i * delta);
		return table;
	}

}

package tests;

public class TrigonometricTest {
	static double delta = 0.0001 * Math.PI / 180;
	static int len = 0;
	static int totalRuns = 100;

	public static void main(String[] args) {
		for (int i = 0; i < 10; i++)
			test();
	}



	public static void Test1() {
		len = (int) (2 * Math.PI / delta);
		var table = generateTable();
		double r = 0;
		// warm up
		for (int i = 0; i < 2; i++) {
			test0();
			r = test1(r);
			r = test2(table, r);
		}

		// start real test
		long t0 = System.nanoTime();
		test0();
		long t1 = System.nanoTime();
		r = test1(r);
		long t2 = System.nanoTime(); 

		double totalTimes = len * totalRuns;
		double t_loop = (t1 - t0) / 1000000.0;
		double t_test1 = (t2 - t1) / 1000000.0; 
		double diff = t_test1 - t_loop;
		double speed = totalTimes / diff;

		System.out.printf("----\ntotal runs = %.0f, sum = %.4f.\n", totalTimes, r);
		System.out.printf("test0: %.4f ms, test1: %.4f ms\ndiff = %.4f, speed: %.0f Times/ms.\n", t_loop, t_test1, diff, speed); 
	}

	public static void test() {
		len = (int) (2 * Math.PI / delta);
		var table = generateTable();
		System.out.println("---------------------------------");
		System.out.printf("table.length: %.2E, len = %.2E.\n", 1.0 * table.length, 1.0 * len);
		double r = 0;
		// warm up
		for (int i = 0; i < 2; i++) {
			test0();
			r = test1(r);
			r = test2(table, r);
		}

		// start real test
		long t0 = System.nanoTime();
		test0();
		long t1 = System.nanoTime();
		r = test1(r);
		long t2 = System.nanoTime();
		r = test2(table, r);
		long t3 = System.nanoTime();

		double totalTimes = len * totalRuns;
		long t_empty = (t1 - t0) / 1000000;
		long t_test1 = (t2 - t1) / 1000000;
		long t_test2 = (t3 - t2) / 1000000;

		System.out.printf("totalTimes = %.2E, sum(table) = %.4f.\n", totalTimes, r);
		System.out.printf("test0: %d ms, test1: %d ms, test2: %d ms.\n", t_empty, t_test1, t_test2);
		System.out.printf("Pure time: test1 = %d ms, test2 = %d ms, test1/test2 = %.2f.\n", t_test1 - t_empty,
				t_test2 - t_empty, 1.0 * (t_test1 - t_empty) / (t_test2 - t_empty));
	}

	public static double test0() {
		double p = 0;
		for (int runTimes = 0; runTimes < totalRuns; runTimes++)
			for (int i = 0; i < len; i++)
				p += delta;
		return p;
	}

	public static double test1(double r) {
		double p = 0;
		for (int runTimes = 0; runTimes < totalRuns; runTimes++)
			for (int i = 0; i < len; i++) {
				r += Math.sin(p);
				p += delta;
			}
		return r;
	}

	public static double test2(double[] table, double r) {
		for (int runTimes = 0; runTimes < totalRuns; runTimes++)
			for (int i = 0; i < table.length; i++)
				r += table[i];
		return r;
	}

	public static double[] generateTable() {
		double[] table = new double[len];
		for (int i = 0; i < table.length; i++)
			table[i] = Math.sin(i * delta);
		return table;
	}

}

你可能感兴趣的:(Java程序设计,性能测试)