溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java、PHP、Python與MySQL交互的性能測(cè)試

發(fā)布時(shí)間:2020-07-06 21:25:18 來源:網(wǎng)絡(luò) 閱讀:1448 作者:觸龍 欄目:數(shù)據(jù)庫

這幾天看源碼弄清了一件事:WEB服務(wù)器接收瀏覽器請(qǐng)求、將請(qǐng)求傳給PHP/Python進(jìn)程(FCGI等)、與數(shù)據(jù)庫進(jìn)行交互都是用socket(套接字)。 
也就是說,這些行為都是進(jìn)程間通信。一臺(tái)WEB服務(wù)器在硬件、操作系統(tǒng)不變的情況下,它的性能主要取決于socket通信的速度。如果所有進(jìn)程都在一臺(tái)服務(wù)器上的話,這個(gè)速度就取決于通信的效率了。 

例如與MySQL數(shù)據(jù)庫交互時(shí),程序會(huì)調(diào)用驅(qū)動(dòng)程序來訪問數(shù)據(jù)庫,這個(gè)驅(qū)動(dòng)程序主要做這幾件事: 



1.創(chuàng)建socket,連接到MySQL。 

2.將程序調(diào)用的API翻譯成SQL語句,通過socket發(fā)送給MySQL;MySQL執(zhí)行后,將結(jié)果發(fā)送回來;驅(qū)動(dòng)程序?qū)⒔Y(jié)果(字符串)傳給程序,如有需要,還可以自動(dòng)翻譯成程序能識(shí)別的變量類型(如整型)。 

3.斷開連接。 


可見連接的速度、翻譯的速度和接受響應(yīng)的及時(shí)性是最主要的3個(gè)方面。 

弄明白這點(diǎn)后就不難發(fā)現(xiàn),與數(shù)據(jù)庫的執(zhí)行時(shí)間相比,這些只是很少的一部分;而且PHP和Python使用的都是C實(shí)現(xiàn),而JDBC是用Java實(shí)現(xiàn),所以根本不必?fù)?dān)心PHP和Python的性能。 
不過在翻譯方面還存在算法和實(shí)現(xiàn)上的差異,客戶端還可以緩存一些語句,所以仍然會(huì)出現(xiàn)一些性能上的差異。 
為證明我的想法,我特意去測(cè)試了一番。 

首先列出測(cè)試平臺(tái): 

引用


CPU:Intel Core2 Duo T9400 @ 2.53GHz 
內(nèi)存:3GB 
操作系統(tǒng):Windows XP Pro SP2 
MySQL:5.1.36 
Java:1.6.0_17-b04 
JDBC:MySQL Connector/J 5.1.10 
PHP:5.2.11 (cli) 
MySQLi:5.2.11.11 
Python:2.6.4 
MySQL-Python:1.2.3c1 



所用的庫都是最新版的,也都采用了最為推薦的庫。 
但數(shù)據(jù)庫并沒有使用最新的穩(wěn)定版,因?yàn)槲覒械弥叵铝恕?.4.3-beta測(cè)試版也試過,在連續(xù)插入時(shí),性能比5.1快1~2個(gè)數(shù)量級(jí),估計(jì)是服務(wù)器端緩存和設(shè)置的原因。 

測(cè)試項(xiàng)目: 



1.創(chuàng)建100萬個(gè)隨機(jī)數(shù),并生成插入這些隨機(jī)數(shù)的SQL語句。 

2.連接本地?cái)?shù)據(jù)庫,如不成功,嘗試創(chuàng)建數(shù)據(jù)庫。 

3.刪除并創(chuàng)建數(shù)據(jù)庫表,引擎類型為InnoDB,主鍵為自動(dòng)遞增的整數(shù),此外有個(gè)浮點(diǎn)型的字段(無索引)。 

4.分成100組,每次插入1萬個(gè)隨機(jī)數(shù)。(因?yàn)槊拷M的執(zhí)行量都很大,因此啟用自動(dòng)提交事務(wù)。) 

