找回密码
 立即注册
首页 业界区 业界 PHP 现代特性速查 写出更简洁安全的代码(中篇) ...

PHP 现代特性速查 写出更简洁安全的代码(中篇)

旱由 4 小时前
PHP 现代特性速查 写出更简洁安全的代码(中篇)

三部曲第二篇,如果讲怎么用现代 PHP 特性让代码更安全、更快、更好维护。上篇讲了 attributes、命名参数、构造器属性提升、类型化属性、enums、只读 DTO 和一等公民可调用对象。默认你已经看过了——中篇接着往下讲。
下面是 10 个实战特性和惯用法(PHP 7.1 → 8.x),每个都有例子、效果和使用场景。写给想直接上手的人——不是语言八卦。
原文链接
Match 表达式 — 更智能、更安全的 switch

替代了什么:冗长易错的 switch 和意外 fall-through。
什么时候用:值映射、分支返回、替代多 case 的 switch。
  1. // 输入
  2. $statusCode = 404;
  3. // 使用 match (PHP 8.0+)
  4. $statusText = match ($statusCode) {
  5.     200 => 'OK',
  6.     201 => 'Created',
  7.     400, 422 => 'Client Error',
  8.     404 => 'Not Found',
  9.     default => 'Unknown',
  10. };
  11. echo $statusText; // 输出: "Not Found"
复制代码
好在哪

  • 表达式直接返回值,不用临时变量
  • 严格比较(===)不会类型转换
  • 没有 fall-through,每个分支必须有返回
注意:match 用 === 比较——类型要对上。
Null-safe 操作符 (?->) — 安全链式调用

替代了什么:isset() 或 optional() 包装的长链式调用。
什么时候用:读嵌套对象,值可能缺失且返回 null 是合理的。
  1. $user = null;
  2. $country = $user?->profile?->address?->country;
  3. var_dump($country); // 输出: null
复制代码
好在哪:少写判断,避免 Call to a member function on null 异常。service 层、DTO 映射、Blade 模板都能用,意图清楚。
注意:?-> 只读;不能用在数组上,可能掩盖数据模型问题——确定值可能缺失时再用。
Spread 运算符 (...) — 数组合并

替代了什么:冗长的 array_merge() 和手动复制。
什么时候用:合并配置、默认值覆盖、拼接数组。
  1. $defaults = ['color' => 'blue', 'size' => 'M'];
  2. $overrides = ['color' => 'red'];
  3. $config = [...$defaults, ...$overrides];
  4. print_r($config);
  5. // 输出: ['color' => 'red', 'size' => 'M']
复制代码
好在哪:语义清楚,保留键,后面的覆盖前面的。PHP 7.4+ 可用。
注意:只合并一层——嵌套数组要用其他办法。
Variadic 参数和解包 — 灵活参数

替代了什么:笨拙的参数列表和手动转发。
什么时候用:包装函数、日志辅助、函数组合。
  1. function sum(int ...$numbers): int {
  2.     return array_sum($numbers);
  3. }
  4. echo sum(1,2,3); // 输出: 6
  5. // 数组解包成参数
  6. $args = [4,5,6];
  7. echo sum(...$args); // 输出: 15
复制代码
生成器(Generators)— 惰性迭代省内存

替代了什么:Model::all() 或 file() 把大数据集全加载到内存。
什么时候用:流式 CSV 导出、处理日志、分页/分块数据。
  1. function readLargeCsv(string $path): Generator {
  2.     $handle = fopen($path, 'r');
  3.     while (($line = fgetcsv($handle)) !== false) {
  4.         yield $line;
  5.     }
  6.     fclose($handle);
  7. }
  8. // 用法
  9. foreach (readLargeCsv('big.csv') as $row) {
  10.     // 处理行,不会把整个文件加载到内存
  11. }
复制代码
好在哪:内存占用恒定;后台任务和 CLI 脚本必备。
注意:generators 序列化不一样;队列任务要转换一下。
解构赋值(Destructuring)— 快速提取

替代了什么:函数返回数组时冗长的索引访问。
什么时候用:返回多个值、解析 DB 行、配置元组。
  1. function queryStats(): array {
  2.     return ['total' => 100, 'failed' => 2];
  3. }
  4. ['total' => $t, 'failed' => $f] = queryStats();
  5. echo "$t / $f"; // 输出: "100 / 2"
  6. // 数字列表解构
  7. [$a, $b] = [1, 2];
复制代码
好在哪:绑定清楚,少用中间变量。PHP 7.1+ 可用。
命名构造器(Named Constructors)— 明确构造

替代了什么:模糊的 new 表达式、不一致的对象状态。
什么时候用:构造函数有歧义或要表达意图时(如 fromArray、fromCredentials)。
  1. class Money {
  2.     private function __construct(private int $cents) {}
  3.     public static function fromFloat(float $amount): self {
  4.         return new self((int) round($amount * 100));
  5.     }
  6.     public static function fromCents(int $cents): self {
  7.         return new self($cents);
  8.     }
  9. }
  10. $m = Money::fromFloat(12.34);
  11. // 输出: Money 实例,表示 1234 分
复制代码
好在哪:创建方式自解释,支持多种构造方法,验证集中。
assert() — 快速失败检查(仅开发)

替代了什么:静默的无效状态,后来变成难查的 bug。
什么时候用:开发模式检查、关键路径的契约假设。
  1. // 仅开发: 确保值是 int
  2. assert(is_int($count), 'count must be int');
复制代码
好在哪:开销小、表达清楚的快速失败。php.ini 配置 zend.assertions=1 和 assert.exception=1 让开发环境抛异常。
注意:别在生产用 assert() 验证——能被禁用;用类型化属性和显式检查保证运行时安全。
特质(Traits)— 不用继承的代码复用

替代了什么:不相关类之间重复的辅助方法。
什么时候用:共享行为(日志辅助、小工具),继承不合适的地方。
  1. trait LoggerTrait {
  2.     public function log(string $msg): void {
  3.         // 简单日志
  4.         error_log($msg);
  5.     }
  6. }
  7. class PaymentService {
  8.     use LoggerTrait;
  9.     public function pay() { $this->log("paying"); }
  10. }
复制代码
好在哪:复用代码,不用深层继承。
注意:traits 容易变垃圾场——保持专注和小型,避免紧耦合。
匿名类(Anonymous Classes)— 一次性对象

替代了什么:只在一个地方或测试中用的小辅助类。
什么时候用:测试 stub、轻量级策略对象。
  1. // 快速创建实现
  2. $repo = new class implements UserRepoInterface {
  3.     public function find(int $id) { return null; }
  4. };
复制代码
好在哪:少建文件,单个作用域内提供强类型对象。测试替身好用。
组合使用:真实例子

假设 CLI 任务流式处理订单、转换后写入 S3。组合生成器、命名构造器、可变参数和 match:
  1. function streamOrders(Generator $src): Generator {
  2.     foreach ($src as $row) {
  3.         yield Order::fromArray($row); // named ctor
  4.     }
  5. }
  6. // 管道
  7. $orders = streamOrders(readCsv('orders.csv'));
  8. foreach ($orders as $order) {
  9.     $status = match ($order->state) {
  10.         'paid' => 'processing',
  11.         'cancelled' => 'archived',
  12.         default => 'pending'
  13.     };
  14.     // 批量上传 S3 (variadic helper)
  15.     batchUpload(...$order->toParts());
  16. }
复制代码
这种写法省内存、意图明确、可组合。
快速参考 — 什么时候用

特性使用场景match值映射,更安全的 switch?->读嵌套可选属性... (spread)合并配置或数组可变参数转发参数、builder API生成器处理大数据流解构赋值快速绑定多个返回值命名构造器清晰的对象创建assert开发环境契约检查特质共享专注行为匿名类测试替身 / 一次性对象结语

中篇讲了现代 PHP 特性怎么让代码更有表达力、更安全、更省内存。这些不是玩具——用好了能简化推理、减少样板代码。
下篇(第三篇):高级应用模式——依赖注入最佳实践、零停机迁移、性能微优化、生成器、纤程、生产就绪速查表。

来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

相关推荐

您需要登录后才可以回帖 登录 | 立即注册