PHP中区分类的声明属性与动态属性

PHP中区分类的声明属性与动态属性

php中,识别对象属性是预先在类中声明的还是在运行时动态添加的,是一个常见的需求。本文将介绍一种通过结合使用`get_class_vars()`和`get_object_vars()`函数的方法,来精确区分这两种属性,从而帮助开发者更好地理解和调试对象的内部结构。

在PHP面向对象编程中,我们经常会遇到两种类型的对象属性:一种是在类定义时显式声明的属性(Declared Properties),另一种是在对象实例化后,在运行时动态添加的属性(Dynamic Properties)。理解如何区分这两种属性对于代码审查、调试和实现更灵活的对象操作至关重要。

核心方法概述

区分声明属性和动态属性的核心思想是:

获取一个类所有声明的公共、保护和私有属性(不包括静态属性)。获取一个对象实例当前所有可访问的属性(包括声明的属性和动态添加的属性)。通过比较这两组属性的键名,其差异部分即为动态属性。

实现步骤

我们将通过以下步骤来识别动态属性:

1. 获取类中声明的属性

使用 get_class_vars() 函数可以获取指定类中所有已声明的(非静态)默认属性及其初始值。该函数返回一个关联数组,键是属性名,值是属性的默认值。为了使其更具通用性,我们可以结合 get_class() 函数来动态获取对象的类名。

立即学习“PHP免费学习笔记(深入)”;