5.用SELECT COUNT(*)統(tǒng)計(jì)小于0.1的隨機(jī)數(shù)個(gè)數(shù)。(約10萬個(gè)) 

6.用SELECT *取出再統(tǒng)計(jì)大于0.9的隨機(jī)數(shù)個(gè)數(shù)。(約10萬個(gè)) 

7.將所有0.4~0.5之間的隨機(jī)數(shù)加1。(約10萬個(gè)) 

8.將所有0.5~0.6之間的行刪除。(約20萬個(gè)) 

9.斷開數(shù)據(jù)庫連接。 

10.再次連接數(shù)據(jù)庫。 



測(cè)試代碼: 

Java: 

Java代碼  Java、PHP、Python與MySQL交互的性能測(cè)試

  1. import java.sql.Connection;  

  2. import java.sql.DriverManager;  

  3. import java.sql.SQLException;  

  4. import java.sql.Statement;  

  5. import java.sql.ResultSet;  

  6. import java.util.Random;  

  7.   

  8.   

  9. public final class Test {  

  10.   

  11.     public static void main(String[] args) {  

  12.         final int SIZE1 = 10000;  

  13.         final int SIZE2 = 100;  

  14.         final String DB_ENGINE = "InnoDB"// InnoDB Memory MyISAM  

  15.         final double NANO_TIME_PER_SEC = 1000000000.0;  

  16.         System.out.printf("測(cè)試數(shù)據(jù)量:%d\n", SIZE1 * SIZE2);  

  17.         System.out.printf("測(cè)試引擎:%s\n", DB_ENGINE);  

  18.           

  19.         long t1 = System.nanoTime(), t2, t3 = 0, t4, t5, t6, t7, t8, t9, t10, t11;  

  20.         Connection conn = null;  

  21.         Statement stmt = null;  

  22.         ResultSet rs = null;  

  23.           

  24.         Random r = new Random();  

  25.         String[] sqls = new String[SIZE2];  

  26.         for (int i = 0; i < SIZE2; ++i){  

  27.             StringBuilder buffer = new StringBuilder("INSERT INTO test (value) VALUES (");  

  28.             for (int j = 0; j < SIZE1; ++j){  

  29.                 buffer.append(r.nextDouble()).append("),(");  

  30.             }  

  31.             sqls[i] = buffer.substring(0, buffer.length() -2);  

  32.         }  

  33.         t2 = System.nanoTime();  

  34.           

  35.         try {  

  36.             conn = DriverManager.getConnection("jdbc:mysql://localhost/testdb?user=root&password=123456");  

  37.             t3 = System.nanoTime();  

  38.             stmt = conn.createStatement();  

  39.         } catch (SQLException e) {  

  40.             try {  

  41.                 conn = DriverManager.getConnection("jdbc:mysql://localhost/?user=root&password=123456");  

  42.                 t3 = System.nanoTime();  

  43.                 stmt = conn.createStatement();  

  44.                 stmt.execute("CREATE DATABASE testdb");  

  45.             } catch (SQLException ex) {  

  46.                 System.out.println("SQLException: " + ex.getMessage());  

  47.                 System.out.println("SQLState: " + ex.getSQLState());  

  48.                 System.out.println("VendorError: " + ex.getErrorCode());  

  49.             }  

  50.         }  

  51.           

  52.         try {  

  53.             stmt.execute("DROP TABLE test");  

  54.         } catch (SQLException e) {  

  55.         }  

  56.         try {  

  57.             stmt.execute("CREATE TABLE test (`id` INT AUTO_INCREMENT PRIMARY KEY, `value` REAL) ENGINE = " + DB_ENGINE);  

  58.         } catch (SQLException e) {  

  59.         }  

  60.         t4 = System.nanoTime();  

  61.           

  62.         try {  

  63.             for (String sql: sqls){  

  64.                 stmt.execute(sql);  

  65.             }  

  66.             t5 = System.nanoTime();  

  67.   

  68.             rs = stmt.executeQuery("SELECT COUNT(*) FROM test WHERE value < 0.1");  

  69.             if (rs.next())  

  70.                 System.out.printf("共有%d個(gè)小于0.1的隨機(jī)數(shù)\n", rs.getInt(1));  

  71.             t6 = System.nanoTime();  

  72.   

  73.             rs = stmt.executeQuery("SELECT * FROM test WHERE value > 0.9");  

  74.             if (rs.last())  

  75.                 System.out.printf("共有%d個(gè)大于0.9的隨機(jī)數(shù)\n", rs.getRow());  

  76.             t7 = System.nanoTime();  

  77.   

  78.             stmt.executeUpdate("UPDATE test SET value = value + 0.1 WHERE value > 0.4 AND value < 0.5");  

  79.             t8 = System.nanoTime();  

  80.   

  81.             stmt.execute("DELETE FROM test WHERE value > 0.5 AND value < 0.6");  

  82.             t9 = System.nanoTime();  

  83.   

  84.             stmt.close();  

  85.             conn.close();  

  86.             t10 = System.nanoTime();  

  87.               

  88.             conn = DriverManager.getConnection("jdbc:mysql://localhost/?user=root&password=123456");  

  89.             t11 = System.nanoTime();  

  90.             conn.close();  

  91.   

  92.             System.out.printf("創(chuàng)建隨機(jī)數(shù):%f\n", (t2 - t1) / NANO_TIME_PER_SEC);  

  93.             System.out.printf("初次連接數(shù)據(jù)庫:%f\n", (t3 - t2) / NANO_TIME_PER_SEC);  

  94.             System.out.printf("再次連接數(shù)據(jù)庫:%f\n", (t11 - t10) / NANO_TIME_PER_SEC);  

  95.             System.out.printf("初始化數(shù)據(jù)庫和表:%f\n", (t4 - t3) / NANO_TIME_PER_SEC);  

  96.             System.out.printf("插入:%f\n", (t5 - t4) / NANO_TIME_PER_SEC);  

  97.             System.out.printf("選擇(COUNT):%f\n", (t6 - t5) / NANO_TIME_PER_SEC);  

  98.             System.out.printf("選擇:%f\n", (t7 - t6) / NANO_TIME_PER_SEC);  

  99.             System.out.printf("更新:%f\n", (t8 - t7) / NANO_TIME_PER_SEC);  

  100.             System.out.printf("刪除:%f\n", (t9 - t8) / NANO_TIME_PER_SEC);  

  101.             System.out.printf("關(guān)閉連接:%f\n", (t10 - t9) / NANO_TIME_PER_SEC);  

  102.             System.out.printf("總時(shí)間:%f\n", (t10 - t1) / NANO_TIME_PER_SEC);  

  103.               

  104.         } catch (SQLException ex) {  

  105.             System.out.println("SQLException: " + ex.getMessage());  

  106.             System.out.println("SQLState: " + ex.getSQLState());  

  107.             System.out.println("VendorError: " + ex.getErrorCode());  

  108.         }  

  109.   

  110.     }  

  111.   

  112. }  


