精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

五個(gè)常見(jiàn) PHP數(shù)據(jù)庫(kù)問(wèn)題

開(kāi)發(fā) 后端
本文揭露了 PHP 應(yīng)用程序中出現(xiàn)的五個(gè)常見(jiàn)數(shù)據(jù)庫(kù)問(wèn)題,包括數(shù)據(jù)庫(kù)模式設(shè)計(jì)、數(shù)據(jù)庫(kù)訪問(wèn)和使用數(shù)據(jù)庫(kù)的業(yè)務(wù)邏輯代碼以及它們的解決方案。詳細(xì)閱讀本文,希望對(duì)你能有所幫助。

  如果只有一種方式使用數(shù)據(jù)庫(kù)是正確的,您可以用很多的方式創(chuàng)建數(shù)據(jù)庫(kù)設(shè)計(jì)、數(shù)據(jù)庫(kù)訪問(wèn)和基于數(shù)據(jù)庫(kù)的 PHP 業(yè)務(wù)邏輯代碼,但最終一般以錯(cuò)誤告終。本文說(shuō)明了數(shù)據(jù)庫(kù)設(shè)計(jì)和訪問(wèn)數(shù)據(jù)庫(kù)的 PHP 代碼中出現(xiàn)的五個(gè)常見(jiàn)問(wèn)題,以及在遇到這些問(wèn)題時(shí)如何修復(fù)它們。

  問(wèn)題 1:直接使用 MySQL

  一個(gè)常見(jiàn)問(wèn)題是較老的 PHP 代碼直接使用 mysql_ 函數(shù)來(lái)訪問(wèn)數(shù)據(jù)庫(kù)。清單 1 展示了如何直接訪問(wèn)數(shù)據(jù)庫(kù)。

  清單 1. Access/get.php

  1. <?php  
  2. function get_user_id( $name )  
  3. {  
  4.   $db = mysql_connect( 'localhost''root''password' );  
  5.   mysql_select_db( 'users' );  
  6.  
  7.   $res = mysql_query( "SELECT id FROM users WHERE login='".$name."'" );  
  8.   while$row = mysql_fetch_array( $res ) ) { $id = $row[0]; }  
  9.  
  10.   return $id;  
  11. }  
  12.  
  13. var_dump( get_user_id( 'jack' ) );  
  14. ?> 

  注意使用了 mysql_connect 函數(shù)來(lái)訪問(wèn)數(shù)據(jù)庫(kù)。還要注意查詢(xún),其中使用字符串連接來(lái)向查詢(xún)添加 $name 參數(shù)。

  該技術(shù)有兩個(gè)很好的替代方案:PEAR DB 模塊和 PHP Data Objects (PDO) 類(lèi)。兩者都從特定數(shù)據(jù)庫(kù)選擇提供抽象。因此,您的代碼無(wú)需太多調(diào)整就可以在 IBM® DB2®、MySQL、PostgreSQL 或者您想要連接到的任何其他數(shù)據(jù)庫(kù)上運(yùn)行。

  使用 PEAR DB 模塊和 PDO 抽象層的另一個(gè)價(jià)值在于您可以在 SQL 語(yǔ)句中使用 ? 操作符。這樣做可使 SQL 更加易于維護(hù),且可使您的應(yīng)用程序免受 SQL 注入攻擊。

  使用 PEAR DB 的替代代碼如下所示。

  清單 2. Access/get_good.php

     
  1. <?php  
  2. require_once("DB.php");  
  3.  
  4. function get_user_id( $name )  
  5. {  
  6.   $dsn = 'mysql://root:password@localhost/users';  
  7.   $db =& DB::Connect( $dsnarray() );  
  8.   if (PEAR::isError($db)) { die($db->getMessage()); }  
  9.  
  10.   $res = $db->query( 'SELECT id FROM users WHERE login=?',  
  11.   array$name ) );  
  12.   $id = null;  
  13.   while$res->fetchInto( $row ) ) { $id = $row[0]; }  
  14.  
  15.   return $id;  
  16. }  
  17.  
  18. var_dump( get_user_id( 'jack' ) );  
  19. ?> 

  注意,所有直接用到 MySQL 的地方都消除了,只有 $dsn 中的數(shù)據(jù)庫(kù)連接字符串除外。此外,我們通過(guò) ? 操作符在 SQL 中使用 $name 變量。然后,查詢(xún)的數(shù)據(jù)通過(guò) query() 方法末尾的 array 被發(fā)送進(jìn)來(lái)。

