当前位置: 首页 > article >正文

PHP关键字Self、Static和parent的区别详解

在使用PHP代码时您可能经常会遇到parent::、static::和self::。但是当你第一次作为一个开发人员开始的时候有时候你会很困惑不知道它们是做什么的以及它们之间的区别。在我第一次作为开发人员开始工作后的很长一段时间里我认为static::和self::是完全一样的。parent::是什么假设我们有一个BaseTestCase类它有一个setUp方法1234567891011classBaseTestCase{publicfunctionsetUp(): void{echoRun base test case set up here...;}}(newBaseTestCase())-setUp();// Output is: Run base test case set up here...;正如我们所看到的当我们调用 setUp 方法时它按预期运行并输出文本。现在让我们假设我们想要创建一个新的FeatureTest类来继承BaseTestCase类。如果我们想运行FeatureTest类的setUp方法我们可以这样做12345678classFeatureTestextendsBaseTestCase{//}(newFeatureTest())-setUp();// Output is: Run base test case set up here...;正如我们所看到的我们没有在FeatureTest中定义setUp方法所以在BaseTestCase中定义的方法将被运行。现在假设我们想在运行FeatureTest中的setUp方法时运行一些额外的逻辑。例如如果这些类是作为PhpUnit测试的一部分使用的测试用例那么我们可能需要在数据库中创建模型或设置测试值。一开始你可能错误地认为你可以在你的FeatureTest方法中定义setUp方法然后调用$this-setUp()。老实说当我第一次学习编程的时候我总是陷入这个陷阱所以我们的代码可能看起来像这样1234567891011classFeatureTestextendsBaseTestCase{publicfunctionsetUp(): void{$this-setUp();echoRun extra feature test set up here...;}}(newFeatureTest())-setUp();但是您会发现如果我们运行这段代码我们最终会陷入一个循环导致您的应用程序崩溃。这是因为我们递归地要求setUp一遍又一遍地调用它自己。你可能会得到类似这样的输出1234Fatal error: Out of memory (allocated 31457280 bytes) (tried to allocate 262144 bytes) in /in/1MXtt on line 15mmap() failed: [12] Cannot allocate memorymmap() failed: [12] Cannot allocate memoryProcess exited with code 255.因此我们需要告诉PHP在BaseTestCase中使用setUp方法而不是使用$this-setUp()。为了做到这一点我们可以像这样用parent::setUp()替换$this-setUp()12345678910111213classFeatureTestextendsBaseTestCase{publicfunctionsetUp(): void{parent::setUp();echoRun extra feature test set up here...;}}(newFeatureTest())-setUp();// Output is: Run base test case set up here... Run extra feature test set up here...;现在正如你所看到的当我们在FeatureTest类中运行setUp方法时我们首先运行BaseTestCase中的代码然后继续运行子类中定义的其余代码。值得注意的是您并不总是需要将parent::调用放在方法的顶部。实际上您可以将其放置在方法中任何最适合代码目的的位置。例如如果你想先在FeatureTest类中运行你的代码然后在BaseTestCase类中运行你可以像这样将parent::setUp()调用移动到方法的底部self::是什么假设我们有一个Model类它有一个静态的connection属性和一个makeConnection方法。我们还可以想象我们有一个User类它继承了Model类并覆盖了connection属性。这两个类可能看起来像这样1234567891011121314classModel{publicstaticstring$connectionmysql;publicfunctionmakeConnection(): void{echoMaking connection to: .self::$connection;}}classUserextendsModel{publicstaticstring$connectionpostgres;}现在让我们在两个类上运行makeConnection方法看看我们会得到什么输出1234567(newModel())-makeConnection();// Output is: Making connection to mysql(newUser())-makeConnection();// Output is: Making connection to mysql;正如我们所看到的这两个调用都导致了Model类的connection属性被使用。这是因为self使用了在方法所在的类上定义的属性。在这两种情况下makeConnection方法在Model类上是打开的因为User类上不存在一个方法。为了进一步说明这一点我们将向User类添加makeConnection方法如下所示12345678910111213141516171819classModel{publicstaticstring$connectionmysql;publicfunctionmakeConnection(): void{echoMaking connection to: .self::$connection;}}classUserextendsModel{publicstaticstring$connectionpostgres;publicfunctionmakeConnection(): void{echoMaking connection to: .self::$connection;}}现在如果我们再次调用这两个方法我们会得到以下输出1234567(newModel())-makeConnection();// Output is: Making connection to mysql(newUser())-makeConnection();// Output is: Making connection to postgres;正如您所看到的对makeConnection的调用现在将使用User类上的connection字段因为这是该方法存在的地方。static::是什么现在我们已经知道了self::的作用让我们来看看static::。为了更好地理解它的作用让我们更新上面的代码示例使用static::而不是self::如下所示1234567891011121314classModel{publicstatic$connectionmysql;publicfunctionmakeConnection(){echoMaking connection to: .static::$connection;}}classUserextendsModel{publicstatic$connectionpostgres;}如果我们在两个类上运行makeConnection方法我们会得到以下输出1234567(newModel())-makeConnection();// Output is: Making connection to mysql(newUser())-makeConnection();// Output is: Making connection to postgres;正如我们所看到的这个输出与我们之前使用self::$connection时不同。对User类上的makeConnection方法的调用使用了User类上的connection属性而不是Model类该方法实际所属的类。这是由于PHP中一个名为“后期静态绑定”的特性。根据PHP文档这个特性被命名为“后期静态绑定”从内部的角度考虑。“后期绑定”来自这样一个事实即static将不会使用定义方法的类来解析而是使用运行时信息来计算。它也被称为“静态绑定”因为它可以用于但不限于静态方法调用。因此在我们的示例中使用了User类上的connection属性因为我们在同一个类上调用了makeConnection方法。然而值得注意的是如果connection属性在User类上不存在它将回退到使用Model类上的属性。