PHP: 

Php代碼  Java、PHP、Python與MySQL交互的性能測(cè)試

  1. <?php  

  2. define('SIZE1', 10000);  

  3. define('SIZE2', 100);  

  4. define('DB_ENGINE''InnoDB'); // InnoDB Memory MyISAM  

  5.   

  6. printf("測(cè)試數(shù)據(jù)量:%d\n", SIZE1 * SIZE2);  

  7. printf("測(cè)試引擎:%s\n", DB_ENGINE);  

  8.   

  9. $t1 = microtime(true);  

  10. for ($i = 0; $i < SIZE2; ++$i){  

  11.     for ($j = 0; $j < SIZE1; ++$j){  

  12.         $buffer[] = lcg_value();  

  13.     }  

  14.     $sqls[$i] = 'INSERT INTO test (value) VALUES ('.join('),('$buffer).')';  

  15.     unset($buffer);  

  16. }  

  17.   

  18. $t2 = microtime(true);  

  19.   

  20. $db = new mysqli('localhost''root''123456''testdb');  

  21. $t3 = microtime(true);  

  22. if (mysqli_connect_errno()) {  

  23.     $db = new mysqli('localhost''root''123456');  

  24.     $t3 = microtime(true);  

  25.     $db->query('CREATE DATABASE testdb');  

  26.     $db->select_db('testdb');  

  27. }  

  28.   

  29. $db->query('DROP TABLE test');  

  30. $db->query('CREATE TABLE test (`id` INT AUTO_INCREMENT PRIMARY KEY, `value` REAL) ENGINE = '.DB_ENGINE);  

  31. $t4 = microtime(true);  

  32.   

  33. foreach ($sqls as $key=>$sql) {  

  34.     $db->query($sql);  

  35. }  

  36. $t5 = microtime(true);  

  37.   

  38. $result = $db->query('SELECT COUNT(*) FROM test WHERE value < 0.1')->fetch_row();  

  39. printf("共有%d個(gè)小于0.1的隨機(jī)數(shù)\n"$result[0]);  

  40. $t6 = microtime(true);  

  41.   

  42. $result = $db->query('SELECT * FROM test WHERE value > 0.9');  

  43. printf("共有%d個(gè)大于0.9的隨機(jī)數(shù)\n"$result->num_rows);  

  44. $t7 = microtime(true);  

  45.   

  46. $db->query('UPDATE test SET value = value + 0.1 WHERE value > 0.4 AND value < 0.5');  

  47. $t8 = microtime(true);  

  48.   

  49. $db->query('DELETE FROM test WHERE value > 0.5 AND value < 0.6');  

  50. $t9 = microtime(true);  

  51.   

  52. $db->close();  

  53. $t10 = microtime(true);  

  54.   

  55. $db = new mysqli('localhost''root''123456''testdb');  

  56. $t11 = microtime(true);  

  57. $db->close();  

  58.   

  59. printf("創(chuàng)建隨機(jī)數(shù):%f\n"$t2 - $t1);  

  60. printf("初次連接數(shù)據(jù)庫:%f\n"$t3 - $t2);  

  61. printf("再次連接數(shù)據(jù)庫:%f\n"$t11 - $t10);  

  62. printf("初始化數(shù)據(jù)庫和表:%f\n"$t4 - $t3);  

  63. printf("插入:%f\n"$t5 - $t4);  

  64. printf("選擇(COUNT):%f\n"$t6 - $t5);  

  65. printf("選擇:%f\n"$t7 - $t6);  

  66. printf("更新:%f\n"$t8 - $t7);  

  67. printf("刪除:%f\n"$t9 - $t8);  

  68. printf("關(guān)閉連接:%f\n"$t10 - $t9);  

  69. printf("總時(shí)間:%f\n"$t10 - $t1);  

  70. ?>  


