PHP5-7 結合文檔 可能會使用到的新特性 個人總結


PHP 5-7:

1.改進的性能 - PHPNG代碼合並在PHP7中,這是比 PHP5快兩倍 - 待展開
2.降低內存消耗 - 優化后PHP7使用較少的資源 - 待展開

//PHP7使用新的 Zend Engine 3.0 提高應用程序性能兩倍,比PHP5.6 低了 50% 的內存消耗。 它可以服務於更多的並發用戶,無需任何額外的硬件。
/**
在執行的Magento事務相比PHP5.6,PHP7要快兩倍
在執行 Drupal 事務相比PHP5.6,PHP7證明快兩倍
PHP7 證明自己比 PHP5.6 快兩倍,因為在執行 WordPress 事務相比
*/

3.標量類型聲明 - 現在,參數和返回值類型可以被強制執行(強類型語言)

/**
 支持(string,int,float,bool,interfaces,array,callable,類名) 
 	- 7.1新增 Nullable,void,iterable(與callable類似,實現了Traversavle接口的對象)
 	- 7.2新增 object(可用於逆變(contravariant)參數輸入和協變(covariant)返回任何對象類型)
 	- 7.4支持在類的屬性中添加指定類型。
ps:
 不支持別名,例如boolean,integer。
 callable可理解為可被調用的函數。(可用is_callable()驗證,相對的有callback的概念。)
*/
// Coercive mode 強制模式 (默認) 
function sumOfInts(int ...$ints) //標量類型聲明 int型
{
    return array_sum($ints);
}

var_dump(sumOfInts(2, '3', 4.1)); //int(9)
function arraysSum(array ...$arrays): array //返回值類型聲明 數組型
{
    return array_map(function(array $array): int {
        return array_sum($array);
    }, $arrays);
}

print_r(arraysSum([1,2,3], [4,5,6], [7,8,9])); 
//Array([0]=>6[1]=>15[2]=>24)
declare (strict_types = 1); //開啟嚴格模式,檢查參數的類型,不一致會報錯
//可為空(Nullable)類型(PHP7.1)
//參數以及返回值的類型現在可以通過在類型前加上一個問號使之允許為空。
function testReturn(): ?string {}
//Void 函數(PHP7.1)
//一個新的返回值類型void被引入。 返回值聲明為 void 類型的方法要么干脆省去 return 語句,要么使用一個空的 return 語句。 對於 void 函數來說,null 不是一個合法的返回值。
function swap(&$left, &$right) : void {}
//7.4 有限返回類型協變與參數類型逆變  繼承關系
class A {}
class B extends A {}

class Producer {
    public function method(): A {}
}
class ChildProducer extends Producer {
    public function method(): B {}
}

4.一致性的64位支持 - 64位架構機器持續支持

phpinfo(); //Architecture
//32 位 PHP 是 4Bytes,而 64位 PHP 是 8Bytes
if (PHP_INT_SIZE == 4) {
    $bit = 32;
} else {
    $bit = 64;
}

5.1.改進異常層次結構 - 異常層次結構得到改善;

PHP 7 改變了大多數錯誤的報告方式。不同於傳統(PHP 5)的錯誤報告機制,現在大多數錯誤被作為 Error 異常拋出。

這種 Error 異常可以像 Exception 異常一樣被第一個匹配的 try / catch 塊所捕獲。如果沒有匹配的 catch 塊,則調用異常處理函數(事先通過 set_exception_handler() 注冊)進行處理。 如果尚未注冊異常處理函數,則按照傳統方式處理:被報告為一個致命錯誤(Fatal Error)。

Error 類並非繼承自 Exception 類,所以不能用 catch (Exception e)...來捕獲Error。你可以用catch(Errore) { … },或者通過注冊異常處理函數( set_exception_handler())來捕獲 Error。

/**
異常層次結構
Throwable
 Error
  ArithmeticError
   DivisionByZeroError
  AssertionError
  ParseError
  TypeError
 Exception
 */

5.2.許多致命錯誤轉換成異常 - 異常的范圍增大覆蓋為許多致命的錯誤轉化異常

(5,6)php5.6.x到PHP7.0.x 不向后兼容的變更