#p#

  問(wèn)題 2:不使用自動(dòng)增量功能

  與大多數(shù)現(xiàn)代數(shù)據(jù)庫(kù)一樣,MySQL 能夠在每記錄的基礎(chǔ)上創(chuàng)建自動(dòng)增量惟一標(biāo)識(shí)符。除此之外,我們?nèi)匀粫?huì)看到這樣的代碼,即首先運(yùn)行一個(gè) SELECT 語(yǔ)句來(lái)找到最大的 id,然后將該 id 增 1,并找到一個(gè)新記錄。清單 3 展示了一個(gè)示例壞模式。

  清單 3. Badid.sql

  1. DROP TABLE IF EXISTS users;  
  2. CREATE TABLE users (  
  3.   id MEDIUMINT,  
  4.   login TEXT,  
  5.   password TEXT  
  6. );  
  7.  
  8. INSERT INTO users VALUES ( 1, 'jack''pass' );  
  9. INSERT INTO users VALUES ( 2, 'joan''pass' );  
  10. INSERT INTO users VALUES ( 1, 'jane''pass' );  

  這里的 id 字段被簡(jiǎn)單地指定為整數(shù)。所以,盡管它應(yīng)該是惟一的,我們還是可以添加任何值,如 CREATE 語(yǔ)句后面的幾個(gè) INSERT 語(yǔ)句中所示。清單 4 展示了將用戶(hù)添加到這種類(lèi)型的模式的 PHP 代碼。

  清單 4. Add_user.php

  1. <?php  
  2. require_once("DB.php");  
  3.  
  4. function add_user( $name$pass )  
  5. {  
  6.   $rows = array();  
  7.  
  8.   $dsn = 'mysql://root:password@localhost/bad_badid';  
  9.   $db =& DB::Connect( $dsnarray() );  
  10.   if (PEAR::isError($db)) { die($db->getMessage()); }  
  11.  
  12.   $res = $db->query( "SELECT max(id) FROM users" );  
  13.   $id = null;  
  14.   while$res->fetchInto( $row ) ) { $id = $row[0]; }  
  15.  
  16.   $id += 1;  
  17.  
  18.   $sth = $db->prepare( "INSERT INTO users VALUES(?,?,?)" );  
  19.   $db->execute( $stharray$id$name$pass ) );  
  20.  
  21.   return $id;  
  22. }  
  23.  
  24. $id = add_user( 'jerry''pass' );  
  25.  
  26. var_dump( $id );  
  27. ?>  

  add_user.php 中的代碼首先執(zhí)行一個(gè)查詢(xún)以找到 id 的最大值。然后文件以 id 值加 1 運(yùn)行一個(gè) INSERT 語(yǔ)句。該代碼在負(fù)載很重的服務(wù)器上會(huì)在競(jìng)態(tài)條件中失敗。另外,它也效率低下。

  那么替代方案是什么呢?使用 MySQL 中的自動(dòng)增量特性來(lái)自動(dòng)地為每個(gè)插入創(chuàng)建惟一的 ID。更新后的模式如下所示。

  清單 5. Goodid.php

  1. DROP TABLE IF EXISTS users;  
  2. CREATE TABLE users (  
  3.   id MEDIUMINT NOT NULL AUTO_INCREMENT,  
  4.   login TEXT NOT NULL,  
  5.   password TEXT NOT NULL,  
  6.   PRIMARY KEY( id )  
  7. );  
  8.  
  9. INSERT INTO users VALUES ( null, 'jack''pass' );  
  10. INSERT INTO users VALUES ( null, 'joan''pass' );  
  11. INSERT INTO users VALUES ( null, 'jane''pass' );  

  我們添加了 NOT NULL 標(biāo)志來(lái)指示字段必須不能為空。我們還添加了 AUTO_INCREMENT 標(biāo)志來(lái)指示字段是自動(dòng)增量的,添加 PRIMARY KEY 標(biāo)志來(lái)指示那個(gè)字段是一個(gè) id。這些更改加快了速度。清單 6 展示了更新后的 PHP 代碼,即將用戶(hù)插入表中。

  清單 6. Add_user_good.php

  1. <?php  
  2. require_once("DB.php");  
  3.  
  4. function add_user( $name$pass )  
  5. {  
  6.   $dsn = 'mysql://root:password@localhost/good_genid';  
  7.   $db =& DB::Connect( $dsnarray() );  
  8.   if (PEAR::isError($db)) { die($db->getMessage()); }  
  9.  
  10.   $sth = $db->prepare( "INSERT INTO users VALUES(null,?,?)" );  
  11.   $db->execute( $stharray$name$pass ) );  
  12.  
  13.   $res = $db->query( "SELECT last_insert_id()" );  
  14.   $id = null;  
  15.   while$res->fetchInto( $row ) ) { $id = $row[0]; }  
  16.  
  17.   return $id;  
  18. }  
  19.  
  20. $id = add_user( 'jerry''pass' );  
  21.  
  22. var_dump( $id );  
  23. ?>  

  現(xiàn)在我不是獲得最大的 id 值,而是直接使用 INSERT 語(yǔ)句來(lái)插入數(shù)據(jù),然后使用 SELECT 語(yǔ)句來(lái)檢索最后插入的記錄的 id。該代碼比最初的版本及其相關(guān)模式要簡(jiǎn)單得多,且效率更高。