Python: 

Python代碼  Java、PHP、Python與MySQL交互的性能測(cè)試

  1. # -*- coding: gbk -*-  

  2.   

  3. import MySQLdb  

  4. from random import random  

  5. from time import clock  

  6.   

  7. SIZE1 = 10000  

  8. SIZE2 = 100  

  9. DB_ENGINE = 'InnoDB' # InnoDB Memory MyISAM  

  10. print '測(cè)試數(shù)據(jù)量:', SIZE1 * SIZE2  

  11. print '測(cè)試引擎:', DB_ENGINE  

  12.   

  13. t1 = clock()  

  14.   

  15. sqls = ['INSERT INTO test (value) VALUES (%s)' % '),('.join([`random()` for i in xrange(SIZE1)]) for j in xrange(SIZE2)]  

  16. t2 = clock()  

  17.   

  18. try:  

  19.   con = MySQLdb.connect(user='root', passwd='123456', db='testdb')  

  20.   t3 = clock()  

  21.   cu = con.cursor()  

  22. except:  

  23.   con = MySQLdb.connect(user='root', passwd='123456')  

  24.   t3 = clock()  

  25.   cu = con.cursor()  

  26.   cu.execute('CREATE DATABASE testdb')  

  27.   con.select_db('testdb')  

  28.   

  29. con.autocommit(True)  

  30.   

  31. try:  

  32.   cu.execute('DROP TABLE test')  

  33. except:  

  34.   pass  

  35. cu.execute('''''CREATE TABLE test ( 

  36. `id` INT AUTO_INCREMENT PRIMARY KEY, 

  37. `value` REAL) 

  38. ENGINE = %s''' % DB_ENGINE)  

  39. t4 = clock()  

  40.   

  41. for sql in sqls:  

  42.   cu.execute(sql)  

  43. t5 = clock()  

  44.   

  45. cu.execute('SELECT COUNT(*) FROM test WHERE value < 0.1')  

  46. print '共有%d個(gè)小于0.1的隨機(jī)數(shù)' % cu.fetchone()[0]  

  47. t6 = clock()  

  48.   

  49. cu.execute('SELECT * FROM test WHERE value > 0.9')  

  50. print '共有%d個(gè)大于0.9的隨機(jī)數(shù)' % len(cu.fetchall())  

  51. t7 = clock()  

  52.   

  53. cu.execute('UPDATE test SET value = value + 0.1 WHERE value > 0.4 AND value < 0.5')  

  54. t8 = clock()  

  55.   

  56. cu.execute('DELETE FROM test WHERE value > 0.5 AND value < 0.6')  

  57. t9 = clock()  

  58.   

  59. cu.close()  

  60. con.close()  

  61. t10 = clock()  

  62.   

  63. con = MySQLdb.connect(user='root', passwd='123456', db='testdb')  

  64. t11 = clock()  

  65. con.close()  

  66.   

  67. print '創(chuàng)建隨機(jī)數(shù):', t2 - t1  

  68. print '初次連接數(shù)據(jù)庫:', t3 - t2  

  69. print '再次連接數(shù)據(jù)庫:', t11 - t10  

  70. print '初始化數(shù)據(jù)庫:', t4 - t3  

  71. print '插入:', t5 - t4  

  72. print '選擇(COUNT)', t6 - t5  

  73. print '選擇:', t7 - t6  

  74. print '更新:', t8 - t7  

  75. print '刪除:', t9 - t8  

  76. print '關(guān)閉連接:', t10 - t9  

  77. print '總時(shí)間:', t10 - t1  