相关文章:

PHP关键字Self、Static和parent的区别详解

在使用PHP代码时,您可能经常会遇到parent::、static::和self::。但是当你第一次作为一个开发人员开始的时候,有时候你会很困惑,不知道它们是做什么的,以及它们之间的区别。在我第一次作为开发人员开始工作后的很长一段时间里&…...

网站标题优化对SEO排名的影响是什么

网站标题优化对SEO排名的影响是什么 在当今的互联网时代,网站的排名直接影响到其流量和转化率。搜索引擎优化(SEO)是提升网站排名的关键手段之一,而网站标题优化在整个SEO策略中占据重要地位。网站标题优化对SEO排名的影响究竟有…...

OpenClaw镜像体验:Qwen3.5-9B云端沙盒快速入门

OpenClaw镜像体验:Qwen3.5-9B云端沙盒快速入门 1. 为什么选择云端沙盒体验OpenClaw? 第一次接触OpenClaw时,我被它"本地AI智能体"的定位所吸引——一个能在我的电脑上自动执行各种任务的开源框架。但当我真正开始尝试本地部署时&…...

丹青幻境完整使用流程:从铺陈画纸到揭榜留存,一步不漏

丹青幻境完整使用流程:从铺陈画纸到揭榜留存,一步不漏 1. 走进丹青幻境 "见微知著,凝光成影。执笔入画,神游万象。"丹青幻境是一款基于Z-Image架构与Cosplay LoRA技术打造的数字艺术创作工具。它将现代AI算力与传统水…...

STM32串口IAP实现与固件远程更新指南

1. STM32串口IAP实现原理与实战指南IAP(In Application Programming)技术是嵌入式开发中一项非常实用的功能,它允许我们在产品发布后通过预留的通信接口对固件进行远程更新。作为一名嵌入式开发者,我曾在多个工业项目中成功应用串…...

OpenClaw+千问3.5-9B:个人内容助手搭建全流程

OpenClaw千问3.5-9B:个人内容助手搭建全流程 1. 为什么需要个人内容助手 作为一个长期与文字打交道的内容创作者,我经常陷入这样的困境:每天要花大量时间在资料收集、素材整理和初稿撰写上。最痛苦的是,当灵感来临时&#xff0c…...

Arduino UNO R4专用ME310G1通信库:AT封装与低功耗LPWAN集成

1. 项目概述ME310G1 是 CodeZoo 基于 Telit ME310G1-W3 LTE-M/NB-IoT 模块开发的官方 Arduino 通信库,专为 Arduino UNO R4 平台深度适配。该库并非从零构建,而是以 Telit 官方 Charlie Arduino Library 为基础进行系统性重构与硬件抽象层重写&#xff0…...

腾讯混元HY-MT1.5-1.8B翻译模型:开箱即用的本地化部署方案

腾讯混元HY-MT1.5-1.8B翻译模型:开箱即用的本地化部署方案 1. 引言:为什么选择本地化翻译模型 在当今全球化的商业环境中,跨语言沟通已成为日常工作的重要组成部分。传统云端翻译服务虽然方便,但在数据安全、网络依赖和响应速度…...

Qwen3.5-9B图文对话模型5分钟快速部署教程:零基础小白也能搞定

Qwen3.5-9B图文对话模型5分钟快速部署教程:零基础小白也能搞定 1. 引言:为什么选择Qwen3.5-9B Qwen3.5-9B是一款强大的多模态大模型,能够同时处理文本和图像输入,进行智能对话和内容理解。相比前代产品,它具有三大核…...

Wan2.2-I2V-A14B实战案例:自媒体创作者10分钟搭建专属视频生成工具

Wan2.2-I2V-A14B实战案例:自媒体创作者10分钟搭建专属视频生成工具 1. 为什么选择Wan2.2-I2V-A14B 对于自媒体创作者来说,视频内容制作往往是最耗时耗力的环节。传统视频制作需要拍摄、剪辑、特效等多个步骤,而Wan2.2-I2V-A14B模型可以直接…...

StructBERT模型加速技巧:利用GPU CUDA进行批量推理优化

StructBERT模型加速技巧:利用GPU CUDA进行批量推理优化 你是不是也遇到过这样的情况?手头有成千上万条文本需要处理,比如做相似度计算、情感分析或者分类,但用模型一条一条地跑,速度慢得让人抓狂。看着GPU的利用率上不…...

小白友好!DeepSeek-R1-Distill-Qwen-1.5B一键部署对话应用指南

小白友好!DeepSeek-R1-Distill-Qwen-1.5B一键部署对话应用指南 1. 为什么选择这个模型? DeepSeek-R1-Distill-Qwen-1.5B是一款特别适合个人开发者和中小企业使用的轻量级AI对话模型。它最大的特点就是"小而强"——虽然体积小到能在手机上运行…...

SPIRAN ART SUMMONER效果展示:斯皮拉天空岛+浮空船+云层透光体积渲染

SPIRAN ART SUMMONER效果展示:斯皮拉天空岛浮空船云层透光体积渲染 1. 幻光视觉盛宴:斯皮拉世界的惊艳呈现 SPIRAN ART SUMMONER 是一个将先进图像生成技术与经典游戏美学完美融合的创作平台。基于 Flux.1-Dev 模型的核心能力,这个系统能够…...

中科方德V5系统X11vnc安装全攻略:从配置到自启动一步到位

中科方德V5系统X11vnc深度配置指南:解锁远程桌面的高阶玩法 国产操作系统的崛起为技术生态带来了全新选择,中科方德V5作为其中的佼佼者,其安全稳定的特性深受企业用户青睐。当我们需要在这套系统上实现远程桌面控制时,X11vnc凭借其…...

Ostrakon-VL-8B多场景落地:药房药品陈列合规检查自动化方案

Ostrakon-VL-8B多场景落地:药房药品陈列合规检查自动化方案 1. 项目背景与价值 在零售药店日常运营中,药品陈列合规检查是一项重要但繁琐的工作。传统人工巡检方式存在效率低、标准不统一、记录不完整等问题。Ostrakon-VL-8B多模态大模型为解决这一问题…...

从零到一:用锐捷AC热备+VAC实战搭建一个高可用企业无线网络(附配置清单)

企业级无线网络高可用架构实战:锐捷AC热备与VAC深度整合指南 在数字化转型浪潮中,无线网络已成为企业核心生产力工具。某跨国制造企业曾因AC单点故障导致全厂区Wi-Fi瘫痪8小时,直接损失超千万元——这个真实案例揭示了高可用无线架构的必要性…...

从CPU序列号到加密授权:Qt跨平台硬件绑定开发指南(Windows/Linux双平台)

从CPU序列号到加密授权:Qt跨平台硬件绑定开发指南(Windows/Linux双平台) 在工业控制、医疗设备等对软件授权管理要求严格的领域,如何确保软件只能运行在特定设备上是一个关键问题。传统的序列号授权方式容易被复制和传播&#xff…...

当CANopen遇上EtherCAT:用倍福EL6751网关连接伺服驱动器的实战心得

当CANopen遇上EtherCAT:用倍福EL6751网关连接伺服驱动器的实战心得 在工业自动化领域,EtherCAT凭借其高实时性和拓扑灵活性已成为主流总线协议,而CANopen则因其成熟稳定在中小型设备中广泛应用。当需要将支持CANopen协议的伺服驱动器&#xf…...

用ESP32和SSD1680驱动墨水屏,手把手教你做个低功耗电子价签原型

用ESP32和SSD1680打造低功耗电子价签:从硬件选型到云端更新全解析 在零售场景中,电子价签正逐步取代传统纸质标签,成为数字化门店的标配。而基于ESP32和SSD1680驱动墨水屏的方案,凭借其超低功耗、无线更新和低成本优势&#xff0c…...

告别仿真器:手把手教你用树莓派4B+SOEM库驱动真实EtherCAT伺服电机

树莓派4B实战EtherCAT:从零构建工业级伺服控制系统 工业自动化领域的技术迭代从未停歇,而EtherCAT作为实时以太网协议的佼佼者,正逐步取代传统现场总线。但大多数教程停留在仿真阶段,让开发者难以跨越理论与实践的鸿沟。本文将带你…...

Stable-Diffusion-v1-5-archive部署故障排查:端口/服务/日志三步定位法

Stable-Diffusion-v1-5-archive部署故障排查:端口/服务/日志三步定位法 部署 Stable Diffusion v1.5 Archive 镜像后,页面打不开、图片生成失败,是不是让你有点头疼?别急,这通常是服务启动过程中的一些小问题。今天&a…...

STM32高级定时器TIM1互补PWM配置实战:从GPIO初始化到死区时间设置

STM32高级定时器TIM1互补PWM配置实战:从GPIO初始化到死区时间设置 在电机控制、电源管理等工业应用中,互补PWM输出是确保功率器件安全运行的核心技术。STM32的高级定时器TIM1凭借其灵活的互补输出、可编程死区时间和硬件刹车功能,成为这类应用…...

ChatGLM3-6B效果展示:32k长文本流式响应真实对话作品集

ChatGLM3-6B效果展示:32k长文本流式响应真实对话作品集 本文所有对话案例均基于本地部署的ChatGLM3-6B-32k模型生成,展示了真实场景下的智能对话效果 1. 项目核心能力概览 ChatGLM3-6B-32k是一个专门为本地部署优化的智能对话模型,具备三大核…...

OpenClaw硬件配置指南:千问3.5-35B-A3B-FP8本地运行最佳实践

OpenClaw硬件配置指南:千问3.5-35B-A3B-FP8本地运行最佳实践 1. 为什么需要硬件优化? 当我第一次尝试在MacBook Pro M1 Max上运行千问3.5-35B-A3B-FP8模型时,系统几乎立即触发了内存压力警告。风扇开始狂转,而模型响应速度慢得令…...

地址相似度匹配新选择:MGeo镜像5分钟快速部署,支持中文地址实体对齐

地址相似度匹配新选择:MGeo镜像5分钟快速部署,支持中文地址实体对齐 1. 为什么需要专业的地址相似度匹配? 在日常业务中,地址数据往往存在多种表达方式。比如"北京市海淀区中关村大街1号"和"北京海淀中关村大街一…...

基于Chord和LSTM的时序行为分析:运动员动作识别实战

基于Chord和LSTM的时序行为分析:运动员动作识别实战 1. 体育训练正面临一场静悄悄的变革 上周去健身房,看到一位教练用手机拍下学员深蹲的动作,然后打开一个工具反复回放、暂停、比对标准动作。他告诉我:“以前要靠眼睛盯&#…...

Intv_ai_mk11 后端开发实战:构建高并发AI对话API服务

Intv_ai_mk11 后端开发实战:构建高并发AI对话API服务 1. 高并发AI服务的挑战与机遇 想象一下这样的场景:你的AI对话服务刚上线就迎来百万级用户涌入,每秒数千次请求让服务器不堪重负,响应时间从200ms飙升到5秒以上。这不是危言耸…...

Pixel Dimension Fissioner 商业设计案例:为品牌生成动态视觉识别系统素材

Pixel Dimension Fissioner 商业设计案例:为品牌生成动态视觉识别系统素材 1. 动态视觉识别的数字革命 当品牌视觉从静态纸张跃入数字屏幕,传统VI手册里的规范条款突然显得力不从心。去年某国际饮料品牌做过一项调研:在Instagram上&#xf…...

AIGlasses OS Pro在智能导航中的应用:实时道路分割与信号识别实操

AIGlasses OS Pro在智能导航中的应用:实时道路分割与信号识别实操 1. 智能导航技术概述 一副看似普通的智能眼镜,如何实现精准的道路导航和信号识别?这背后是AIGlasses OS Pro智能视觉系统的强大能力在发挥作用。作为专为智能眼镜设计的视觉…...

开箱即用的AI画质增强方案:超清画质增强镜像功能体验与测评

开箱即用的AI画质增强方案:超清画质增强镜像功能体验与测评 1. 引言:为什么需要AI画质增强? 1.1 数字图像面临的挑战 在数字时代,我们每天都会接触到大量低质量图像:老照片褪色模糊、网络图片压缩严重、监控视频分辨…...