## 泛型
**泛型(Generics)** 是一種重要的編程范式,它允許程序員在編寫代碼時使用類型參數,這些類型參數在編譯時或運行時可以被具體的類型所替代。泛型的使用能夠增加代碼的復用性、靈活性和可維護性。使得這種數據類型能夠適用于不同的數據類型,從而實現代碼的復用和高效。
PHP是一種動態類型語言,不像C++、Java等語言有強類型機制,因此在PHP中實現泛型編程不是一件容易的事情。
## PHP中的泛型
在PHP官方文檔中,并沒有直接提及`泛型`這個概念。這并不意味著PHP不支持泛型,而是說PHP沒有像Java或C#那樣顯式地提供泛型的語法支持。實際上這并不意味著PHP無法實現泛型的功能。
在PHP中,可以通過一些技巧和手段來模擬泛型的行為。例如可以使用接口(Interface)和類型提示(Type Hinting)來實現類似于泛型的功能。通過定義一個接口作為類型參數,我們可以實現類似泛型的類型檢查和類型約束。雖然這種方法與Java或C#中的泛型有所不同,但它確實提供了一種在PHP中實現泛型功能的方式。
一個簡單的例子
```php
<?php
/**
* @desc CollectionInterface
* @author Tinywan(ShaoBo Wan)
*/
declare(strict_types=1);
interface CollectionInterface
{
public function add($element);
public function remove($element);
public function contains($element): bool;
public function size(): int;
}
class ArrayCollection implements CollectionInterface
{
private array $elements = [];
public function add($element)
{
$this->elements[] = $element;
}
public function remove($element)
{
if (($key = array_search($element, $this->elements, true)) !== false) {
unset($this->elements[$key]);
}
}
public function contains($element): bool
{
return in_array($element, $this->elements, true);
}
public function size(): int
{
return count($this->elements);
}
}
```
在上面例子中,使用了一個`CollectionInterface`接口來定義了一個通用的集合接口,然后實現了一個`ArrayCollection`類來實現`CollectionInterface`接口,這個類就可以用于操作任何類型的數據。
## PHP為什么不支持泛型?
這可能與PHP的設計理念和歷史背景有關。**PHP是一種弱類型語言,它允許變量在運行時動態地改變類型。這種靈活性使得PHP在Web開發等領域具有廣泛的應用。實際上這種靈活性也帶來了一些問題,比如類型安全問題。泛型作為一種強類型特性,可以在一定程度上提高代碼的類型安全性。** 但是在PHP這種弱類型語言中引入泛型可能會與其設計理念產生沖突。
PHP最初是為了簡化Web開發而設計的,它的語法和功能都比較簡單和直接。隨著PHP的發展,雖然不斷有新的特性和語法被加入到PHP中,但PHP始終保持著一種簡潔和易用的風格。在這種背景下引入復雜的泛型語法可能會增加PHP的學習成本和開發難度。
## PHP泛型即將來臨
### 完全具體化的泛型
使用泛型,您可以使用占位符定義類的屬性和方法類型。然后可以在創建類的實例時指定這些。這使代碼可重用性和類型安全跨不同的數據類型。`具體化`的泛型是定義泛型類型信息并將其延續到運行時的實現,允許在運行時強制執行泛型需求。
作為PHP語法,這可能看起來像這樣
```php
class Entry<KeyType, ValueType>
{
public function __construct(protected KeyType $key, protected ValueType $value)
{
}
public function getKey(): KeyType
{
return $this->key;
}
public function getValue(): ValueType
{
return $this->value;
}
}
new Entry<int, BlogPost>("123", new BlogPost());
```
在實例化的類中,泛型類型`KeyType`將被替換為`int`,`ValueType`的每個實例將被替換為`BlogPost`,從而導致對象的行為類似于以下類定義:
```php
class IntBlogPostEntry
{
public function __construct(protected int $key, protected BlogPost $value)
{
}
public function getKey(): int
{
return $this->key;
}
public function getValue(): BlogPost
{
return $this->value;
}
}
```
泛型的使用往往會增加代碼的冗長性,因為它要求每次引用泛型類型時都指定類型參數。這在下面的PHP代碼片段中得到了演示:
```php
function f(List<Entry<int,BlogPost>> $entries): Map<int, BlogPost>
{
return new Map<int, BlogPost>($entries);
}
function g(List<BlogPostId> $ids): List<BlogPost>
{
return map<int, BlogPostId, BlogPost>($ids, $repository->find(...));
}
```
類型推斷可以通過讓編譯器自動為我們推斷適當的類型來減少這種冗長。例如,在上面的示例中,編譯器可能會自動確定`new Map()`和`map()`的正確類型。但是,這在PHP中很難實現。引用`Nikita`的話:`主要是由于PHP編譯器對代碼庫的視圖非常有限(它一次只能看到一個文件)`。
請看下面的例子
```php
class Box<T>
{
public function __construct(public T $value) {}
}
new Box(getValue());
```
在這種情況下,`getValue()`表達式的類型是未知的,直到函數在運行時加載,使得無法推斷`new Box(.)`中的`T`。在編譯期間。
我們可以在運行時根據函數的返回值分配T,但這會導致類型不穩定。在前面的例子中,`new Box()`的類型將取決于`getValue()`的返回值的實現,這可能太具體了:聯合收割機結合`Box`是不變的這一事實,當試圖對Box實例做任何有用的事情時,這段代碼將很快中斷:
```php
interface ValueInterface {}
class A implements ValueInterface {}
class B implements ValueInterface {}
function getValue(): ValueInterface
{
return new A();
}
function doSomething(Box<ValueInterface> $box)
{
}
$box = new Box(getValue()) // runtime: Box<A>, statically: Box<ValueInterface>
doSomething($box); // accepts Box<ValueInterface>, not Box<A>
```
當類型基于不依賴于實現的編譯時/靜態信息時,類型是最有用的。
注意:在這個例子中,`Box`是不變的,因為它通常是泛型類的情況。這意味著無論X和Y之間的關系如何,`Box<X>`都不是`Box<Y>`的子類型或超類型,因此Box<A>不是`Box<ValueInterface>`的子類型,并且`doSomething()`不能接受`Box<A>`。
## 集合
泛型的一個主要用例是需要類型化數組。在PHP中,瑞士軍刀數組類型的使用(和濫用)有很多原因。但是你目前不能強制將類型用作鍵或值。
在一個并行項目中,我們一直在研究一種專用的Collections語法,作為完整泛型的一種挑戰性較小的替代方案。
集合有三種形式:集合、序列和字典。集合和序列只定義一個值類型,而字典有鍵和值類型。其語法可以如下所示:
```
class Article
{
public function __construct(public string $subject) {}
}
collection(Seq) Articles<Article>
{
}
collection(Dict) YearBooks<int => Book>
{
}
```
然后你可以像對普通類一樣實例化序列和集合:
```
$a1 = new Articles();
$b1 = new YearBooks();
```
Sequences和Dictionaries將自動[定義許多方法](https://github.com/php/php-src/compare/master...derickr:php-src:collections#diff-eeb1e0848e9a25b7492398bf5ddf9be15995a67d44a23c336869bf9f36910d1b),提供類似PHP已經擁有的大量`array_*`函數的基本功能。如果使用定義的方法來添加或更新集合中的元素,則鍵和值的類型必須與集合中定義的類型相匹配。
在上面的例子中,YearBooks字典的`add()`方法要求使用int作為鍵,Book作為值。對于主要的操作方法(add、get、unset和isset),ArrayAccess風格的重載操作也可以工作,以及潛在的操作符重載。
集合的一個缺點是你需要聲明它們。按照已采用的做法,這意味著每個集合在單獨的文件中有一行聲明。
另一個問題是潛在的更高的內存使用,因為對于每個類,PHP必須保留一個相應的類條目,包括所有相關方法的列表。
第三個問題是兼容類型的集合之間沒有instanceof/is-a關系,例如:
```
class A {}
class B extends A {}
seq As<A> {}
seq Bs<B> {}
new B() instanceof A // true
new Bs() instanceof As // false
```
集合雖然功能不太強大,但在許多用例中可以替代泛型,但沒有太多的復雜性。上面概述的實現也明顯更容易。[還提供了](https://github.com/derickr/php-src/tree/collections)一個實驗性分支。然而,如果發現完整的泛型是可行的并且得到支持,那么直接在標準泛型上實現Seq、Set和Dict將是非常可取的。
- 設計模式系列
- 工廠方法模式
- 序言
- 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