MySQL-Python還有個(gè)底層的模塊,一并測(cè)試下: 

Python代碼  Java、PHP、Python與MySQL交互的性能測(cè)試

  1. # -*- coding: gbk -*-  

  2.   

  3. import _mysql  

  4. from MySQLdb.converters import conversions  

  5. from random import random  

  6. from time import clock  

  7.   

  8. SIZE1 = 10000  

  9. SIZE2 = 100  

  10. DB_ENGINE = 'InnoDB' # InnoDB Memory MyISAM  

  11. print '測(cè)試數(shù)據(jù)量:', SIZE1 * SIZE2  

  12. print '測(cè)試引擎:', DB_ENGINE  

  13.   

  14. t1 = clock()  

  15.   

  16. sqls = ['INSERT INTO test (value) VALUES (%s)' % '),('.join([`random()` for i in xrange(SIZE1)]) for j in xrange(SIZE2)]  

  17. t2 = clock()  

  18.   

  19. try:  

  20.   con = _mysql.connect(user='root', passwd='123456', db='testdb', conv=conversions)  

  21.   t3 = clock()  

  22. except:  

  23.   con = _mysql.connect(user='root', passwd='123456', conv=conversions)  

  24.   t3 = clock()  

  25.   con.query('CREATE DATABASE testdb')  

  26.   con.select_db('testdb')  

  27.   

  28. con.autocommit(True)  

  29.   

  30. try:  

  31.   con.query('DROP TABLE test')  

  32. except:  

  33.   pass  

  34. con.query('''''CREATE TABLE test ( 

  35. `id` INT AUTO_INCREMENT PRIMARY KEY, 

  36. `value` REAL) 

  37. ENGINE = %s''' % DB_ENGINE)  

  38. t4 = clock()  

  39.   

  40. for sql in sqls:  

  41.   con.query(sql)  

  42. t5 = clock()  

  43.   

  44. con.query('SELECT COUNT(*) FROM test WHERE value < 0.1')  

  45. print '共有%d個(gè)小于0.1的隨機(jī)數(shù)' % con.store_result().fetch_row()[0]  

  46. t6 = clock()  

  47.   

  48. con.query('SELECT * FROM test WHERE value > 0.9')  

  49. print '共有%d個(gè)大于0.9的隨機(jī)數(shù)' % con.store_result().num_rows()  

  50. t7 = clock()  

  51.   

  52. con.query('UPDATE test SET value = value + 0.1 WHERE value > 0.4 AND value < 0.5')  

  53. t8 = clock()  

  54.   

  55. con.query('DELETE FROM test WHERE value > 0.5 AND value < 0.6')  

  56. t9 = clock()  

  57.   

  58. con.close()  

  59. t10 = clock()  

  60.   

  61. con = _mysql.connect(user='root', passwd='123456', db='testdb', conv=conversions)  

  62. t11 = clock()  

  63. con.close()  

  64.   

  65. print '創(chuàng)建隨機(jī)數(shù):', t2 - t1  

  66. print '初次連接數(shù)據(jù)庫:', t3 - t2  

  67. print '再次連接數(shù)據(jù)庫:', t11 - t10  

  68. print '初始化數(shù)據(jù)庫:', t4 - t3  

  69. print '插入:', t5 - t4  

  70. print '選擇(COUNT)', t6 - t5  

  71. print '選擇:', t7 - t6  

  72. print '更新:', t8 - t7  

  73. print '刪除:', t9 - t8  

  74. print '關(guān)閉連接:', t10 - t9  

  75. print '總時(shí)間:', t10 - t1  



