首页 游戏 软件 资讯 排行榜 专题
首页
编程语言
PHP怎样实现备忘录设计模式_PHP实现备忘录设计模式方法【架构】

PHP怎样实现备忘录设计模式_PHP实现备忘录设计模式方法【架构】

热心网友
28
转载
2026-05-06

PHP备忘录模式:避开序列化陷阱,实现轻量级状态回滚

PHP怎样实现备忘录设计模式_PHP实现备忘录设计模式方法【架构】

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

说到在PHP里实现备忘录模式,一个常见的误区是立刻想到要创建独立的Memento类。但实际情况是,绝大多数场景下,你根本不需要一个单独的Memento类。更简洁、更安全的做法,是直接在业务对象内部添加sa veState()restoreState()方法,用数组来保存关键属性。为什么?因为真实项目里90%的“回滚”需求,其实只是需要恢复某几个字段的值,而不是要完整克隆或冻结整个复杂的对象关系图。生搬硬套教科书上的“发起人-备忘录-管理者”三角色结构,反而会引入不必要的耦合,隐藏序列化风险,并且对PDOStatementClosureDateTime这类“不可靠”成员束手无策。

绝大多数场景下不应单独写Memento类,而应在业务对象中用sa veState()和restoreState()方法以数组存关键属性;因90%回滚只需改几个字段,标准三角色结构反而增加耦合、隐藏序列化风险,且PDO、Closure、DateTime等不可靠项无法安全序列化。

为什么不要直接 serialize($this)

把整个对象序列化保存,看起来是最省事的方案,对吧?但这条路踩坑率极高,堪称“一步一雷”:

资源失效PDO数据库连接、fopen()打开的文件句柄等资源类型,序列化后完全失效,反序列化时甚至会直接报错,例如Serialization of 'PDO' is not allowed
闭包致命:包含Closure(匿名函数)的对象根本无法序列化,尝试serialize()会直接抛出致命错误。
时间陷阱DateTime对象虽然可以序列化,但还原后其内部的时区指针可能发生错位。用var_dump()查看可能一切正常,但调用->format('c')时却会输出错误的时间。
循环引用:如果对象间存在父子互持等循环引用关系,serialize()可能会陷入无限递归,或者静默地截断数据。
私有隐患:所有private属性如果包含了资源句柄或只读逻辑,在恢复后,对象会处于一种“半损坏”状态,调用其方法很可能导致程序崩溃。

怎么安全地保存和恢复状态

核心原则其实很简单:只保存你明确知道需要回滚的字段,其他一律忽略。这样既能控制风险,又能提升性能。

定义保存方法:在业务类中定义一个sa veState()方法,让它返回一个纯粹的数组。这里的关键是,只挑选那些真正需要回溯状态的属性,并且处理好特殊类型。例如,把DateTime对象转换成字符串:

public function sa veState(): array
{
    return [
        'name' => $this->name,
        'status' => $this->status,
        'updated_at' => $this->updated_at instanceof DateTime ? $this->updated_at->format('Y-m-d H:i:s') : null,
    ];
}

实现恢复方法:对应的restoreState(array $state)方法,采用最直接的foreach循环赋值。这样做可以绕过可能存在的__set()魔术方法或自定义setter方法,避免在恢复状态时触发不必要的业务副作用:

public function restoreState(array $state): void
{
    foreach ($state as $key => $value) {
        if (property_exists($this, $key)) {
            $this->$key = $value;
        }
    }
}

牢记几个要点:时间字段统一转为字符串存储;资源型属性(如$this->pdo)根本不要放进数组;而计算属性(例如getFullName()这类方法返回的值)则无需保存,恢复基础字段后它们自然会得到正确结果。

多级撤销用 SplStack 而不是数组堆栈

如果你的需求不止一步回滚,而是需要支持多次undo,那么用SplStack来存储状态快照数组,比使用普通数组更可靠。

值拷贝优势SplStack进行的是值拷贝,而非引用共享。这意味着你修改当前对象的状态时,绝不会意外污染已经存入栈中的历史快照。
操作安全:它不依赖array_push()array_pop()的手动索引管理,避免了数组越界或顺序错乱的潜在风险。
使用示例

$history = new SplStack();
$history->push($obj->sa veState()); // 保存当前状态
$obj->doSomething(); // 执行某些操作,改变状态
$obj->restoreState($history->pop()); // 恢复到最近一次保存的状态

重要提醒:记住,只向SplStack里推送数组、字符串、数字、null这类纯数据。千万不要把DateTime实例或任何对象直接push进去,否则又回到了序列化的老问题上。

哪些情况才值得写完整 Memento

那么,独立备忘录类就完全无用武之地了吗?也不是。但启用它需要满足相当严格的条件,可以说是“不得已而为之”的选择:

强一致性要求:对象内部存在强不变性约束。例如,一个账户对象的“金额”和“状态”必须同步变更,分开存储会破坏业务一致性,必须作为一个整体快照保存。
跨生命周期存续:需要在多个方法调用之间,甚至跨请求生命周期保持完整的中间状态。比如一个复杂的分步表单提交流程,每一步都可能需要回退到上一步的完整状态。
快照即对象:业务逻辑明确要求“快照本身就是一个完整的、可传递的值对象”,并且这个对象本身不持有任何外部依赖(没有PDO、没有文件句柄、没有闭包)。
团队规范保障:团队已经建立了严格的编码规范,确保所有Memento类的构造函数只接收经过校验的数组(白名单模式),绝不允许直接传入原始业务对象,从源头杜绝污染。

即便以上条件都满足,也强烈建议将Memento类标记为final,禁止继承。这是为了防止子类通过扩展偷偷引入不可序列化的属性,让好不容易建立起来的防线功亏一篑。

来源:https://www.php.cn/faq/2321439.html
免责声明: 游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

相关攻略

PHP如何实现数组去重保留键名_PHP实现数组去重保留键名方法【操作】
编程语言
PHP如何实现数组去重保留键名_PHP实现数组去重保留键名方法【操作】

PHP数组去重保留键名:五种方法深度解析 在PHP开发实践中,数组去重是一项常见需求。然而,许多开发者会遇到一个棘手问题:使用常规方法去重后,数组的键名被重新索引,导致原有的关联关系丢失。标准的array_unique()函数在处理关联数组时虽能保留键名,但其默认的字符串比较方式可能引发类型隐式转换

热心网友
05.06
PHP如何防止点击劫持攻击_PHP防止点击劫持攻击方法【安全】
编程语言
PHP如何防止点击劫持攻击_PHP防止点击劫持攻击方法【安全】

PHP如何防止点击劫持攻击:五种协同防护策略详解 如果你的PHP应用页面被发现可以被随意嵌入到第三方网站的iframe中,甚至可能诱导用户进行非本意的操作,那么这很可能就是点击劫持攻击在“敲门”了。这种安全漏洞的危害不容小觑,但好在,我们可以通过一套组合拳来有效防御。下面要介绍的,正是五种经过验证、

热心网友
05.06
PHP函数如何利用非统一内存访问优化_PHP适配NUMA硬件架构【方法】
编程语言
PHP函数如何利用非统一内存访问优化_PHP适配NUMA硬件架构【方法】

PHP函数如何利用非统一内存访问优化_PHP适配NUMA硬件架构【方法】 先说一个核心结论:PHP函数本身,无法直接利用非统一内存访问(NUMA)架构来优化性能。 这听起来可能有点反直觉,但原因在于PHP的运行机制。它运行在Zend虚拟机之上,所有的内存分配,无论是通过glibc的malloc还是P

热心网友
05.06
PHP怎样实现闭包函数传参_PHP实现闭包函数传参方法【函数式】
编程语言
PHP怎样实现闭包函数传参_PHP实现闭包函数传参方法【函数式】

PHP闭包传参:动态输入与固化上下文的双轨制 深入探讨PHP闭包的参数传递机制,其核心可归结为两条相辅相成的路径:动态参数传递与上下文固化捕获。前者在调用闭包时实时传入可变数据,后者则通过use关键字在定义时锁定外部环境变量。这两种方式并非互斥,而是构成了PHP闭包灵活处理数据的“双轨制”,分别应对

热心网友
05.06
PHP怎样实现字符串反转功能_PHP实现字符串反转功能方法【文本】
编程语言
PHP怎样实现字符串反转功能_PHP实现字符串反转功能方法【文本】

PHP怎样实现字符串反转功能_PHP实现字符串功能方法【文本】 在PHP开发中,字符串反转是一个常见且实用的操作需求。无论是处理用户输入、数据格式化还是算法实现,掌握多种字符串反转方法都至关重要。本文将系统性地讲解PHP中实现字符串反转的十二种核心技巧,涵盖从内置函数、基础循环到高级算法与多字节安全

热心网友
05.06

最新APP

宝宝过生日
宝宝过生日
应用辅助 04-07
台球世界
台球世界
体育竞技 04-07
解绳子
解绳子
休闲益智 04-07
骑兵冲突
骑兵冲突
棋牌策略 04-07
三国真龙传
三国真龙传
角色扮演 04-07

热门推荐

史上最长寿标准版!iP17生产周期延长:苹果刀法变了
科技数码
史上最长寿标准版!iP17生产周期延长:苹果刀法变了

iPhone 17:为何成为苹果史上最长寿的爆款? 最近科技圈有个消息传得挺热:iPhone 17标准版的生产周期被大幅拉长了。这可不是简单的产能调整,背后是苹果近期完成的大规模产能扩展。看来,这款热门机型已经瞄准了今年下半年的双11战场,准备再掀一波销售热潮。 消息一出,不少网友都在猜测原因。矛头

热心网友
05.06
小米有品新款mini智能电动平衡车深度体验:便携智能,解锁城市出行新方式
科技数码
小米有品新款mini智能电动平衡车深度体验:便携智能,解锁城市出行新方式

在快节奏的都市生活中,一款兼具便携性与环保特性的出行工具正成为越来越多人的选择 城市通勤的“最后一公里”难题,催生了对灵活出行方案的持续探索。近期,小米有品推出的mini智能电动平衡车,以其独特的设计理念和深度智能化功能,迅速吸引了市场的目光。它不仅仅是一款酷玩装备,更切实地为青少年和上班族提供了高

热心网友
05.06
护眼与智能兼备:科大讯飞AI学习机深度评测,为孩子选对学习好帮手
科技数码
护眼与智能兼备:科大讯飞AI学习机深度评测,为孩子选对学习好帮手

在数字化教育蓬勃发展的当下,家长们为孩子挑选学习设备时,既希望设备具备护眼功能,又期望能满足多样化的学习需求。传统平板电脑功能虽丰富,但长时间使用易引发视力疲劳;普通学习机功能又相对单一,难以契合现代教育的发展趋势。在此背景下,科大讯飞AI学习机系列凭借先进的护眼技术与智能学习系统,成为众多家长和学

热心网友
05.06
以太坊(ETH)财库黑马ETHZilla解析:蒂尔和EF深度加持 mNAV高达6
web3.0
以太坊(ETH)财库黑马ETHZilla解析:蒂尔和EF深度加持 mNAV高达6

目录 ethzilla是谁? ETHZilla独特其他ETH DAT之处 1、Peter Thiel持股ETHZilla近30% 2、Vitalik和以太坊基金会入局 3、聚焦DeFi和链上策略 结语 以太坊财库概念的热度,最近真是肉眼可见。伴随着这股热潮,ETH价格也强势突破了4700美元,距离历

热心网友
05.06
国内彩电一年仅卖2763万台 创10年新低
科技数码
国内彩电一年仅卖2763万台 创10年新低

全球彩电市场:存量博弈下的冰与火之歌 最近,行业调研机构奥维睿沃(A VC Revo)发布了一份引人关注的报告,揭示了2025年全球彩电市场的真实图景。数据显示,全球彩电整体出货量达到2 64亿台,同比仅微跌0 1%,市场基本盘看似稳固。 然而,拆开来看,内部结构正在发生深刻变化。LCD液晶电视依然

热心网友
05.06