#p#

  問(wèn)題 3:使用多個(gè)數(shù)據(jù)庫(kù)

  偶爾,我們會(huì)看到一個(gè)應(yīng)用程序中,每個(gè)表都在一個(gè)單獨(dú)的數(shù)據(jù)庫(kù)中。在非常大的數(shù)據(jù)庫(kù)中這樣做是合理的,但是對(duì)于一般的應(yīng)用程序,則不需要這種級(jí)別的分割。此外,不能跨數(shù)據(jù)庫(kù)執(zhí)行關(guān)系查詢(xún),這會(huì)影響使用關(guān)系數(shù)據(jù)庫(kù)的整體思想,更不用說(shuō)跨多個(gè)數(shù)據(jù)庫(kù)管理表會(huì)更困難了。

  那么,多個(gè)數(shù)據(jù)庫(kù)應(yīng)該是什么樣的呢?首先,您需要一些數(shù)據(jù)。清單 7 展示了分成 4 個(gè)文件的這樣的數(shù)據(jù)。

  清單 7. 數(shù)據(jù)庫(kù)文件

  1. Files.sql:  
  2. CREATE TABLE files (  
  3.   id MEDIUMINT,  
  4.   user_id MEDIUMINT,  
  5.   name TEXT,  
  6.   path TEXT  
  7. );  
  8.  
  9. Load_files.sql:  
  10. INSERT INTO files VALUES ( 1, 1, 'test1.jpg''files/test1.jpg' );  
  11. INSERT INTO files VALUES ( 2, 1, 'test2.jpg''files/test2.jpg' );  
  12.  
  13. Users.sql:  
  14. DROP TABLE IF EXISTS users;  
  15. CREATE TABLE users (  
  16.   id MEDIUMINT,  
  17.   login TEXT,  
  18.   password TEXT  
  19. );  
  20.  
  21. Load_users.sql:  
  22. INSERT INTO users VALUES ( 1, 'jack''pass' );  
  23. INSERT INTO users VALUES ( 2, 'jon''pass' );  

  在這些文件的多數(shù)據(jù)庫(kù)版本中,您應(yīng)該將 SQL 語(yǔ)句加載到一個(gè)數(shù)據(jù)庫(kù)中,然后將 users SQL 語(yǔ)句加載到另一個(gè)數(shù)據(jù)庫(kù)中。用于在數(shù)據(jù)庫(kù)中查詢(xún)與某個(gè)特定用戶(hù)相關(guān)聯(lián)的文件的 PHP 代碼如下所示。

  清單 8. Getfiles.php

  1. <?php  
  2. require_once("DB.php");  
  3.  
  4. function get_user( $name )  
  5. {  
  6.   $dsn = 'mysql://root:password@localhost/bad_multi1';  
  7.   $db =& DB::Connect( $dsnarray() );  
  8.   if (PEAR::isError($db)) { die($db->getMessage()); }  
  9.  
  10.   $res = $db->query( "SELECT id FROM users WHERE login=?",  
  11.   array$name ) );  
  12.   $uid = null;  
  13.   while$res->fetchInto( $row ) ) { $uid = $row[0]; }  
  14.  
  15.   return $uid;  
  16. }  
  17.  
  18. function get_files( $name )  
  19. {  
  20.   $uid = get_user( $name );  
  21.  
  22.   $rows = array();  
  23.  
  24.   $dsn = 'mysql://root:password@localhost/bad_multi2';  
  25.   $db =& DB::Connect( $dsnarray() );  
  26.   if (PEAR::isError($db)) { die($db->getMessage()); }  
  27.  
  28.   $res = $db->query( "SELECT * FROM files WHERE user_id=?",  
  29.   array$uid ) );  
  30.   while$res->fetchInto( $row ) ) { $rows[] = $row; }  
  31.  
  32.   return $rows;  
  33. }  
  34.  
  35. $files = get_files( 'jack' );  
  36.  
  37. var_dump( $files );  
  38. ?>  

  get_user 函數(shù)連接到包含用戶(hù)表的數(shù)據(jù)庫(kù)并檢索給定用戶(hù)的 ID。get_files 函數(shù)連接到文件表并檢索與給定用戶(hù)相關(guān)聯(lián)的文件行。

  做所有這些事情的一個(gè)更好辦法是將數(shù)據(jù)加載到一個(gè)數(shù)據(jù)庫(kù)中,然后執(zhí)行查詢(xún),比如下面的查詢(xún)。

  清單 9. Getfiles_good.php

  1. <?php  
  2. require_once("DB.php");  
  3.  
  4. function get_files( $name )  
  5. {  
  6.   $rows = array();  
  7.  
  8.   $dsn = 'mysql://root:password@localhost/good_multi';  
  9.   $db =& DB::Connect( $dsnarray() );  
  10.   if (PEAR::isError($db)) { die($db->getMessage()); }  
  11.  
  12.   $res = $db->query(  
  13.   "SELECT files.* FROM users, files WHERE  
  14.   users.login=? AND users.id=files.user_id",  
  15.   array$name ) );  
  16.   while$res->fetchInto( $row ) ) { $rows[] = $row; }  
  17.  
  18.   return $rows;  
  19. }  
  20.  
  21. $files = get_files( 'jack' );  
  22.  
  23. var_dump( $files );  
  24. ?>  

  該代碼不僅更短,而且也更容易理解和高效。我們不是執(zhí)行兩個(gè)查詢(xún),而是執(zhí)行一個(gè)查詢(xún)。

  盡管該問(wèn)題聽(tīng)起來(lái)有些牽強(qiáng),但是在實(shí)踐中我們通常總結(jié)出所有的表應(yīng)該在同一個(gè)數(shù)據(jù)庫(kù)中,除非有非常迫不得已的理由。

#p#

  問(wèn)題 4:不使用關(guān)系

  關(guān)系數(shù)據(jù)庫(kù)不同于編程語(yǔ)言,它們不具有數(shù)組類(lèi)型。相反,它們使用表之間的關(guān)系來(lái)創(chuàng)建對(duì)象之間的一到多結(jié)構(gòu),這與數(shù)組具有相同的效果。我在應(yīng)用程序中看到的一個(gè)問(wèn)題是,工程師試圖將數(shù)據(jù)庫(kù)當(dāng)作編程語(yǔ)言來(lái)使用,即通過(guò)使用具有逗號(hào)分隔的標(biāo)識(shí)符的文本字符串來(lái)創(chuàng)建數(shù)組。請(qǐng)看下面的模式。

  清單 10. Bad.sql

  1. DROP TABLE IF EXISTS files;  
  2. CREATE TABLE files (  
  3.   id MEDIUMINT,  
  4.   name TEXT,  
  5.   path TEXT  
  6. );  
  7.  
  8. DROP TABLE IF EXISTS users;  
  9. CREATE TABLE users (  
  10.   id MEDIUMINT,  
  11.   login TEXT,  
  12.   password TEXT,  
  13.   files TEXT  
  14. );  
  15.  
  16. INSERT INTO files VALUES ( 1, 'test1.jpg''media/test1.jpg' );  
  17. INSERT INTO files VALUES ( 2, 'test1.jpg''media/test1.jpg' );  
  18. INSERT INTO users VALUES ( 1, 'jack''pass''1,2' );  

  系統(tǒng)中的一個(gè)用戶(hù)可以具有多個(gè)文件。在編程語(yǔ)言中,應(yīng)該使用數(shù)組來(lái)表示與一個(gè)用戶(hù)相關(guān)聯(lián)的文件。在本例中,程序員選擇創(chuàng)建一個(gè) files 字段,其中包含一個(gè)由逗號(hào)分隔的文件 id 列表。要得到一個(gè)特定用戶(hù)的所有文件的列表,程序員必須首先從用戶(hù)表中讀取行,然后解析文件的文本,并為每個(gè)文件運(yùn)行一個(gè)單獨(dú)的 SELECT 語(yǔ)句。該代碼如下所示。

  清單 11. Get.php

  1. <?php  
  2. require_once("DB.php");  
  3.  
  4. function get_files( $name )  
  5. {  
  6.   $dsn = 'mysql://root:password@localhost/bad_norel';  
  7.   $db =& DB::Connect( $dsnarray() );  
  8.   if (PEAR::isError($db)) { die($db->getMessage()); }  
  9.  
  10.   $res = $db->query( "SELECT files FROM users WHERE login=?",  
  11.   array$name ) );  
  12.   $files = null;  
  13.   while$res->fetchInto( $row ) ) { $files = $row[0]; }  
  14.  
  15.   $rows = array();  
  16.  
  17.   foreach( split( ',',$files ) as $file )  
  18.   {  
  19.     $res = $db->query( "SELECT * FROM files WHERE id=?",  
  20.       array$file ) );  
  21.     while$res->fetchInto( $row ) ) { $rows[] = $row; }  
  22.   }  
  23.  
  24.   return $rows;  
  25. }  
  26.  
  27. $files = get_files( 'jack' );  
  28.  
  29. var_dump( $files );  
  30. ?>  

  該技術(shù)很慢,難以維護(hù),且沒(méi)有很好地利用數(shù)據(jù)庫(kù)。惟一的解決方案是重新架構(gòu)模式,以將其轉(zhuǎn)換回到傳統(tǒng)的關(guān)系形式,如下所示。

  清單 12. Good.sql

  1. DROP TABLE IF EXISTS files;  
  2. CREATE TABLE files (  
  3.   id MEDIUMINT,  
  4.   user_id MEDIUMINT,  
  5.   name TEXT,  
  6.   path TEXT  
  7. );  
  8.  
  9. DROP TABLE IF EXISTS users;  
  10. CREATE TABLE users (  
  11.   id MEDIUMINT,  
  12.   login TEXT,  
  13.   password TEXT  
  14. );  
  15.  
  16. INSERT INTO users VALUES ( 1, 'jack''pass' );  
  17. INSERT INTO files VALUES ( 1, 1, 'test1.jpg''media/test1.jpg' );  
  18. INSERT INTO files VALUES ( 2, 1, 'test1.jpg''media/test1.jpg' );  

  這里,每個(gè)文件都通過(guò) user_id 函數(shù)與文件表中的用戶(hù)相關(guān)。這可能與任何將多個(gè)文件看成數(shù)組的人的思想相反。當(dāng)然,數(shù)組不引用其包含的對(duì)象 —— 事實(shí)上,反之亦然。但是在關(guān)系數(shù)據(jù)庫(kù)中,工作原理就是這樣的,并且查詢(xún)也因此要快速且簡(jiǎn)單得多。清單 13 展示了相應(yīng)的 PHP 代碼。

  清單 13. Get_good.php

  1. <?php  
  2. require_once("DB.php");  
  3.  
  4. function get_files( $name )  
  5. {  
  6.   $dsn = 'mysql://root:password@localhost/good_rel';  
  7.   $db =& DB::Connect( $dsnarray() );  
  8.   if (PEAR::isError($db)) { die($db->getMessage()); }  
  9.  
  10.   $rows = array();  
  11.   $res = $db->query(  
  12.     "SELECT files.* FROM users,files WHERE users.login=?  
  13.       AND users.id=files.user_id",  
  14.         array$name ) );  
  15.   while$res->fetchInto( $row ) ) { $rows[] = $row; }  
  16.  
  17.   return $rows;  
  18. }  
  19.  
  20. $files = get_files( 'jack' );  
  21.  
  22. var_dump( $files );  
  23. ?>  

  這里,我們對(duì)數(shù)據(jù)庫(kù)進(jìn)行一次查詢(xún),以獲得所有的行。代碼不復(fù)雜,并且它將數(shù)據(jù)庫(kù)作為其原有的用途使用。

#p#

  問(wèn)題 5:n+1 模式

  我真不知有多少次看到過(guò)這樣的大型應(yīng)用程序,其中的代碼首先檢索一些實(shí)體(比如說(shuō)客戶(hù)),然后來(lái)回地一個(gè)一個(gè)地檢索它們,以得到每個(gè)實(shí)體的詳細(xì)信息。我們將其稱(chēng)為 n+1 模式,因?yàn)椴樵?xún)要執(zhí)行這么多次 —— 一次查詢(xún)檢索所有實(shí)體的列表,然后對(duì)于 n 個(gè)實(shí)體中的每一個(gè)執(zhí)行一次查詢(xún)。當(dāng) n=10 時(shí)這還不成其為問(wèn)題,但是當(dāng) n=100 或 n=1000 時(shí)呢?然后肯定會(huì)出現(xiàn)低效率問(wèn)題。清單 14 展示了這種模式的一個(gè)例子。

  清單 14. Schema.sql

  1. DROP TABLE IF EXISTS authors;  
  2. CREATE TABLE authors (  
  3.   id MEDIUMINT NOT NULL AUTO_INCREMENT,  
  4.   name TEXT NOT NULL,  
  5.   PRIMARY KEY ( id )  
  6. );  
  7.  
  8. DROP TABLE IF EXISTS books;  
  9. CREATE TABLE books (  
  10.   id MEDIUMINT NOT NULL AUTO_INCREMENT,  
  11.   author_id MEDIUMINT NOT NULL,  
  12.   name TEXT NOT NULL,  
  13.   PRIMARY KEY ( id )  
  14. );  
  15.  
  16. INSERT INTO authors VALUES ( null, 'Jack Herrington' );  
  17. INSERT INTO authors VALUES ( null, 'Dave Thomas' );  
  18.  
  19. INSERT INTO books VALUES ( null, 1, 'Code Generation in Action' );  
  20. INSERT INTO books VALUES ( null, 1, 'Podcasting Hacks' );  
  21. INSERT INTO books VALUES ( null, 1, 'PHP Hacks' );  
  22. INSERT INTO books VALUES ( null, 2, 'Pragmatic Programmer' );  
  23. INSERT INTO books VALUES ( null, 2, 'Ruby on Rails' );  
  24. INSERT INTO books VALUES ( null, 2, 'Programming Ruby' );  

  該模式是可靠的,其中沒(méi)有任何錯(cuò)誤。問(wèn)題在于訪問(wèn)數(shù)據(jù)庫(kù)以找到一個(gè)給定作者的所有書(shū)籍的代碼中,如下所示。

  清單 15. Get.php

  1. <?php  
  2. require_once('DB.php');  
  3.  
  4. $dsn = 'mysql://root:password@localhost/good_books';  
  5. $db =& DB::Connect( $dsnarray() );  
  6. if (PEAR::isError($db)) { die($db->getMessage()); }  
  7.  
  8. function get_author_id( $name )  
  9. {  
  10.   global $db;  
  11.  
  12.   $res = $db->query( "SELECT id FROM authors WHERE name=?",  
  13.     array$name ) );  
  14.   $id = null;  
  15.   while$res->fetchInto( $row ) ) { $id = $row[0]; }  
  16.   return $id;  
  17. }  
  18.  
  19. function get_books( $id )  
  20. {  
  21.   global $db;  
  22.  
  23.   $res = $db->query( "SELECT id FROM books WHERE author_id=?",  
  24.     array$id ) );  
  25.   $ids = array();  
  26.   while$res->fetchInto( $row ) ) { $ids []= $row[0]; }  
  27.   return $ids;  
  28. }  
  29.  
  30. function get_book( $id )  
  31. {  
  32.   global $db;  
  33.  
  34.   $res = $db->query( "SELECT * FROM books WHERE id=?"array$id ) );  
  35.   while$res->fetchInto( $row ) ) { return $row; }  
  36.   return null;  
  37. }  
  38.  
  39. $author_id = get_author_id( 'Jack Herrington' );  
  40. $books = get_books( $author_id );  
  41. foreach$books as $book_id ) {  
  42.   $book = get_book( $book_id );  
  43.   var_dump( $book );  
  44. }  
  45. ?>  

  如果您看看下面的代碼,您可能會(huì)想,“嘿,這才是真正的清楚明了。” 首先,得到作者 id,然后得到書(shū)籍列表,然后得到有關(guān)每本書(shū)的信息。的確,它很清楚明了,但是其高效嗎?回答是否定的。看看只是檢索 Jack Herrington 的書(shū)籍時(shí)要執(zhí)行多少次查詢(xún)。一次獲得 id,另一次獲得書(shū)籍列表,然后每本書(shū)執(zhí)行一次查詢(xún)。三本書(shū)要執(zhí)行五次查詢(xún)!

  解決方案是用一個(gè)函數(shù)來(lái)執(zhí)行大量的查詢(xún),如下所示。

  清單 16. Get_good.php

  1. <?php  
  2. require_once('DB.php');  
  3.  
  4. $dsn = 'mysql://root:password@localhost/good_books';  
  5. $db =& DB::Connect( $dsnarray() );  
  6. if (PEAR::isError($db)) { die($db->getMessage()); }  
  7.  
  8. function get_books( $name )  
  9. {  
  10.   global $db;  
  11.  
  12.   $res = $db->query(  
  13.     "SELECT books.* FROM authors,books WHERE  
  14.       books.author_id=authors.id AND authors.name=?",  
  15.       array$name ) );  
  16.   $rows = array();  
  17.   while$res->fetchInto( $row ) ) { $rows []= $row; }  
  18.   return $rows;  
  19. }  
  20.  
  21. $books = get_books( 'Jack Herrington' );  
  22. var_dump( $books );  
  23. ?>  

  現(xiàn)在檢索列表需要一個(gè)快速、單個(gè)的查詢(xún)。這意味著我將很可能必須具有幾個(gè)這些類(lèi)型的具有不同參數(shù)的方法,但是實(shí)在是沒(méi)有選擇。如果您想要具有一個(gè)擴(kuò)展的 PHP 應(yīng)用程序,那么必須有效地使用數(shù)據(jù)庫(kù),這意味著更智能的查詢(xún)。

  本例的問(wèn)題是它有點(diǎn)太清晰了。通常來(lái)說(shuō),這些類(lèi)型的 n+1 或 n*n 問(wèn)題要微妙得多。并且它們只有在數(shù)據(jù)庫(kù)管理員在系統(tǒng)具有性能問(wèn)題時(shí)在系統(tǒng)上運(yùn)行查詢(xún)剖析器時(shí)才會(huì)出現(xiàn)。

  結(jié)束語(yǔ)

  數(shù)據(jù)庫(kù)是強(qiáng)大的工具,就跟所有強(qiáng)大的工具一樣,如果您不知道如何正確地使用就會(huì)濫用它們。識(shí)別和解決這些問(wèn)題的訣竅是更好地理解底層技術(shù)。長(zhǎng)期以來(lái),我老聽(tīng)到業(yè)務(wù)邏輯編寫(xiě)人員抱怨,他們不想要必須理解數(shù)據(jù)庫(kù)或 SQL 代碼。他們把數(shù)據(jù)庫(kù)當(dāng)成對(duì)象使用,并疑惑性能為什么如此之差。

  他們沒(méi)有認(rèn)識(shí)到,理解 SQL 對(duì)于將數(shù)據(jù)庫(kù)從一個(gè)困難的必需品轉(zhuǎn)換成強(qiáng)大的聯(lián)盟是多么重要。如果您每天使用數(shù)據(jù)庫(kù),但是不熟悉 SQL,那么請(qǐng)閱讀 The Art of SQL,這本書(shū)寫(xiě)得很好,實(shí)踐性也很強(qiáng),可以指導(dǎo)您基本了解數(shù)據(jù)庫(kù)。

【編輯推薦】

  1. 51CTO技術(shù)沙龍:PHP自測(cè)題
  2. 2011 PHP技術(shù)峰會(huì)盛大啟航
  3. 敬請(qǐng)期待51CTO技術(shù)沙龍《見(jiàn)微知著 大型網(wǎng)站PHP開(kāi)發(fā)之道》
  4. 史上最強(qiáng)大的PHP Web面試題(會(huì)做可進(jìn)百度)
  5. 優(yōu)秀的PHP開(kāi)發(fā)者是怎樣煉成的?
責(zé)任編輯:韓亞珊 來(lái)源: IBM
相關(guān)推薦

2009-11-18 16:10:00

2011-04-07 16:43:23

MySQL數(shù)據(jù)庫(kù)

2013-07-09 15:54:10

VDI虛擬化

2015-04-17 10:49:35

云數(shù)據(jù)庫(kù)數(shù)據(jù)庫(kù)云服務(wù)

2011-07-12 16:42:41

2011-05-26 14:49:50

ORACLE數(shù)據(jù)庫(kù)

2010-06-12 15:36:01

2009-12-08 17:23:12

PHP PDO類(lèi)

2017-04-18 22:50:10

OSPF疑難問(wèn)題

2017-03-14 13:39:08

2018-01-31 17:50:33

數(shù)據(jù)庫(kù)MySQL優(yōu)化

2018-08-02 15:40:59

2011-04-15 11:29:31

數(shù)據(jù)庫(kù)設(shè)計(jì)

2023-01-24 15:58:39

數(shù)據(jù)庫(kù)架構(gòu)編程語(yǔ)言

2017-04-19 22:58:28

MySQL分布式數(shù)據(jù)

2014-10-21 10:30:33

2018-10-25 08:00:00

數(shù)據(jù)庫(kù)開(kāi)源數(shù)據(jù)庫(kù)開(kāi)源技術(shù)

2010-05-13 11:45:56

MySQL數(shù)據(jù)庫(kù)

2011-03-11 16:25:53

Oracle數(shù)據(jù)庫(kù)

2011-12-09 10:13:15

數(shù)據(jù)庫(kù)加密
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

精品国产aⅴ一区二区三区东京热 久久久久99人妻一区二区三区 | 国语对白在线刺激| 国产精品一区免费在线观看| 欧美极度另类性三渗透| www.久久av| 国产视频一区二| 欧美午夜精品久久久久久浪潮| 日本一区免费在线观看| 国产美女明星三级做爰| 99av国产精品欲麻豆| 日韩中文字幕视频在线观看| 亚洲成人福利视频| 高清成人在线| 亚洲国产日产av| 亚洲免费精品视频| 天天摸天天碰天天爽天天弄| 久久国产精品一区二区| 欧美亚洲免费电影| 黄页网站免费观看| 久久精品国产www456c0m| 亚洲成人激情在线观看| xxww在线观看| 中国色在线日|韩| 一区二区三区 在线观看视频| 欧美专区一二三 | 亚洲美女视频网| 69久久精品无码一区二区| 九九热线视频只有这里最精品| 洋洋成人永久网站入口| 一区二区在线中文字幕电影视频 | 中文字幕亚洲精品| 久久午夜夜伦鲁鲁片| 日本亚洲视频| 9191精品国产综合久久久久久| 中文字幕无码不卡免费视频| av电影在线地址| 亚洲精品成人悠悠色影视| 视频一区二区在线| 色吊丝在线永久观看最新版本| 国产美女主播视频一区| 国产精品香蕉国产| 中文在线观看av| 日韩精品一二三四| 欧美一级视频一区二区| 国产a∨精品一区二区三区仙踪林| 一区二区三区在线观看免费| 最新日韩中文字幕| 手机看片国产日韩| 日韩伦理视频| 最近的2019中文字幕免费一页 | 自拍偷拍精品| 日韩高清人体午夜| 国产草草浮力影院| 久久电影在线| 日韩av综合网站| 国产极品一区二区| 日韩高清一级| 亚洲久久久久久久久久| 成人影视免费观看| 亚洲免费福利一区| 亚洲欧美综合v| 久久精品—区二区三区舞蹈| 精品视频国产| 日韩性xxxx爱| 欧美三级日本三级| 国精品一区二区| 98视频在线噜噜噜国产| 亚洲精品午夜国产va久久成人| 夜久久久久久| 国产精品久久久久高潮| 影音先锋国产在线| 国精产品一区一区三区mba视频| 成人精品一区二区三区电影黑人| av一区二区三| 99久久免费视频.com| 久久亚洲国产精品日日av夜夜| 深夜影院在线观看| 中文字幕av资源一区| 一区二区视频在线观看| 欧美6一10sex性hd| 婷婷开心激情综合| 蜜臀av免费观看| 久久久久久久久久久久电影| 亚洲成色777777在线观看影院| 免费在线观看你懂的| 青草国产精品| 久久久久久一区二区三区| 国产www在线| 久久99精品久久只有精品| 成人av资源网| 国产免费a∨片在线观看不卡| 国产精品久久久久久久蜜臀| 最近免费观看高清韩国日本大全| free性欧美| 欧美日韩一区二区三区免费看| 老女人性生活视频| 日韩动漫一区| 欧美成人精品在线播放| 中文字幕黄色片| 国产精品乡下勾搭老头1| 麻豆成人在线播放| 91福利国产在线观看菠萝蜜| 欧美日韩色婷婷| 99九九99九九九99九他书对| 欧美久久精品| 久久高清视频免费| 中文字幕在线观看视频免费| 国产精品综合视频| 午夜视频久久久| 交100部在线观看| 这里只有精品视频在线观看| 亚洲国产第一区| 亚洲人体av| 国产大片精品免费永久看nba| 亚洲美女福利视频| 国产精品成人网| 成年人视频在线免费| 成人性生交大片免费看中文视频| 色综合亚洲精品激情狠狠| 国产又爽又黄的视频| 国产高清在线精品| 在线免费观看成人网| 日本高清不卡一区二区三区视频 | 西西444www无码大胆| 激情五月***国产精品| 成人黄色网免费| 国产精品一级伦理| 欧美视频免费在线| 日韩少妇一区二区| 韩国av一区| 亚洲综合精品一区二区| 日本最黄一级片免费在线| 色天使色偷偷av一区二区 | 综合色天天鬼久久鬼色| 国产主播中文字幕| 少妇精品久久久一区二区| 91精品国产99久久久久久| 黄色小视频免费观看| 亚洲女爱视频在线| 亚洲制服中文字幕| 久久久久久久久丰满| 欧美涩涩网站| 亚洲毛片在线免费观看| 一级片免费网址| 国产1区2区3区精品美女| 精品国产一区二区三区在线| 久久麻豆视频| zzjj国产精品一区二区| 亚洲一区 中文字幕| 国产精品少妇自拍| 中文字幕免费高清在线| 99久久影视| 亚洲最大激情中文字幕| 日本中文字幕中出在线| 日韩欧美一级在线播放| 国产精品九九九九九九| 国产iv一区二区三区| 国产91沈先生在线播放| 黄色成人美女网站| 欧美在线免费看| 国产系列在线观看| 欧美日韩国产综合视频在线观看| 亚洲天堂av中文字幕| 国产综合色视频| 国产成人艳妇aa视频在线| 国产成人aa在线观看网站站| 欧美在线视频网| 国产在线视频网站| 欧美二区在线观看| 久久久久人妻一区精品色欧美| 波多野结衣在线aⅴ中文字幕不卡| 每日在线观看av| 红桃视频在线观看一区二区| 成人做爰www免费看视频网站| 久久免费电影| 国产亚洲精品久久久久久牛牛| 国产一区二区网站| 亚洲午夜激情网页| 性欧美精品中出| 国产真实乱偷精品视频免| 妞干网在线观看视频| 蜜臀久久99精品久久一区二区| 国产美女久久精品香蕉69| 日韩另类在线| 一本色道久久88精品综合| 国产精选久久久| 欧美日韩国产色视频| 欧美巨胸大乳hitomi| 高清视频一区二区| 日本熟妇人妻中出| 欧美三区视频| 先锋影音亚洲资源| 成人激情自拍| 国产男人精品视频| av中文字幕在线看| y97精品国产97久久久久久| 欧洲精品久久一区二区| 欧美性猛交xxxx乱大交退制版 | 亚洲欧美天堂在线| 99香蕉国产精品偷在线观看 | 欧美日韩免费视频| 国产在线视频卡一卡二| 国产精品免费久久| 极品粉嫩小仙女高潮喷水久久| 久久99精品久久久久久动态图| 黄页网站大全在线观看| 91精品啪在线观看国产81旧版| 精品视频在线观看| 日本免费一区二区视频| 国产精品丝袜久久久久久不卡| 91福利在线免费| 久久久久www| av电影在线观看一区二区三区| 亚洲国产精品国自产拍av秋霞| 国产乱码一区二区| 欧美在线视频日韩| 好看的av在线| 亚洲成人av电影| 久久久久久久久99| 亚洲少妇屁股交4| 日韩不卡av在线| 91捆绑美女网站| 亚洲av熟女高潮一区二区| 国产在线国偷精品免费看| 黑人粗进入欧美aaaaa| 一区二区三区四区五区精品视频 | 久久青青色综合| 欧美人交a欧美精品| 亚洲乱亚洲乱妇| 在线播放日韩av| 黄色片视频在线观看| 日韩精品视频免费专区在线播放| 亚洲国产成人一区二区| 91精品国产福利在线观看| 亚洲自拍偷拍另类| 欧美日韩亚洲综合在线 欧美亚洲特黄一级 | 色94色欧美sute亚洲线路一ni| 日本三级黄色大片| 亚洲成人免费av| 日韩三级小视频| 亚洲v中文字幕| 自拍偷拍欧美亚洲| 欧美午夜片在线免费观看| 日本少妇在线观看| 精品高清一区二区三区| 好看的av在线| 欧美综合视频在线观看| 少妇又紧又色又爽又刺激视频| 欧美性大战久久| 一级成人免费视频| 777xxx欧美| 国产白浆在线观看| 欧美成人在线直播| 天堂在线视频观看| 日韩精品在线观看一区二区| 欧美777四色影视在线| 亚洲精品永久免费| 阿v免费在线观看| xxxx欧美18另类的高清| av网站免费在线观看| 久久青草精品视频免费观看| 欧美少妇网站| 国产精品 欧美在线| 久久精品国产福利| 亚洲自拍偷拍福利| 国偷自产av一区二区三区| 欧美不卡在线一区二区三区| 少妇精品久久久一区二区三区| 日韩影视精品| 色琪琪久久se色| 精品成在人线av无码免费看| 国产日韩亚洲欧美精品| 五月婷婷之综合激情| 国产一区二区三区国产| 中文字幕一区二区人妻电影丶| 久久久久综合网| 国产精品丝袜一区二区| 亚洲高清中文字幕| 成年人视频免费| 欧美一区二区精品久久911| 天堂在线视频网站| 色七七影院综合| 波多野结衣在线观看| 国产成人极品视频| 亚洲天堂av资源在线观看| 久久涩涩网站| 亚洲综合专区| 国产精品97在线| 国产伦精品一区二区三区视频青涩| 国产一卡二卡三卡四卡| 欧美国产欧美综合| 久久精品性爱视频| 欧美日韩一级片网站| 人人妻人人澡人人爽精品日本| 中文字幕亚洲欧美一区二区三区 | 天天影视综合| 免费看国产曰批40分钟| 老司机免费视频一区二区三区| 中文字幕18页| 国产精品萝li| 国产寡妇亲子伦一区二区三区四区| 制服丝袜亚洲精品中文字幕| 青青草视频免费在线观看| 精品国产一区二区三区在线观看| 丁香高清在线观看完整电影视频| 国产精品一区二区久久精品| 欧美jizz19性欧美| 91大学生片黄在线观看| 日韩电影免费一区| 中文字幕在线播放一区| 亚洲精品乱码久久久久久日本蜜臀| 少妇高潮av久久久久久| 精品国产成人系列| caopo在线| 国产精品男女猛烈高潮激情| 欧洲亚洲视频| www.射射射| 国产99久久久精品| 青青操在线播放| 欧洲另类一二三四区| 日本免费一区二区三区最新| 欧美黑人国产人伦爽爽爽| 91精品国产自产观看在线| 日本一区二区精品视频| 宅男噜噜噜66一区二区| 丰满岳乱妇一区二区 | 最近免费中文字幕大全免费版视频| 欧美刺激脚交jootjob| yellow91字幕网在线| 国产视频观看一区| 日韩免费久久| 9久久婷婷国产综合精品性色| 2020国产成人综合网| 亚洲国产成人精品激情在线| 精品国产人成亚洲区| 亚洲区欧洲区| 不卡一区二区三区四区五区| 在线精品小视频| 超碰中文字幕在线观看| 依依成人精品视频| 国产高清免费av| 色综合五月天导航| 亚洲专区**| 丰满的少妇愉情hd高清果冻传媒| 国产精品白丝av| 免费无码毛片一区二区app| 精品久久人人做人人爽| av丝袜在线| 欧美尤物一区| 蜜臀av一级做a爰片久久| 日韩精品久久久久久久的张开腿让| 欧美另类z0zxhd电影| av在线看片| 国产精品一区二区三区在线| 亚洲黄色三级| 麻豆av免费观看| 欧美制服丝袜第一页| aaa在线观看| 91亚洲精品久久久| 伊人成人在线视频| 国产成人精品无码免费看夜聊软件| 欧美中文一区二区三区| 久久久久久国产精品免费无遮挡| 亚洲va国产va天堂va久久| 亚洲天堂男人| 久久久久久国产精品无码| 欧美日韩一级二级三级| 97超碰在线公开在线看免费| 国产一区视频观看| 日韩av不卡在线观看| 懂色av懂色av粉嫩av| 亚洲黄色免费三级| 韩日精品一区| 国产免费内射又粗又爽密桃视频| 成人国产精品免费观看| 免费的毛片视频| 久久国产精品久久精品| 欧美黑人巨大videos精品| 狠狠热免费视频| 一区二区日韩电影| 国产九九在线| 国产精品国产精品国产专区蜜臀ah| 免费视频一区| 全网免费在线播放视频入口| 亚洲毛片在线看| 日本少妇精品亚洲第一区| 玩弄japan白嫩少妇hd| 亚洲欧美日韩国产综合| 天天av天天翘| 成人午夜在线视频一区| 亚洲在线免费| 老女人性淫交视频| 亚洲性视频网站| av毛片精品| 天天爽人人爽夜夜爽| 亚洲国产成人av| 黄网址在线观看| 日本精品免费| 99精品桃花视频在线观看| 国产永久免费视频|