每種測(cè)試3次(等硬盤燈無閃爍后才進(jìn)行下一次測(cè)試),取最好的一次作為測(cè)試結(jié)果: 
Java: 

引用

測(cè)試數(shù)據(jù)量:1000000 
測(cè)試引擎:InnoDB 
共有99465個(gè)小于0.1的隨機(jī)數(shù) 
共有99859個(gè)大于0.9的隨機(jī)數(shù) 
創(chuàng)建隨機(jī)數(shù):2.367840 
初次連接數(shù)據(jù)庫:0.220420 
再次連接數(shù)據(jù)庫:0.013174 
初始化數(shù)據(jù)庫和表:0.075140 
插入:12.139346 
選擇(COUNT):1.130345 
選擇:1.017769 
更新:6.173245 
刪除:9.380070 
關(guān)閉連接:0.002131 
總時(shí)間:32.506307



PHP: 

引用

測(cè)試數(shù)據(jù)量:1000000 
測(cè)試引擎:InnoDB 
共有99898個(gè)小于0.1的隨機(jī)數(shù) 
共有100152個(gè)大于0.9的隨機(jī)數(shù) 
創(chuàng)建隨機(jī)數(shù):1.506294 
初次連接數(shù)據(jù)庫:0.003146 
再次連接數(shù)據(jù)庫:0.001808 
初始化數(shù)據(jù)庫和表:0.131754 
插入:12.046944 
選擇(COUNT):1.236742 
選擇:1.238153 
更新:6.115232 
刪除:8.145547 
關(guān)閉連接:0.000125 
總時(shí)間:30.423937



Python(MySQLdb): 

引用

