PHP程式設計師必須會的 45 個PHP 面試題
Q1: == 和 === 之間有什麼區別?
話題: PHP
困難: ⭐
- 如果是兩個不同的類型,運算符
==
則在兩個不同的類型之間進行強制轉換 ===
操作符執行』類型安全比較『
這意味著只有當兩個操作數具有相同的類型和相同的值時,它才會返回 TRUE。
1 === 1: true 1 == 1: true 1 === "1": false // 1 是一個整數, "1" 是一個字元串 1 == "1": true // "1" 強制轉換為整數,即1 "foo" === "foo": true // 這兩個操作數都是字元串,並且具有相同的值
為了能夠通過引用傳遞變數,我們在其前面使用&,如下所示:
$var1 = &$var2
Q3: $GLOBAL 是什麼意思?
話題: PHP
困難: ⭐$GLOBALS
是關聯數組,包含對腳本全局範圍內當前定義的所有變數的引用。
Q4: ini_set () 有什麼用處?
話題: PHP
困難: ⭐
PHP 允許用戶使用 ini_set () 修改 php.ini 中提到的一些設置。此函數需要兩個字元串參數。第一個是要修改的設置的名稱,第二個是要分配給它的新值。
給定的程式碼行將啟用腳本的 display_error 設置 (如果它被禁用)。
ini_set('display_errors', '1');
我們需要將上面的語句放在腳本的頂部,以便該設置一直保持啟用狀態,直到最後。此外,通過 ini_set () 設置的值僅適用於當前腳本。此後,PHP 將開始使用 php.ini 中的原始值。
Q5: 我應該在什麼時候使用 require 和 include 呢?
話題: PHP
困難: ⭐⭐
require() 函數與 include() 函數相同,只是它處理錯誤的方式不同。如果出現錯誤,include() 函數會生成警告,但腳本會繼續執行。require() 函數會產生致命錯誤,腳本會停止。
我的建議是 99.9% 的時間裡只使用 require_once
。
使用 require
或 include
代替意味著您的程式碼在其他地方不可重用,即您引入的腳本實際上是在執行程式碼,而不是提供類或某些類功能庫。
Q6: PHP 中的 stdClass 是什麼?
主題: PHP
難度: ⭐⭐
stdClass
只是將其他類型強制轉換為對象時使用的通用」 空』』類。stdClass
不是 PHP 中對象的基類。這可以很容易地證明:
class Foo{} $foo = new Foo(); echo ($foo instanceof stdClass)?'Y':'N'; // 輸出'N'
對於匿名對象,動態屬性等很有用。
考慮 StdClass
的一種簡單使用場景是替代關聯數組。請參見下面的示例,該示例顯示 json_decode()
如何允許獲取 StdClass 實例或關聯數組。
同樣但未在本示例中顯示的SoapClient::__soapCall
返回一個StdClass
實例。
//帶有StdClass的示例 $json = '{ "foo": "bar", "number": 42 }'; $stdInstance = json_decode($json); echo $stdInstance - > foo.PHP_EOL; //"bar" echo $stdInstance - > number.PHP_EOL; //42 //Example with associative array $array = json_decode($json, true); echo $array['foo'].PHP_EOL; //"bar" echo $array['number'].PHP_EOL; //42
Q7: PHP 中的 die () 和 exit () 函數有什麼不同?
話題: PHP
困難: ⭐⭐
沒有區別,它們是一樣的。 選擇die()
而不是exit()
的唯一好處可能是你節省了額外鍵入一個字母的時間.
Q8: 它們之間的主要區別是什麼
話題: PHP
困難: ⭐⭐
const
和 define
的根本區別在於,const
在編譯時定義常量,而 define
在運行時定義常量。
const FOO = 'BAR'; define('FOO', 'BAR'); // but if (...) { const FOO = 'BAR'; // 無效 } if (...) { define('FOO', 'BAR'); // 有效 }
同樣在 PHP 5.3 之前,const 命令不能在全局範圍內使用。你只能在類中使用它。當你想要設置與該類相關的某種常量選項或設置時,應使用此選項。或者你可能想要創建某種枚舉。一個好的 const 用法的例子是擺脫了魔術數字。
Define 可以用於相同的目的,但只能在全局範圍內使用。它應該僅用於影響整個應用程式的全局設置。
除非你需要任何類型的條件或表達式定義,否則請使用 consts 而不是 define()—— 這僅僅是為了可讀性!
Q9: isset () 和 array_key_exists () 之間有什麼區別?
話題: PHP
困難: ⭐⭐
- array_key_exists 它會告訴你數組中是否存在鍵,並在 $a 不存在時報錯。
- 如果 key 或變數存在且不是 null,isset 才會返回 true。當 $a 不存在時,isset 不會報錯。
考慮:
$a = array('key1' => 'Foo Bar', 'key2' => null); isset($a['key1']); // true array_key_exists('key1', $a); // true isset($a['key2']); // false array_key_exists('key2', $a); // true
Q10: var_dump () 和 print_r () 有什麼不同?
話題: PHP
困難: ⭐⭐
var_dump
函數用於顯示變數 / 表達式的結構化資訊,包括變數類型和變數值。數組遞歸瀏覽,縮進值以顯示結構。它還顯示哪些數組值和對象屬性是引用。print_r()
函數以我們可讀的方式顯示有關變數的資訊。數組值將以鍵和元素的格式顯示。類似的符號用於對象。
考慮:
$obj = (object) array('qualitypoint', 'technologies', 'India');
var_dump($obj)
將在螢幕的輸出下方顯示:
object(stdClass)#1 (3) { [0]=> string(12) "qualitypoint" [1]=> string(12) "technologies" [2]=> string(5) "India" }
print_r($obj)
將在螢幕的輸出下方顯示。
stdClass Object ( [0] => qualitypoint [1] => technologies [2] => India )
Q11: 解釋不同的 PHP 錯誤是什麼
話題: PHP
困難: ⭐⭐
- notice 不是一個嚴重的錯誤,它說明執行過程中出現了一些錯誤,一些次要的錯誤,比如一個未定義的變數。
- 當出現更嚴重的錯誤,如 include () 命令引入不存在的文件時,會給出警告 warning。 這個錯誤和上面的錯誤發生,腳本都將繼續。
fatal error
致命錯誤將終止程式碼。未能滿足 require () 將生成這種類型的錯誤。
Q12: 如何在 PHP 中啟用錯誤報告?
話題: PHP
困難: ⭐⭐
檢查 php.ini 中的 「display_errors」 是否等於 「on」,或者在腳本中聲明 「ini_set(‘display_error’,1)」。
然後,在你的程式碼中包含 「ERROR_REPORTING(E_ALL)」,以便在腳本執行期間顯示所有類型的錯誤消息。
Q13: 使用默認參數聲明某些函數
話題: PHP
困難: ⭐⭐
思考:
function showMessage($hello = false){ echo ($hello) ? 'hello' : 'bye'; }
Q14: PHP 是否支援多重繼承?
話題: PHP
困難: ⭐⭐
PHP 只支援單一繼承;這意味著使用關鍵字』extended』只能從一個類擴展一個類。
Q15: 在 PHP 中,對象是按值傳遞還是按引用傳遞?
話題: PHP
困難: ⭐⭐
在 PHP 中,通過值傳遞的對象。
Q16:$a != $b 和 $a !== $b ,之間有什麼區別?
話題: PHP
困難: ⭐⭐!=
表示不等於(如果 $a 不等於 $b,則為 True),!==
表示不全等(如果 $a 與 $b 不相同,則為 True).
Q17: 在 PHP 中,什麼是 PDO?
話題: PHP
困難: ⭐⭐
PDO 代表 PHP 數據對象。
它是一組 PHP 擴展,提供核心 PDO 類和資料庫、特定驅動程式。它提供了供應商中立、輕量級的數據訪問抽象層。因此,無論我們使用哪種資料庫,發出查詢和獲取數據的功能都是相同的。它側重於數據訪問抽象,而不是資料庫抽象。
Q18: 說明我們如何在 PHP 中處理異常?
Topic: PHP
Difficulty: ⭐⭐
當程式執行出現異常報錯時,後面的程式碼將不會再執行,這時 PHP 將會嘗試匹配第一個 catch 塊進行異常的處理,如果沒有捕捉到異常程式將會報致命錯誤並顯示」Uncaught Exception」。
可以在 PHP 中拋出和捕獲異常。
為了處理異常,程式碼可以被包圍在」try」 塊中.
每個 try 必須至少有一個對應的catch
塊 。多個不同的 catch 塊可用於捕獲不同類的異常。
在 catch 塊中也可以拋出異常(或重新拋出之前的異常)。
思考:
try { print "this is our try block n"; throw new Exception(); } catch (Exception $e) { print "something went wrong, caught yah! n"; } finally { print "this part is always executed n"; }
Q19: 區分 echo 和 print ()
Topic: PHP
Difficulty: ⭐⭐
echo
和 print
基本上是一樣的。他們都是用來列印輸出數據的。
區別在於:
- echo 沒有返回值,而 print 的返回值為 1,因此 print 可以在表達式中使用。
- echo 可以接受多個參數一起輸出 (但是這種多個的輸出方式很少見),而 print 一次只可以輸出一個參數。
- echo 的輸出比 print 效率要高一些 .
Q20: require_once 和 require 在什麼場景下使用?
Topic: PHP
Difficulty: ⭐⭐⭐
require_once() 作用與 require() 的作用是一樣的,都是引用或包含外部的一個 php 文件,require_once() 引入文件時會檢查文件是否已包含,如果已包含,不再包含 (require) 它。
我建議在 99.9% 的時候要使用require_once
使用require
或include
意味著您的程式碼不可在其他地方重用,即您要拉入的腳本實際上是在執行程式碼,而不是提供類或某些函數庫。
Q21: 判斷 PHP 數組是否是關聯數組
Topic: PHP
Difficulty: ⭐⭐⭐思考:
function has_string_keys(array $array) { return count(array_filter(array_keys($array), 'is_string')) > 0; }
如果$array
至少有一個字元串類型的 key ,它將被視為關聯數組。
Q22: 如何將變數和數據從 PHP 傳至 Javascript
Topic: PHP
Difficulty: ⭐⭐⭐
這裡有幾種實現方法:
- 使用 Ajax 從服務端獲取你需要的數據。
思考get-data.php
echo json_encode(42);
思考index.html:
<script> function reqListener () { console.log(this.responseText); } var oReq = new XMLHttpRequest(); // new 一個請求對象 oReq.onload = function() { // 在這裡你可以操作響應數據 // 真實的數據來自 this.responseText alert(this.responseText); // 將提示: 42 }; oReq.open("get", "get-data.php", true); // ^ 不要阻塞的其餘部分執行。 // 不要等到請求結束再繼續。 oReq.send(); </script>
- 可以在網頁任何地方輸出數據,然後使用 JavaScript 從 DOM 中獲取資訊.
<div id="dom-target" style="display: none;"> <?php $output = "42"; // 此外, 做一些操作,獲得 output. echo htmlspecialchars($output); /* 你必須避免特殊字元,不然結果將是無效HTML。 */ ?> </div> <script> var div = document.getElementById("dom-target"); var myData = div.textContent; </script>
- 直接在 JavaScript 程式碼中 echo 數據。
<script> var data = <?php echo json_encode("42", JSON_HEX_TAG); ?>; // Don't forget the extra semicolon! </script>
Q23: 有一個方法可以複製一個 PHP 數組至另一個數組嗎?
Topic: PHP
Difficulty: ⭐⭐⭐
PHP 數組通過複製進行賦值,而對象通過引用進行賦值。所有默認情況下,PHP 將複製這個數組。這裡有一個 PHP 參考,一目了然:
$a = array(1,2); $b = $a; // $b 是一個不同的數組 $c = &$a; // $c 是 $a 的引用
Q25:此程式碼將返回什麼?解釋結果。
主題:PHP
難度:⭐⭐⭐
考慮程式碼。結果將返回什麼?
$something = 0; echo ('password123' == $something) ? 'true' : 'false';
答案是true
。您永遠不要將其==
用於字元串比較。即使將字元串與字元串進行比較,PHP也會將它們隱式轉換為浮點數,並且如果它們看起來是數字,則進行數字比較。===
還可以
例如
'1e3' == '1000' // true
也返回true。
Q26:array_map,array_walk和array_filter之間到底有什麼區別?
主題:PHP
難度:⭐⭐⭐
array_walk
接受一個數組和一個函數F,並通過用F(x)替換每個元素x對其進行修改。array_map
進行完全相同的操作,除了不代替就地修改,而是返回帶有轉換後元素的新數組。array_filter
使用函數F,而不是變換元素,將刪除F(x)不為真的所有元素
Q27:解釋exec()與system()與passthru()之間的區別?
主題:PHP 難度:⭐⭐⭐
- exec() 用於調用系統命令,也許自己處理輸出。
- system() 用於執行系統命令並立即顯示輸出-大概是文本。
- passthru() 用於執行您希望從中返回原始值的系統命令-大概是二進位文件。
Q28:您將如何使用PHP創建Singleton類?
主題:PHP
難度:⭐⭐⭐
/** * Singleton class * */ final class UserFactory { /** * Call this method to get singleton * * @return UserFactory */ public static function Instance() { static $inst = null; if ($inst === null) { $inst = new UserFactory(); } return $inst; } /** * Private ctor so nobody else can instantiate it * */ private function __construct() { } }
使用方法:
$fact = UserFactory::Instance(); $fact2 = UserFactory::Instance();
但:
$fact = new UserFactory()
引發錯誤。
Q29:PDO的query()與execute()有什麼區別?
主題:PHP
難度:⭐⭐⭐
query
運行標準SQL語句,並要求您正確轉義所有數據,以避免SQL注入和其他問題。execute
運行一個準備好的語句,該語句使您可以綁定參數,以避免需要轉義或引用參數。如果多次重複查詢,execute也將表現更好。
最佳實踐是堅持準備好的語句並執行以提高安全性。除了在客戶端提供轉義功能外,準備好的語句在伺服器端也會編譯一次,然後可以在每次執行時傳遞不同的參數。
問題30:空合併運算符的用途是什麼?
主題:PHP
難度:⭐⭐⭐
空合併運算符如果存在且不為NULL,則返回其第一個操作數。否則,它將返回其第二個操作數。
例:
$name = $firstName ?? $username ?? $placeholder ?? "Guest";
Q31:區分異常和錯誤
主題:PHP
難度:⭐⭐⭐
Error
無法從中恢復。錯誤的唯一解決方案是終止執行。Exception
通過使用try-catch塊或將異常拋出給調用者,您可以從中恢復- 您將無法處理Errors使用try-catch塊。即使您使用try-catch塊處理它們,如果它們發生,您的應用程式也將無法恢復。另一方面,Exceptions可以使用try-catch塊進行處理,並且可以使程式流程正常(如果發生)。
Exceptions
與Errors
運行應用程式的環境有關的應用程式有關。
Q32:什麼是異常類函數?
主題:PHP
難度:⭐⭐⭐
在Exception
類中可以使用以下功能。
- getMessage() −例外消息
- getCode() −例外程式碼
- getFile() -源文件名
- getLine() −源程式碼行
- getTrace() − n個數組 backtrace()
- getTraceAsString() −格式化的跟蹤字元串
- Exception::__toString 給出異常的字元串表示形式。
Q33:區分參數化和非參數化功能
主題:PHP
難度:⭐⭐⭐
- 未參數化的函數在調用時不接受任何參數。
- 參數化函數在調用時採用一個或多個參數。當輸出取決於運行時給定的動態值時,可以在程式的運行時使用它們。有兩種方法可以訪問參數化函數:
- 按值調用:(這裡我們直接傳遞值)
- 按引用調用:(這裡我們傳遞存儲值的地址位置)
Q34:通過引用解釋函數調用
主題:PHP
難度:⭐⭐⭐
在通過引用調用的情況下,如果在函數內部修改了實際值,則將對其進行修改。在這種情況下,我們需要使用&
帶形式參數的符號。在&
表示可變的參考。
例:
function adder(&$str2) { $str2 .= 'Call By Reference'; } $str = 'This is '; adder($str); echo $str;
輸出:
This is Call By Reference
Q35:為什麼我們使用extract()?
主題:PHP
難度:⭐⭐⭐
該extract()
函數將變數從數組導入局部符號表。
此函數將數組鍵用作變數名,將值用作變數值。對於每個元素,它將在當前符號表中創建一個變數。
此函數返回成功提取的變數數。
例:
$a = "Original"; $my_array = array("a" => "Cat","b" => "Dog", "c" => "Horse"); extract($my_array); echo "\$a = $a; \$b = $b; \$c = $c";
輸出:
$a = Cat; $b = Dog; $c = Horse
Q36:解釋什麼是PHP閉包,為什麼它使用「 use」標識符?
主題:PHP
難度:⭐⭐⭐⭐
考慮以下程式碼:
public function getTotal($tax) { $total = 0.00; $callback = function ($quantity, $product) use ($tax, &$total) { $pricePerItem = constant(__CLASS__ . "::PRICE_" . strtoupper($product)); $total += ($pricePerItem * $quantity) * ($tax + 1.0); }; array_walk($this->products, $callback); return round($total, 2); }
您能解釋一下為什麼使用它嗎?
這就是PHP表示閉包的方式。基本上,這意味著您允許匿名函數在其作用域之外「捕獲」局部變數(在本例中為,$tax
並引用$total
)並保留其值(或在$ total的情況下,對$ total本身的引用) )作為匿名函數本身內的狀態。
閉包是一個單獨的命名空間,通常,您不能訪問此命名空間之外定義的變數。
use
允許您訪問(使用)閉包內部的後續變數。use
是早期的約束力。這意味著在定義閉包時將複製變數值。因此,在閉包內部修改$ tax不會產生外部影響,除非它像對象一樣是指針。- 您可以像的情況一樣將變數作為指針傳遞
&$total
。這樣,修改$total
DOES的值會產生外部影響,原始變數的值會更改。
Q37:PHP中最新的靜態綁定到底是什麼?
主題:PHP
難度:⭐⭐⭐⭐
基本上,可以歸結為以下事實:self
關鍵字不遵循相同的繼承規則。self
始終解析為使用它的類。這意味著,如果您在父類中創建方法並從子類中調用該方法,self
則不會像您期望的那樣引用該子項。
後期靜態綁定引入了static
關鍵字的新用法,從而解決了這一特殊缺點。當您使用時static
,它代表您第一次使用它的類,即。它「綁定」到運行時類。
考慮:
class Car { public static function run() { return static::getName(); } private static function getName() { return 'Car'; } } class Toyota extends Car { public static function getName() { return 'Toyota'; } } echo Car::run(); // Output: Car echo Toyota::run(); // Output: Toyota
Q38:如何衡量PHP腳本的執行時間?
主題:PHP
難度:⭐⭐⭐⭐
我想知道執行一次PHP while循環需要多少毫秒。你可以幫幫我嗎?
您可以使用此microtime
功能。
考慮:
$start = microtime(true); while (...) { } $time_elapsed_secs = microtime(true) - $start;
Q39:合併兩個PHP對象的最佳方法是什麼?
主題:PHP
難度:⭐⭐⭐⭐
//We have this: $objectA->a; $objectA->b; $objectB->c; $objectB->d; //We want the easiest way to get: $objectC->a; $objectC->b; $objectC->c; $objectC->d;
這有效:
$obj_merged = (object) array_merge((array) $obj1, (array) $obj2);
你也可以使用array_merge_recursive
有深層複製行為。
另一種方法是:
foreach($objectA as $k => $v) $objectB->$k = $v;
這比<7版本的PHP中的第一個答案要快(估計要快50%)。但是在PHP> = 7中,第一個答案快了400%。
Q40:比較mysqli或PDO-優缺點是什麼?
主題:PHP
難度:⭐⭐⭐⭐
讓我們命名一些:
- PDO是標準,這是大多數開發人員期望使用的標準。
- 將應用程式從一個資料庫遷移到另一個資料庫並不是很常見,但是遲早您可能會發現自己正在使用不同的RDBMS處理另一個項目。如果您在家中使用PDO,那麼到那時至少要少學習一件事。
- 使用PDO的真正好處是您可以獲取數據,並將其自動注入到對象中。
- PDO具有一些有助於SQL注入的功能
- 從執行速度的角度來看,MySQLi會勝出,但是除非您使用MySQLi有很好的包裝器,否則它處理預準備語句的功能將很糟糕。插入-幾乎相等,選擇-mysqli是對於未準備好的報表,速度提高了2.5%/準備語句的速度提高了6.7%。
Q41:<=>運算符有什麼用途?
主題:PHP
難度:⭐⭐⭐⭐
該<=>
運算符將提供組合比較,因為它將:
- 如果任一側的值相等,則返回0
- 如果左側的值較大,則返回1
- 如果右邊的值更大,則返回-1
考慮:
//Comparing Integers echo 1 <= > 1; //outputs 0 echo 3 <= > 4; //outputs -1 echo 4 <= > 3; //outputs 1 //String Comparison echo "x" <= > "x"; // 0 echo "x" <= > "y"; //-1 echo "y" <= > "x"; //1
Q42:PHP是否具有執行緒?
主題:PHP
難度:⭐⭐⭐⭐
標準php不提供任何多執行緒功能,但是有一個(實驗性)擴展名實際上是- pthreads。第二個最好的辦法是讓一個腳本通過CLI執行另一個腳本,但這有點基本。根據您嘗試執行的操作以及操作的複雜程度,這可能是,也可能不是。
Q43:PHP是單執行緒還是多執行緒?
主題:PHP
難度:⭐⭐⭐⭐
PHP本質上不是單執行緒的。但是,在這種情況下,Unix系統上最常見的PHP安裝是單執行緒安裝,最常見的Apache安裝也是如此,而nginx則沒有基於執行緒的體系結構。在最常見的Windows設置和一些更高級的Unix設置中,PHP可以並且確實在一個進程中操作多個解釋器執行緒。
自2000年以來,PHP作為解釋器就支援多執行緒。
Q44:提供一些方法來模仿PHP中的多個構造函數
主題:PHP
難度:⭐⭐⭐⭐⭐
眾所周知,您不能在PHP類中放置兩個具有唯一參數簽名的__construct函數,但我想這樣做:
class Student { protected $id; protected $name; // etc. public function __construct($id){ $this->id = $id; // other members are still uninitialised } public function __construct($row_from_database){ $this->id = $row_from_database->id; $this->name = $row_from_database->name; // etc. } }
用PHP實現此目的的最佳方法是什麼?
我可能會做這樣的事情:
class Student { public function __construct() { // allocate your stuff } public static function withID( $id ) { $instance = new self(); $instance->loadByID( $id ); return $instance; } public static function withRow( array $row ) { $instance = new self(); $instance->fill( $row ); return $instance; } protected function loadByID( $id ) { // do query $row = my_awesome_db_access_stuff( $id ); $this->fill( $row ); } protected function fill( array $row ) { // fill all properties from array } }
然後,如果我想要一個我知道ID的學生:
$student = Student::withID( $id );
從技術上講,您不是在構建多個構造函數,而只是在構建靜態輔助方法,而是通過這種方式避免在構造函數中使用大量義大利麵條式程式碼。
另一種方法是結合使用工廠風格和流利風格:
class Student { protected $firstName; protected $lastName; // etc. /** * Constructor */ public function __construct() { // allocate your stuff } /** * Static constructor / factory */ public static function create() { $instance = new self(); return $instance; } /** * FirstName setter - fluent style */ public function setFirstName( $firstName) { $this->firstName = $firstName; return $this; } /** * LastName setter - fluent style */ public function setLastName( $lastName) { $this->lastName = $lastName; return $this; } } // create instance $student= Student::create()->setFirstName("John")->setLastName("Doe");
Q45:如何在PHP中實現方法重載?
主題:PHP
難度:⭐⭐⭐⭐⭐
您不能重載PHP函數。函數簽名僅基於它們的名稱,並且不包括參數列表,因此不能有兩個具有相同名稱的函數。
但是,您可以聲明一個可變參量,該可變參函數接受可變數量的參數。您將使用func_num_args()
和func_get_arg()
獲取參數,並正常使用它們。
考慮:
function myFunc() { for ($i = 0; $i < func_num_args(); $i++) { printf("Argument %d: %s\n", $i, func_get_arg($i)); } } /* Argument 0: a Argument 1: 2 Argument 2: 3.5 */ myFunc('a', 2, 3.5);
為了處理異常,程式碼可能被包圍在一個try
塊中。
每個 try 必須至少有一個提示。