class Foo {    public $bar; // 声明属性}$obj = new Foo;$obj->baz = 1; // 动态属性// 获取类中声明的属性$declaredProperties = get_class_vars(get_class($obj));echo "声明属性:n";print_r($declaredProperties);

执行上述代码,$declaredProperties 将输出:

声明属性:Array(    [bar] => )

这里 bar 是一个声明属性,其值为 null 因为没有显式初始化。

2. 获取对象实例的所有属性

使用 get_object_vars() 函数可以获取一个对象实例所有可访问的属性。这包括在类中声明的属性以及在运行时动态添加的属性。它也返回一个关联数组,键是属性名,值是属性的当前值。

// ... (接上文代码)// 获取对象实例的所有属性 (声明 + 动态)$allObjectProperties = get_object_vars($obj);echo "n所有对象属性:n";print_r($allObjectProperties);

执行上述代码,$allObjectProperties 将输出:

所有对象属性:Array(    [bar] =>     [baz] => 1)

这里包含了声明属性 bar 和动态属性 baz。

3. 计算动态属性

现在我们有了两组属性:

$declaredProperties: 仅包含类中声明的属性。$allObjectProperties: 包含所有当前对象实例的属性(声明的 + 动态的)。

通过比较 $allObjectProperties 和 $declaredProperties 的键名差异,我们就能找出那些在 $declaredProperties 中不存在,但在 $allObjectProperties 中存在的属性,这些就是动态属性。array_diff_key() 函数非常适合这个任务,它会返回在第一个数组中存在但在第二个数组中不存在的键值对

// ... (接上文代码)// 计算动态属性$dynamicProperties = array_diff_key($allObjectProperties, $declaredProperties);echo "n动态属性:n";print_r($dynamicProperties);

执行上述代码,$dynamicProperties 将输出:

动态属性:Array(    [baz] => 1)

这正是我们期望的动态属性 baz。

完整示例代码

将上述步骤整合起来,形成一个完整的示例:

baz = 1; // 动态属性$obj->qux = 'dynamic value'; // 另一个动态属性echo "--- 原始对象状态 ---n";print_r($obj);// 1. 获取类中声明的属性// 注意:get_class_vars() 仅返回类中声明的公共、保护和私有属性的默认值,// 且不包括静态属性。对于私有和保护属性,在外部访问时可能不可见,// 但它们的键名会出现在这里。$declaredProperties = get_class_vars(get_class($obj));echo "n--- 类中声明的属性 (get_class_vars) ---n";print_r($declaredProperties);// 2. 获取对象实例的所有属性// get_object_vars() 返回所有可访问的实例属性,包括声明的和动态的。// 对于私有和保护属性,只有在当前作用域内(例如在类内部)调用时才能获取到。// 在本例的全局作用域下,它只会返回公共属性和动态属性。$allObjectProperties = get_object_vars($obj);echo "n--- 对象实例的所有属性 (get_object_vars) ---n";print_r($allObjectProperties);// 3. 计算动态属性// array_diff_key() 比较两个数组的键名,返回在第一个数组中存在但在第二个数组中不存在的键值对。$dynamicProperties = array_diff_key($allObjectProperties, $declaredProperties);echo "n--- 识别出的动态属性 ---n";print_r($dynamicProperties);?>

运行上述完整示例,您将看到清晰的输出,区分了声明属性和动态属性:

--- 原始对象状态 ---Foo Object(    [bar] =>     [protectedProp:protected] => protected    [privateProp:Foo:private] => private    [baz] => 1    [qux] => dynamic value)--- 类中声明的属性 (get_class_vars) ---Array(    [bar] =>     [protectedProp] => protected    [privateProp] => private)--- 对象实例的所有属性 (get_object_vars) ---Array(    [bar] =>     [baz] => 1    [qux] => dynamic value)--- 识别出的动态属性 ---Array(    [baz] => 1    [qux] => dynamic value)

请注意,get_object_vars() 在全局作用域下只会返回公共属性和动态属性。如果需要在类内部访问所有属性(包括私有和保护),则需要在类的方法中调用 get_object_vars($this)。然而,对于识别动态属性而言,这种外部调用方式通常足够,因为动态属性总是公共的。

相关函数解析

get_class_vars(string $class_name):功能:获取一个类的默认属性。参数:类的名称(字符串)。返回值:一个关联数组,键是属性名,值是属性的默认值。特点:不包括静态属性。对于私有和保护属性,其键名也会返回。get_object_vars(object $object):功能:获取一个对象实例的所有可访问属性。参数:对象实例。返回值:一个关联数组,键是属性名,值是属性的当前值。特点:在当前作用域下,它只会返回可访问的属性(例如,在类外部调用时,只返回公共属性和动态属性)。get_class(object $object = null):功能:获取一个对象的类名。参数:可选的对象实例。返回值:如果提供了对象,返回其类名字符串;如果没有提供对象或在类内部调用,返回当前类的名称。array_diff_key(array $array1, array $array2, …):功能:比较两个(或更多)数组的键名,并返回在第一个数组中存在但在其他任何数组中都不存在的键值对。参数:至少两个数组。返回值:一个新数组,包含差异部分。

应用场景与注意事项

调试与内省(Introspection): 当需要检查一个对象在运行时有哪些意料之外的属性被添加时,此方法非常有用。API设计与验证: 在处理来自外部系统或用户输入的动态数据时,可以用来验证对象是否只包含预期属性,防止意外的属性注入。魔术方法 __set() 和 __get(): 如果类使用了这些魔术方法来处理未声明的属性访问,此方法可以帮助理解哪些属性实际上是通过魔术方法处理的,哪些是真正被动态添加的。性能考量: 对于大型对象或频繁的操作,这些反射函数可能会带来轻微的性能开销,但在大多数调试和内省场景下,这种开销可以忽略不计。PHP 8.2+ 弃用动态属性: 从PHP 8.2版本开始,不推荐在没有使用 #[AllowDynamicProperties] 属性的情况下动态创建类属性。在PHP 9.0中,这种行为将被彻底移除。这意味着未来识别动态属性的需求可能会减少,或者需要适配新的属性声明方式。对于旧版PHP项目,此方法仍然有效且重要。

总结

通过巧妙地结合使用 get_class_vars() 和 get_object_vars(),我们可以有效地在PHP中区分一个对象的声明属性和动态属性。这种方法为开发者提供了强大的工具,用于对象的内省、调试和更精细的属性管理,尤其是在处理复杂或不确定的对象结构时。然而,随着PHP语言的发展,动态属性的使用模式也在演变,开发者应关注新版本的PHP特性,以编写更健壮和符合未来标准的面向对象代码。

以上就是PHP中区分类的声明属性与动态属性的详细内容,更多请关注php中文网其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1334656.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月12日 20:29:54
下一篇 2025年12月12日 20:30:06

相关推荐

  • 如何直接访问 Sass 地图变量的值?

    直接访问 sass 地图变量的值 在 sass 中,我们可以使用地图变量来存储一组键值对。而有时候,我们可能需要直接访问其中的某个值。 可以通过 map-get 函数直接从地图中获取特定的值。语法如下: map-get($map, $key) 其中: $map 是我们要获取值的 sass 地图变量。…

    2025年12月24日
    000
  • 您不需要 CSS 预处理器

    原生 css 在最近几个月/几年里取得了长足的进步。在这篇文章中,我将回顾人们使用 sass、less 和 stylus 等 css 预处理器的主要原因,并向您展示如何使用原生 css 完成这些相同的事情。 分隔文件 分离文件是人们使用预处理器的主要原因之一。尽管您已经能够将另一个文件导入到 css…

    2025年12月24日
    000
  • React 嵌套组件中,CSS 样式会互相影响吗?

    react 嵌套组件 css 穿透影响 在 react 中,嵌套组件的 css 样式是否会相互影响,取决于采用的 css 解决方案。 传统 css 如果使用传统的 css,在嵌套组件中定义的样式可能会穿透影响到父组件。例如,在给出的代码中: 立即学习“前端免费学习笔记(深入)”; component…

    2025年12月24日
    000
  • React 嵌套组件中父组件 CSS 修饰会影响子组件样式吗?

    对嵌套组件的 CSS 修饰是否影响子组件样式 提问: 在 React 中,如果对嵌套组件 ComponentA 配置 CSS 修饰,是否会影响到其子组件 ComponentB 的样式?ComponentA 是由 HTML 元素(如 div)组成的。 回答: 立即学习“前端免费学习笔记(深入)”; 在…

    2025年12月24日
    000
  • 在 React 项目中实现 CSS 模块

    react 中的 css 模块是一种通过自动生成唯一的类名来确定 css 范围的方法。这可以防止大型应用程序中的类名冲突并允许模块化样式。以下是在 react 项目中使用 css 模块的方法: 1. 设置 默认情况下,react 支持 css 模块。你只需要用扩展名 .module.css 命名你的…

    2025年12月24日
    000
  • 我如何编写 CSS 选择器

    CSS 方法有很多,但我都讨厌它们。有些多(顺风等),有些少(BEM、OOCSS 等)。但归根结底,它们都有缺陷。 当然,人们使用这些方法有充分的理由,并且解决的许多问题我也遇到过。因此,在这篇文章中,我想写下我自己的关于如何保持 CSS 井井有条的指南。 这并不是一个任何人都可以开始使用的完整描述…

    2025年12月24日
    000
  • action在css中的用法

    CSS 中 action 关键字用于定义鼠标悬停或激活元素时的行为,语法:element:action { style-property: value; }。它可以应用于 :hover 和 :active 伪类,用于创建交互效果,如更改元素外观、显示隐藏元素或启动动画。 action 在 CSS 中…

    2025年12月24日
    000
  • css规则的类型有哪些

    CSS 规则包括:通用规则:选择所有元素类型选择器:根据元素类型选择元素类选择器:根据元素的 class 属性选择元素ID 选择器:根据元素的 id 属性选择元素(唯一)后代选择器:选择特定父元素内的元素子选择器:选择作为特定父元素的直接子元素的元素伪类:基于元素的状态或特性选择元素伪元素:创建元素…

    2025年12月24日
    000
  • 网页设计css样式代码大全,快来收藏吧!

    减少很多不必要的代码,html+css可以很方便的进行网页的排版布局。小伙伴们收藏好哦~ 一.文本设置    1、font-size: 字号参数  2、font-style: 字体格式 3、font-weight: 字体粗细 4、颜色属性 立即学习“前端免费学习笔记(深入)”; color: 参数 …

    2025年12月24日
    000
  • css中id选择器和class选择器有何不同

    之前的文章《什么是CSS语法?详细介绍使用方法及规则》中带了解CSS语法使用方法及规则。下面本篇文章来带大家了解一下CSS中的id选择器与class选择器,介绍一下它们的区别,快来一起学习吧!! id选择器和class选择器介绍 CSS中对html元素的样式进行控制是通过CSS选择器来完成的,最常用…

    2025年12月24日
    000
  • php约瑟夫问题如何解决

    “约瑟夫环”是一个数学的应用问题:一群猴子排成一圈,按1,2,…,n依次编号。然后从第1只开始数,数到第m只,把它踢出圈,从它后面再开始数, 再数到第m只,在把它踢出去…,如此不停的进行下去, 直到最后只剩下一只猴子为止,那只猴子就叫做大王。要求编程模拟此过程,输入m、n, 输出最后那个大王的编号。…

    好文分享 2025年12月24日
    000
  • CSS新手整理的有关CSS使用技巧

    [导读]  1、不要使用过小的图片做背景平铺。这就是为何很多人都不用 1px 的原因,这才知晓。宽高 1px 的图片平铺出一个宽高 200px 的区域,需要 200*200=40, 000 次,占用资源。  2、无边框。推荐的写法是     1、不要使用过小的图片做背景平铺。这就是为何很多人都不用 …

    好文分享 2025年12月23日
    000
  • CSS中实现图片垂直居中方法详解

    [导读] 在曾经的 淘宝ued 招聘 中有这样一道题目:“使用纯css实现未知尺寸的图片(但高宽都小于200px)在200px的正方形容器中水平和垂直居中。”当然出题并不是随意,而是有其现实的原因,垂直居中是 淘宝 工作中最 在曾经的 淘宝UED 招聘 中有这样一道题目: “使用纯CSS实现未知尺寸…

    好文分享 2025年12月23日
    000
  • CSS派生选择器

    [导读] 派生选择器通过依据元素在其位置的上下文关系来定义样式,你可以使标记更加简洁。在 css1 中,通过这种方式来应用规则的选择器被称为上下文选择器 (contextual selectors),这是由于它们依赖于上下文关系来应 派生选择器 通过依据元素在其位置的上下文关系来定义样式,你可以使标…

    好文分享 2025年12月23日
    000
  • CSS 基础语法

    [导读] css 语法 css 规则由两个主要的部分构成:选择器,以及一条或多条声明。selector {declaration1; declaration2;     declarationn }选择器通常是您需要改变样式的 html 元素。每条声明由一个属性和一个 CSS 语法 CSS 规则由两…

    2025年12月23日
    300
  • CSS 高级语法

    [导读] 选择器的分组你可以对选择器进行分组,这样,被分组的选择器就可以分享相同的声明。用逗号将需要分组的选择器分开。在下面的例子中,我们对所有的标题元素进行了分组。所有的标题元素都是绿色的。h1,h2,h3,h4,h5 选择器的分组 你可以对选择器进行分组,这样,被分组的选择器就可以分享相同的声明…

    好文分享 2025年12月23日
    000
  • CSS id 选择器

    [导读] id 选择器id 选择器可以为标有特定 id 的 html 元素指定特定的样式。id 选择器以 ” ” 来定义。下面的两个 id 选择器,第一个可以定义元素的颜色为红色,第二个定义元素的颜色为绿色: red {color:re id 选择器 id 选择器可以为标有特…

    好文分享 2025年12月23日
    000
  • 有关css的绝对定位

    [导读] 定位(左边和顶部) css定位属性将是网虫们打开幸福之门的钥匙: h4 { position: absolute; left: 100px; top: 43px }这项css规则让浏览器将 的起始位置精 确地定在距离浏览器左边100象素,距离其 定位(左边和顶部) css定位属性将是网虫们…

    好文分享 2025年12月23日
    000
  • 响应式HTML5按钮适配不同屏幕方法【方法】

    实现响应式HTML5按钮需五种方法:一、CSS媒体查询按max-width断点调整样式;二、用rem/vw等相对单位替代px;三、Flexbox控制容器与按钮伸缩;四、CSS变量配合requestAnimationFrame优化的JS动态适配;五、Tailwind等框架的响应式工具类。 如果您希望H…

    2025年12月23日
    000
  • html5怎么加php_html5用Ajax与PHP后端交互实现数据传递【交互】

    HTML5不能直接运行PHP,需通过Ajax与PHP通信:前端用fetch发送请求,PHP接收处理并返回JSON,前端解析响应更新DOM;注意跨域、编码、CSRF防护和输入过滤。 HTML5 本身是前端标记语言,不能直接运行 PHP 代码,但可以通过 Ajax(异步 JavaScript)与 PHP…

    2025年12月23日
    300

发表回复

登录后才能评论
关注微信