測(cè)試數(shù)據(jù)量: 1000000 
測(cè)試引擎: InnoDB 
共有100040個(gè)小于0.1的隨機(jī)數(shù) 
共有100351個(gè)大于0.9的隨機(jī)數(shù) 
創(chuàng)建隨機(jī)數(shù): 1.6822107279 
初次連接數(shù)據(jù)庫: 0.0332120423126 
再次連接數(shù)據(jù)庫: 0.00221704155137 
初始化數(shù)據(jù)庫: 0.131054924578 
插入: 11.7999030603 
選擇(COUNT) 1.27067266929 
選擇: 1.16714526567 
更新: 6.29200638629 
刪除: 8.13660563005 
關(guān)閉連接: 0.000131022238861 
總時(shí)間: 30.5129417286



Python(_mysql): 

引用

測(cè)試數(shù)據(jù)量: 1000000 
測(cè)試引擎: InnoDB 
共有99745個(gè)小于0.1的隨機(jī)數(shù) 
共有99869個(gè)大于0.9的隨機(jī)數(shù) 
創(chuàng)建隨機(jī)數(shù): 1.68099074044 
初次連接數(shù)據(jù)庫: 0.0112056141213 
再次連接數(shù)據(jù)庫: 0.00159293988482 
初始化數(shù)據(jù)庫: 0.130169616529 
插入: 12.1364623157 
選擇(COUNT) 1.125517908 
選擇: 0.968366649951 
更新: 6.8042843434 
刪除: 8.9760508668 
關(guān)閉連接: 9.61015995031e-05 
總時(shí)間: 31.8331441566



可以看到,在大批量數(shù)據(jù)測(cè)試中,Java是最慢的,而PHP是最快的。 
不考慮IO性能的波動(dòng)的話,Java主要慢在連接和關(guān)閉數(shù)據(jù)庫。JDBC 4.0在第一次連接數(shù)據(jù)庫時(shí)會(huì)動(dòng)態(tài)加載驅(qū)動(dòng),非常耗時(shí),因此使用Java要記住使用數(shù)據(jù)庫連接池,避免連接浪費(fèi)大量時(shí)間。當(dāng)然,這也造成了數(shù)據(jù)庫的負(fù)擔(dān),勢(shì)必影響內(nèi)存占用。而創(chuàng)建隨機(jī)數(shù)的算法實(shí)現(xiàn)各不相同,所以不具備可比性;令我驚訝的是SELECT的翻譯速度,將字符串轉(zhuǎn)換成浮點(diǎn)數(shù)居然慢于Python,要知道后者的浮點(diǎn)數(shù)可是對(duì)象。 
PHP連接數(shù)據(jù)庫非???,所以完全無需使用連接池,因?yàn)榫S護(hù)連接池會(huì)增加復(fù)雜性。 
Python的表現(xiàn)和PHP差不多,但是第一次連接數(shù)據(jù)庫比較慢(仍比Java快1個(gè)數(shù)量級(jí))。如果不用連接池,則使用FCGI等方式來運(yùn)行比較合適。_mysql模塊的通信很快,但更新和刪除操作卻不太理想,也許是IO性能波動(dòng)的原因。此外,我在連接數(shù)據(jù)庫時(shí)使用了轉(zhuǎn)換參數(shù),實(shí)際上我用的語句都不需要翻譯,不使用的話會(huì)更快一點(diǎn)。 

接著試試小數(shù)據(jù),改成最常用的MyISAM引擎,插入100條(1組)。一般的應(yīng)用不可能一次插入那么多,所以足夠滿足平時(shí)的應(yīng)用了;而且由于數(shù)據(jù)量很小,也基本不受IO影響。 

測(cè)試結(jié)果: 

Java: 

引用

測(cè)試數(shù)據(jù)量:100 
測(cè)試引擎:MyISAM 
共有9個(gè)小于0.1的隨機(jī)數(shù) 
共有10個(gè)大于0.9的隨機(jī)數(shù) 
創(chuàng)建隨機(jī)數(shù):0.001596 
初次連接數(shù)據(jù)庫:0.224135 
再次連接數(shù)據(jù)庫:0.018656 
初始化數(shù)據(jù)庫和表:0.055601 
插入:0.001476 
選擇(COUNT):0.000529 
選擇:0.000433 
更新:0.000304 
刪除:0.000313 
關(guān)閉連接:0.000927 
總時(shí)間:0.285314


PHP: 

引用

測(cè)試數(shù)據(jù)量:測(cè)試數(shù)據(jù)量:100 
測(cè)試引擎:MyISAM 
共有12個(gè)小于0.1的隨機(jī)數(shù) 
共有9個(gè)大于0.9的隨機(jī)數(shù) 
創(chuàng)建隨機(jī)數(shù):0.000649 
初次連接數(shù)據(jù)庫:0.008077 
再次連接數(shù)據(jù)庫:0.001609 
初始化數(shù)據(jù)庫和表:0.060421 
插入:0.001860 
選擇(COUNT):0.000580 
選擇:0.000465 
更新:0.000326 
刪除:0.000373 
關(guān)閉連接:0.000127 
總時(shí)間:0.072878


Python(MySQLdb): 

引用

測(cè)試數(shù)據(jù)量: 100 
測(cè)試引擎: MyISAM 
共有14個(gè)小于0.1的隨機(jī)數(shù) 
共有9個(gè)大于0.9的隨機(jī)數(shù) 
創(chuàng)建隨機(jī)數(shù): 0.000198907961766 
初次連接數(shù)據(jù)庫: 0.0334640296462 
再次連接數(shù)據(jù)庫: 0.00150577796899 
初始化數(shù)據(jù)庫: 0.0123194428342 
插入: 0.00125211444471 
選擇(COUNT) 0.000581079438867 
選擇: 0.000484139744018 
更新: 0.000250311142897 
刪除: 0.000262323842835 
關(guān)閉連接: 7.98984228442e-05 
總時(shí)間: 0.0488922474784



Python(_mysql): 

引用

測(cè)試數(shù)據(jù)量: 100 
測(cè)試引擎: MyISAM 
共有12個(gè)小于0.1的隨機(jī)數(shù) 
共有10個(gè)大于0.9的隨機(jī)數(shù) 
創(chuàng)建隨機(jī)數(shù): 0.000214273043082 
初次連接數(shù)據(jù)庫: 0.0118774872225 
再次連接數(shù)據(jù)庫: 0.00123702872851 
初始化數(shù)據(jù)庫: 0.0315031659052 
插入: 0.00120322554962 
選擇(COUNT) 0.000596165155069 
選擇: 0.000507327048549 
更新: 0.0002447238406 
刪除: 0.00026148574749 
關(guān)閉連接: 5.78285787719e-05 
總時(shí)間: 0.0464656820909



從結(jié)果可以看出,雖然差距都很小,但Python仍然稍微占優(yōu)。不過Java的SELECT操作稍微勝出,而這也是實(shí)際應(yīng)用中最常使用的操作。 
再從語言方面來看,Python無疑是寫得最歡暢的,生成隨機(jī)數(shù)只用了1行代碼;PHP的變量要寫個(gè)$讓我老是出錯(cuò),不過數(shù)據(jù)庫操作不需要處理異常,這點(diǎn)節(jié)省了很多代碼;Java的代碼量很大,而且不得不使用很多try...catch,我甚至懶得以安全的方式將close()放在finally塊里面,因?yàn)樗矔?huì)拋出我根本懶得去管的異常,且會(huì)提示我計(jì)時(shí)變量可能沒有初始化。 

總體上來看,Google放棄Python,只采用C++和Java是有點(diǎn)不明智。因?yàn)轫撁骓憫?yīng)時(shí)間主要在于數(shù)據(jù)庫通信和磁盤文件IO上,語言的影響基本忽略不計(jì)。


向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI