使用`declare(strict_types=1)`來獲得更健壯的PHP代碼
## 介紹
如果您是PHP開發人員,您可能在某些PHP文件的開頭看到過`declare(strict_types=1)`語句。
我第一次看到這個聲明時,我不知道它是做什么的。我以為這是某種注釋,或者是我之前的舊PHP語法,但我錯了(大錯特錯!)。
在這篇文章中,我們將介紹什么是`declare(strict_types=1)`,以及它如何幫助您提高PHP代碼的類型安全性。
## declare(strict_types=1) 是什么?
`declare(strict_types=1)`是一個啟用PHP嚴格模式并在PHP應用程序中強制嚴格類型的語句。
它是在PHP 7.0中添加的,當時類型聲明系統首次在PHP中實現。這意味著它可以在PHP 8項目中使用,因此您可以開始在代碼中充分利用嚴格類型。
當你使用這個語句時,PHP會對函數的參數和返回類型進行嚴格的類型檢查。這意味著如果一個函數需要某種類型的參數或返回值,如果使用了錯誤的類型,PHP將拋出錯誤。這也適用于具有指定類型提示和返回類型的PHP閉包和箭頭函數。
讓我們舉一個不使用`declare(strict_types=1)`的簡單例子:
```php
function add(int $a, int $b): int
{
return $a + $b;
}
```
現在假設我們用字符串參數調用這個函數:
```
echo add('1', '2');
// Output:
// 3
```
PHP會很高興地將字符串參數轉換為整數并返回結果`3`。
在某些情況下,您可能完全不介意這種行為。但它可能會產生一些您沒有預料到的意外后果,并可能導致應用程序中的錯誤。
然而,讓我們假設我們想在這個例子中使用`declare(strict_types=1)`。我們可以通過在文件頂部添加以下語句來實現這一點:
```
declare(strict_types=1);
function add(int $a, int $b): int
{
return $a + $b;
}
```
現在,如果我們用字符串參數調用`add`函數,PHP將拋出一個錯誤:
```
echo add('1', '2');
// Output:
// Fatal error: Uncaught TypeError: Argument 1 passed to add() must be of the type int, string given
```
正如我們在這里看到的,PHP拋出了一個錯誤,因為`add`函數期望傳遞整數,但卻接收到了字符串。
類似地,如果啟用了嚴格的類型檢查,并且我們試圖從方法返回錯誤的數據類型,PHP也會拋出錯誤。例如,假設我們的`add`函數現在接受浮點數而不是整數,并且我們沒有啟用嚴格的類型檢查:
```
function add(float $a, float $b): int
{
return $a + $b;
}
```
我們可以這樣調用函數:
```
echo add(1.25, 2.25);
// Output:
// 3
```
你發現輸出中的問題了嗎?
我們應該得到的答案是`3.5`。然而,因為我們已經將返回類型定義為`int`,所以我們已經將浮點數(應該返回的)轉換為整數,并失去了精度。可以想象,這可能會在我們應用程序的其他部分導致一些問題,我們正在使用這個結果,并且可能需要精度。
現在讓我們通過使用`declare(strict_types=1)`來解決這個問題:
```
declare(strict_types=1);
function add(float $a, float $b): int
{
return $a + $b;
}
```
我們可以這樣調用函數:
```
echo add(1.25, 2.25);
// Output:
// Fatal error: Uncaught TypeError: add(): Return value must be of type int, float returned
```
正如我們所看到的,通過啟用嚴格的類型檢查,我們可以發現函數沒有返回與返回類型聲明匹配的正確數據類型。這很好,因為它可以突出顯示我們代碼中可能存在的錯誤,而我們并不知道。然后,我們可以采取必要的步驟:
* 如果返回類型不正確,請更新它們
* 如果類型提示不正確,請更新類型提示
* 如果數據類型不正確,則更新函數體以返回正確的數據類型
* 修復調用函數的代碼中可能向其傳遞錯誤數據類型的任何錯誤
## 我應該使用`declare(strict_types=1)`嗎?
我個人認為,在所有的PHP文件中使用`declare(strict_types=1)`是一個好主意。我曾經認為僅僅有類型提示和返回類型就足以確保傳遞正確的數據類型,但我現在改變了主意。當我使用`declare(strict_types=1)`時,我對我的代碼更有信心,并且由于使用它而發現了一些bug(特別是當將它添加到舊代碼庫時)。
由于PHP是一種動態類型的語言(而不是嚴格類型的語言),這意味著如果你不想的話,你根本不需要指定任何返回類型或類型提示。相反,PHP將在運行時為您確定類型。然而,即使有可能這樣做,我還是強烈建議不要這樣做。如果你不能在代碼中使用嚴格類型(無論出于什么原因),我仍然建議使用類型提示和返回類型作為最低限度來提高PHP代碼質量。
自從了解它以來,我習慣在我創建的每個新PHP文件中使用它。事實上,我更新了PhpStorm設置中的所有模板,以便它自動包含在我創建的每個文件的頂部。例如,下面是創建一個新的PHP類時使用的模板:
```
<?php
declare(strict_types=1);
#parse("PHP File Header.php")
#if (${NAMESPACE})
namespace ${NAMESPACE};
#end
class ${NAME} {
}
```
這真的很方便,因為它鼓勵我繼續使用`declare(strict_types=1)`,而不需要在創建文件后進行任何手動更改(我肯定會忘記這樣做!)。
對于我的任何Laravel閱讀器,您還可以在運行Artisan命令(如`php artisan make:controller`)時發布用于創建PHP文件的存根。通過發布存根,您可以編輯它們并將`declare(strict_types=1)`添加到頂部。這意味著您使用Artisan命令創建的文件將在已啟用更嚴格類型安全的情況下創建。
當然,如果您打算對現有文件添加更嚴格的類型檢查,我強烈建議您首先要有一個高質量的測試套件。您的PHP代碼可能允許傳遞不正確的數據類型而不引發任何錯誤。但是,通過啟用嚴格的類型檢查,您的代碼將變得不那么寬容,并可能開始拋出錯誤。這可能會導致應用程序以用戶意想不到的方式中斷。
您可能還會發現需要重構一些代碼,使其與`declare(strict_types=1)`兼容。但我不認為這是件壞事。相反,我認為這是一個提高代碼質量的機會。
為了幫助您將`declare(strict_types=1)`添加到代碼中,您可能需要使用PHPStan之類的工具,它可以為您收集這些類型不匹配。
- 設計模式系列
- 工廠方法模式
- 序言
- Windows程序注冊為服務的工具WinSW
- 基礎
- 安裝
- 開發規范
- 目錄結構
- 配置
- 快速入門
- 架構
- 請求流程
- 架構總覽
- URL訪問
- 容器和依賴注入
- 中間件
- 事件
- 代碼層結構
- 四個層次
- 路由
- 控制器
- 請求
- 響應
- 數據庫
- MySQL實時同步數據到ES解決方案
- 阿里云DTS數據MySQL同步至Elasticsearch實戰
- PHP中的MySQL連接池
- PHP異步非阻塞MySQL客戶端連接池
- 模型
- 視圖
- 注解
- @SpringBootApplication(exclude={DataSourceAutoConfiguration.calss})
- @EnableFeignClients(basePackages = "com.wotu.feign")
- @EnableAspectJAutoProxy
- @EnableDiscoveryClient
- 錯誤和日志
- 異常處理
- 日志處理
- 調試
- 驗證
- 驗證器
- 驗證規則
- 擴展庫
- 附錄
- Spring框架知識體系詳解
- Maven
- Maven和Composer
- 構建Maven項目
- 實操課程
- 01.初識SpringBoot
- 第1章 Java Web發展史與學習Java的方法
- 第2章 環境與常見問題踩坑
- 第3章 springboot的路由與控制器
- 02.Java編程思想深度理論知識
- 第1章 Java編程思想總體
- 第2章 英雄聯盟的小案例理解Java中最為抽象的概念
- 第3章 徹底理解IOC、DI與DIP
- 03.Spring與SpringBoot理論篇
- 第1章 Spring與SpringBoot導學
- 第2章 Spring IOC的核心機制:實例化與注入
- 第3章 SpringBoot基本配置原理
- 04.SprinBoot的條件注解與配置
- 第1章 conditonal 條件注解
- 第2章 SpringBoot自動裝配解析
- 05.Java異常深度剖析
- 第1章 Java異常分類剖析與自定義異常
- 第2章 自動配置Url前綴
- 06.參數校驗機制與LomBok工具集的使用
- 第1章 LomBok工具集的使用
- 第2章 參數校驗機制以及自定義校驗
- 07.項目分層設計與JPA技術
- 第1章 項目分層原則與層與層的松耦合原則
- 第2章 數據庫設計、實體關系與查詢方案探討
- 第3章 JPA的關聯關系與規則查詢
- 08.ORM的概念與思維
- 第1章 ORM的概念與思維
- 第2章 Banner等相關業務
- 第3章 再談數據庫設計技巧與VO層對象的技巧
- 09.JPA的多種查詢規則
- 第1章 DozerBeanMapper的使用
- 第2章 詳解SKU的規格設計
- 第3章 通用泛型Converter
- 10.令牌與權限
- 第1章 通用泛型類與java泛型的思考
- 常見問題
- 微服務
- demo
- PHP中Self、Static和parent的區別
- Swoole-Cli
- 為什么要使用現代化PHP框架?
- 公眾號
- 一鍵部署微信公眾號Markdown編輯器(支持適配和主題設計)
- Autodesigner 2.0發布
- Luya 一個現代化PHP開發框架
- PHPZip - 創建、讀取和管理 ZIP 文件的簡單庫
- 吊打Golang的PHP界天花板webman壓測對比
- 簡潔而強大的 YAML 解析庫
- 推薦一個革命性的PHP測試框架:Kahlan
- ServBay下一代Web開發環境
- 基于Websocket和Canvas實現多人協作實時共享白板
- Apipost預執行腳本如何調用外部PHP語言
- 認證和授權的安全令牌 Bearer Token
- Laradock PHP 的 Docker 完整本地開發環境
- 高效接口防抖策略,確保數據安全,避免重復提交的終極解決方案!
- TIOBE 6月榜單:PHP穩步前行,編程語言生態的微妙變化
- Aho-Corasick字符串匹配算法的實現
- Redis鍵空間通知 Keyspace Notification 事件訂閱
- ServBay如何啟用并運行Webman項目
- 使用mpdf實現導出pdf文件功能
- Medoo 輕量級PHP數據庫框架
- 在PHP中編寫和運行單元測試
- 9 PHP運行時基準性能測試
- QR碼生成器在PHP中的源代碼
- 使用Gogs極易搭建的自助Git服務
- Gitea
- webman如何記錄SQL到日志?
- Sentry PHP: 實時監測并處理PHP應用程序中的錯誤
- Swoole v6 Alpha 版本已發布
- Proxypin
- Rust實現的Redis內存數據庫發布
- PHP 8.4.0 Alpha 1 測試版本發布
- 121
- Golang + Vue 開發的開源輕量 Linux 服務器運維管理面板
- 內網穿透 FRP VS Tailscale
- 新一代開源代碼托管平臺Gitea
- 微服務系列
- Nacos云原生配置中心介紹與使用
- 輕量級的開源高性能事件庫libevent
- 國密算法
- 國密算法(商用密碼)
- GmSSL 支持國密SM2/SM3/SM4/SM9/SSL 密碼工具箱
- GmSSL PHP 使用
- 數據庫
- SQLite數據庫的Web管理工具
- 阿里巴巴MySQL數據庫強制規范
- PHP
- PHP安全測試秘密武器 PHPGGC
- 使用declare(strict_types=1)來獲得更健壯的PHP代碼
- PHP中的魔術常量
- OSS 直傳阿里騰訊示例
- PHP源碼編譯安裝APCu擴展實現數據緩存
- BI性能DuckDB數據管理系統
- 為什么別人可以是架構師!而我卻不是?
- 密碼還在用 MD5 加鹽?不如試試 password_hash
- Elasticsearch 在電商領域的應用與實踐
- Cron 定時任務入門
- 如何動態設置定時任務!而不是寫死在Linux Crontab
- Elasticsearch的四種查詢方式,你知道多少?
- Meilisearch vs Elasticsearch
- OpenSearch vs Elasticsearch
- Emlog 輕量級開源博客及建站系統
- 現代化PHP原生協程引擎 PRipple
- 使用Zephir編寫C擴展將PHP源代碼編譯加密
- 如何將PHP源代碼編譯加密,同時保證代碼能正常的運行
- 為什么選擇Zephir給PHP編寫動態擴展庫?
- 使用 PHP + XlsWriter實現百萬級數據導入導出
- Rust編寫PHP擴展
- 阿里云盤開放平臺對接進行文件同步
- 如何構建自己的PHP靜態可執行文件
- IM后端架構
- RESTful設計方法和規范
- PHP編譯器BPC 7.3 發布,成功編譯ThinkPHP8
- 高性能的配置管理擴展 Yaconf
- PHP實現雪花算法庫 Snowflake
- PHP官方現代化核心加密庫Sodium
- pie
- 現代化、精簡、非阻塞PHP標準庫PSL
- PHP泛型和集合
- 手把手教你正確使用 Composer包管理
- JWT雙令牌認證實現無感Token自動續期
- 最先進PHP大模型深度學習庫TransformersPHP
- PHP如何啟用 FFI 擴展
- PHP超集語言PXP
- 低延遲雙向實時事件通信 Socket.IO
- PHP OOP中的繼承和多態
- 強大的現代PHP高級調試工具Kint
- PHP基金會
- 基于webman+vue3高質量中后臺框架SaiAdmin
- 開源免費的定時任務管理系統:Gocron
- 簡單強大OCR工具EasyOCR在PHP中使用
- PHP代碼抽象語法樹工具PHP AST Viewer
- MySQL數據庫管理工具PHPMyAdmin
- Rust編寫的一款高性能多人代碼編輯器Zed
- 超高性能PHP框架Workerman v5.0.0-beta.8 發布
- 高并發系列
- 入門介紹及安裝
- Lua腳本開發 Hello World
- 執行流程與階段詳解
- Nginx Lua API 接口開發
- Lua模塊開發
- OpenResty 高性能的正式原因
- 記一次查找 lua-resty-mysql 庫 insert_id 的 bug
- 包管理工具OPM和LuaRocks使用
- 異步非阻塞HTTP客戶端庫 lua-resty-http
- Nginx 內置綁定變量
- Redis協程網絡庫 lua-resty-redis
- 動態HTML渲染庫 lua-testy-template
- 單獨的
- StackBlitz在線開發環境
- AI
- 基礎概念
- 12312
- 基礎鏡像的坑
- 利用phpy實現 PHP 編寫 Vision Transformer (ViT) 模型
- 語義化版本 2.0.0