<?php
//set_exception_handler() 不再保證收到的一定是 Exception 對象
// PHP 5 時代的代碼將會出現問題
function handler(Exception $e) { ... }
set_exception_handler('handler');

// 兼容 PHP 5 和 7
function handler($e) { ... }

// 僅支持 PHP 7
function handler(Throwable $e) { ... }
?>
//E_STRICT 警告級別變更
場景 新的級別/行為
抽象靜態方法 不再警告,會引發錯誤
重復定義構造器函數 不再警告,會引發錯誤
在繼承的時候,方法簽名不匹配 E_WARNING
在兩個 trait 中包含相同的(兼容的)屬性 不再警告,會引發錯誤
以非靜態調用的方式訪問靜態屬性 E_NOTICE
變量應該以引用的方式賦值 E_NOTICE
變量應該以引用的方式傳遞 E_NOTICE
以靜態方式調用實例方法 E_DEPRECATED

6.已過時的API和擴展刪除 - 不同的舊的和不支持的應用程序和擴展,從最新的版本中刪除

函數
/**
call_user_method(),call)_user_method_array()
所有ereg*函數
mcrypt別名
所有ext/mysql函數
intl別名
......
*/
INI
/**
always_populate_raw_post_data
asp_tags
*/

7.數組,循環等處理改變

/**
修復list()倒序賦值(僅影響數組)
修復數組的生成順序
函數參數附近的括號不再影響行為(一些不允許的函數嵌套行為,5里可以用()處理,7修復了)
修復foreach更改內部指針的問題
foreach 通過值遍歷時,操作的值為數組的副本(無&,對$arr的修改,不會在使用$v時觸發,例如unset($arr[0])后,$arr[0]代表的那個$v仍能被正常使用 ? 5是否也這樣待測試)
foreach通過引用遍歷時,有更好的迭代特性(在迭代中能更好的跟蹤變化)
*/
//修復list()倒序賦值(僅影響數組)
list($a[], $a[], $a[]) = [1, 2, 3];

var_dump($a);
//5 => 3,2,1
//7 => 1,2,3
//修復數組的生成順序
$array = [];
$array["a"] =& $array["b"];
$array["b"] = 1;

var_dump($array);
//5 => ["b"=>&int(1),"a"=>&int(1)]
//7 => ["a"=>&int(1),"b"=>&int(1)]
//函數參數附近的括號不再影響行為(一些不允許的函數嵌套行為,5里可以用()處理,7修復了)
function getArray() {
    return [1, 2, 3];
}

function squareArray(array &$a) {
    foreach ($a as &$v) {
        $v **= 2;
    }
}

// Generates a warning in PHP 7.
squareArray((getArray()));
//修復foreach更改內部指針的問題
$array = [0, 1, 2];
foreach ($array as &$val) {
    var_dump(current($array));
}
//5 => 1,2,false
//7 => 0,0,0
//foreach通過引用遍歷時,有更好的迭代特性(在迭代中能更好的跟蹤變化)
$array = [0];
foreach ($array as &$val) {
    var_dump($val);
    $array[1] = 1;
}
//5 => 0
//7 => 0,1

8.整形,字符串處理改變

INT
/**
無效的8進制字符,現在會報錯(8進制字符的無效數字,之前會被自動刪節)
負位移運算,現在會報錯
超范圍后位移。之前會溢出(原文:這一操作是結構依賴的),現在會始終得到0。
除以0的變化
*/
//除以0的變化
//除法運算符 (/) 會返回一個由 IEEE 754 指定的浮點數:+INF, -INF 或 NAN。取模操作符 (%) 則會拋出一個 DivisionByZeroError 異常,並且不再產生 E_WARNING 錯誤
var_dump(3/0);
var_dump(0/0);
var_dump(0%0);
//5 => false,false,false(三個warning)
//7 => float(INF),float(NAN),Fatal error(前兩個是warning)
String
/**
含十六進制字符串不再被認為是數字(filter_var() 函數可以用於檢查一個 string 是否含有十六進制數字,並將其轉換為integer。返回值為false或int)
支持unicode輸出了,同時也導致對應字符串要慎用 - '\u{'。
*/
//unicode輸出
echo '\u{UNICODE_CODE}'; //輸出對應的unicode字符

9.語法糖

//<=> (太空船操作符|組合比較符)
$a <=> $b; // <:-1,=:0,>:1
//?? (null合並運算符)
// Fetches the value of $_GET['user'] and returns 'nobody'
// if it does not exist.
$username = $_GET['user'] ?? 'nobody';
// This is equivalent to:
$username = isset($_GET['user']) ? $_GET['user'] : 'nobody';

// Coalesces can be chained: this will return the first
// defined value out of $_GET['user'], $_POST['user'], and
// 'nobody'.
$username = $_GET['user'] ?? $_POST['user'] ?? 'nobody';
//通過define定義常量數組(在 PHP5.6 中僅能通過 const 定義)
define('ANIMALS', [
    'dog',
    'cat',
    'bird'
]);

echo ANIMALS[1]; // 輸出 "cat"
//use語句
// Before PHP 7
use com\web3\ClassA;
use com\web3\ClassB;
use com\web3\ClassC as C;

use function com\web3\fn_a;
use function com\web3\fn_b;
use function com\web3\fn_c;

use const com\web3\ConstA;
use const com\web3\ConstB;
use const com\web3\ConstC;

// PHP 7+ code
use com\web3\{ClassA, ClassB, ClassC as C};
use function com\web3\{fn_a, fn_b, fn_c};
use const com\web3\{ConstA, ConstB, ConstC};
//7.1
//短數組語法([])現在作為list()語法的一個備選項,可以用於將數組的值賦給一些變量(包括在foreach中)。
$data = [
    [1, 'Tom'],
    [2, 'Fred'],
];

// list() style
list($id1, $name1) = $data[0];

// [] style
[$id1, $name1] = $data[0];

// list() style
foreach ($data as list($id, $name)) {
    // logic here with $id and $name
}

// [] style
foreach ($data as [$id, $name]) {
    // logic here with $id and $name
}


//list()現在支持鍵名 === 短數組語法也支持
$data = [
    ["id" => 1, "name" => 'Tom'],
    ["id" => 2, "name" => 'Fred'],
];

// list() style
list("id" => $id1, "name" => $name1) = $data[0];

// [] style
["id" => $id1, "name" => $name1] = $data[0];

// list() style
foreach ($data as list("id" => $id, "name" => $name)) {
    // logic here with $id and $name
}

// [] style
foreach ($data as ["id" => $id, "name" => $name]) {
    // logic here with $id and $name
}
//7.1 多異常捕獲
try {
    // some code
} catch (FirstException | SecondException $e) {
    // handle first and second exceptions
}
//7.1 支持為負的字符串偏移量
//現在所有支持偏移量的字符串操作函數 都支持接受負數作為偏移量,包括通過[]或{}操作字符串下標。在這種情況下,一個負數的偏移量會被理解為一個從字符串結尾開始的偏移量。
var_dump("abcdef"[-2]); //e
var_dump(strpos("aabbcc", "b", -3)); //3 從第二個d start開始查找
//7.2 通過名稱加載擴展
//擴展文件不再需要通過文件加載 (Unix下以.so為文件擴展名,在Windows下以 .dll 為文件擴展名) 進行指定。可以在php.ini配置文件進行啟用, 也可以使用 dl() 函數進行啟用。
//7.4 空合並運算賦值符
$array['key'] ??= computeDefault();
// 等同於以下舊寫法
if (!isset($array['key'])) {
    $array['key'] = computeDefault();
}
//7.4 數組展開操作
$parts = ['apple', 'pear'];
$fruits = ['banana', 'orange', ...$parts, 'watermelon'];
// ['banana', 'orange', 'apple', 'pear', 'watermelon'];
//7.4 數值文字分隔符 
//數字文字可以在數字之間包含下划線。
6.674_083e-11; // float
299_792_458;   // decimal
0xCAFE_F00D;   // hexadecimal
0b0101_1111;   // binary
//7.4 箭頭函數
//提供了一種更簡潔的定義函數的方法。
$factor = 10;
$nums = array_map(fn($n) => $n * $factor, [1, 2, 3, 4]);
// $nums = array(10, 20, 30, 40);

//箭頭函數,相比匿名函數,可不使用use,直接獲取其他參數。 function () use () {}
//箭頭函數中所有使用的參數為副本,不會影響其本身

10.其他特性

/**
1.匿名類
2.Closure::call()(Closure::call() 方法加入到臨時綁定(bindTo)的對象范圍,以簡便方式封閉和調用它 它相比綁定到PHP5.6性能更快)
3.過濾unserialize()
4.國際字符(新的 IntlChar 類添加其目的是公開更多的ICU功能。 這個類定義了許多可用於操縱Unicode字符的靜態方法和常數。你需要在使用這個類之前安裝 Intl 擴展)
5.CSPRNG
6.期望(斷言)(期望是向后兼容的增強到舊 assert() 函數。期望允許在生產代碼零成本的斷言,並提供在斷言失敗時拋出自定義異常的能力。assert() 不是一種語言構建體,其中第一個參數是一個表達式的比較字符串或布爾用於測試。)
7.整數除法
8.Session選項
9.生成器
10.類常量可見性(7.1) 現在起支持設置類常量的可見性(面向對象相關)
	- visibility,即public,private,protected
11.允許重寫抽象方法(Abstract method)(7.2)
	- 當一個抽象類繼承於另外一個抽象類的時候,繼承后的抽象類可以重寫被繼承的抽象類的抽象方法
*/
//匿名類(現在支持通過new class 來實例化一個匿名類,這可以用來替代一些“用后即焚”的完整類定義。)
interface Logger {
    public function log(string $msg);
}

class Application {
    private $logger;

    public function getLogger(): Logger {
         return $this->logger;
    }

    public function setLogger(Logger $logger) {
         $this->logger = $logger;
    }
}

$app = new Application;
$app->setLogger(new class implements Logger {
    public function log(string $msg) {
        echo $msg;
    }
});

var_dump($app->getLogger());
//Closure::call()
//before7
<?php
class A {
   private $x = 1;
}

// Define a closure Pre PHP 7 code
$getValue = function() {
   return $this->x;
};

// Bind a clousure
$value = $getValue->bindTo(new A, 'A'); 

print($value()); //1
?>
7+
<?php
class A {
   private $x = 1;
}

// PHP 7+ code, Define
$value = function() {
   return $this->x;
};

print($value->call(new A)); //1
?>
//為unserialize()提供過濾(這個特性旨在提供更安全的方式解包不可靠的數據。它通過白名單的方式來防止潛在的代碼注入。)
//說明: unserialize() 對單一的已序列化的變量進行操作,將其轉換回 PHP 的值。

//文檔的例子
// 將所有的對象都轉換為 __PHP_Incomplete_Class 對象
$data = unserialize($foo, ["allowed_classes" => false]);

// 將除 MyClass 和 MyClass2 之外的所有對象都轉換為 __PHP_Incomplete_Class 對象
$data = unserialize($foo, ["allowed_classes" => ["MyClass", "MyClass2"]);

// 默認情況下所有的類都是可接受的,等同於省略第二個參數
$data = unserialize($foo, ["allowed_classes" => true]);
                           
//別處找的例子
class MyClass1 { 
   public $obj1prop;   
}
class MyClass2 {
   public $obj2prop;
}


$obj1 = new MyClass1();
$obj1->obj1prop = 1;
$obj2 = new MyClass2();
$obj2->obj2prop = 2;

$serializedObj1 = serialize($obj1);
$serializedObj2 = serialize($obj2);

// default behaviour that accepts all classes
// second argument can be ommited.
// if allowed_classes is passed as false, unserialize converts all objects into __PHP_Incomplete_Class object
$data = unserialize($serializedObj1 , ["allowed_classes" => true]);

// converts all objects into __PHP_Incomplete_Class object except those of MyClass1 and MyClass2
$data2 = unserialize($serializedObj2 , ["allowed_classes" => ["MyClass1", "MyClass2"]]);

print($data->obj1prop);
print("<br/>");
print($data2->obj2prop);
//輸出: 1,2
//國際字符
rintf('%x', IntlChar::CODEPOINT_MAX); //10ffff
print (IntlChar::charName('@')); //COMMERCIAL AT
print(IntlChar::ispunct('!')); //true

php5,原有的隨機數為非加密安全偽隨機數。 PHP中的隨機數安全問題 - 知乎

//CSPRNG   在PHP7,以下兩個新的函數引入以產生一個跨平台的方式加密安全整數和字符串。
random_bytes(); //生成加密安全偽隨機字節。適合於使用來生成密碼,密碼學隨機的任意長度的字符串。
random_int(); //生成加密安全偽隨機整數。適合於用於結果是非常重要的加密隨機整數。

斷言,assert,php測試相關。 斷言

//整數除法
$value = intdiv(10,3);
var_dump($value);
print("
");
print($value);
//從PHP7+,session_start()函數接受數組參數覆蓋在php.ini中設置的會話配置指令。這些選項支持 session.lazy,在默認情況下如果PHP會話數據改變,那么會覆蓋任何會話數據信息。

//添加另一種選擇是:read_and_close,這表明會話數據應被讀取,然后該會話應當立即被關閉不變。例如,session.cache_limiter 設置為私有,並設置標志使用下面的代碼片段之后立即關閉會話。
session_start([
   'cache_limiter' => 'private',
   'read_and_close' => true,
]);
//生成器
//在PHP5.5引入生成器的概念。生成器函數每執行一次就得到一個yield標識的值。在PHP7中,當生成器迭代完成后,可以獲取該生成器函數的返回值。通過Generator::getReturn()得到
function generator() {
    yield 1;
    yield 2;
    yield 3;
    return "a";
}
 
$generatorClass = ("generator")();
foreach ($generatorClass as $val) {
    echo $val ." ";
 
}
echo $generatorClass->getReturn();
//1 2 3 a

//在生成器中可以引入另一個或幾個生成器,只需要寫yield from functionName1
function generator1() {
    yield 1;
    yield 2;
    yield from generator2();
    yield from generator3();
}
 
function generator2() {
    yield 3;
    yield 4;
}
 
function generator3() {
    yield 5;
    yield 6;
}
 
foreach (generator1() as $val) {
    echo $val, " ";
}

其他:

curl,http2,套接字,ticks,異步etc

1.debug_zval_dump() 現在打印 “int” 替代 “long”, 打印 “float” 替代 “double”
2.dirname() 增加了可選的第二個參數, depth, 獲取當前目錄向上 depth 級父目錄的名稱。
3.getrusage() 現在支持 Windows.mktime() and gmmktime() 函數不再接受 is_dst 參數。
4.preg_replace() 函數不再支持 “\e” (PREG_REPLACE_EVAL). 應當使用 preg_replace_callback() 替代。
5.setlocale() 函數不再接受 category 傳入字符串。 應當使用 LC_* 常量。
6.exec(), system() and passthru() 函數對 NULL 增加了保護.
7.shmop_open() 現在返回一個資源而非一個int, 這個資源可以傳給shmop_size(), shmop_write(), shmop_read(), shmop_close() 和 shmop_delete().
8.為了避免內存泄露,xml_set_object() 現在在執行結束時需要手動清除 $parse。
9.curl_setopt 設置項CURLOPT_SAFE_UPLOAD變更
TRUE 禁用 @ 前綴在 CURLOPT_POSTFIELDS 中發送文件。 意味着 @ 可以在字段中安全得使用了。 可使用 CURLFile作為上傳的代替。
PHP 5.5.0 中添加,默認值 FALSE。 PHP 5.6.0 改默認值為 TRUE。. PHP 7 刪除了此選項, 必須使用 CURLFile interface 來上傳文件。

如何充分發揮PHP7的性能

1.開啟Opcache

zend_extension=opcache.so
opcache.enable=1
opcache.enable_cli=1

2.使用GCC 4.8以上進行編譯
只有GCC 4.8以上PHP才會開啟Global Register for opline and execute_data支持, 這個會帶來5%左右的性能提升(Wordpres的QPS角度衡量)

3.開啟HugePage (根據系統內存決定)

4.PGO (Profile Guided Optimization)
第一次編譯成功后,用項目代碼去訓練PHP,會產生一些profile信息,最后根據這些信息第二次gcc編譯PHP就可以得到量身定做的PHP7


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM