PHP自定义注解:增强代码灵活性和可扩展性
本文探讨如何在Symfony 3应用中创建和使用自定义注解。注解是我们在类、方法和属性上方看到的文档块元数据/配置,常用于声明控制器路由(@Route()
)、Doctrine ORM映射(@ORM()
)或控制对Rauth等包中各种类和方法的访问。本文将介绍如何自定义注解,并在不加载类的情况下读取类或方法信息。
关键要点:
- PHP自定义注解可用于向代码添加元数据,影响代码行为,使其更灵活、更易于适应。它们可用于定义路由信息、指定验证规则或配置依赖注入。
- 创建PHP自定义注解需要定义一个新的类来表示该注解。此类应具有与您希望在注解中设置的值相对应的属性。
- Doctrine支持自定义注解。您可以定义自己的注解并将其用于Doctrine实体中。Doctrine的注解读取器将解析这些注解,然后可以使用它们来影响实体的行为。
- 使用PHP自定义注解的一个限制是它们不受语言本身的原生支持。这意味着您需要使用提供注解支持的库或工具,例如Doctrine或PHP-DI。
- PHP自定义注解的用途多种多样,包括在Web应用程序中定义路由信息、为表单输入指定验证规则或配置依赖注入。它们还可用于向代码添加文档。
免责声明: 注解与Symfony无关,它是作为Doctrine项目的一部分开发的概念,用于解决将ORM信息映射到类方法的问题。
本文将构建一个名为WorkerBundle的小型可重用捆绑包(仅用于演示目的,并非真正可打包的)。我们将开发一个小型概念,允许定义各种以不同速度“运行”的Worker类型,然后应用程序中的任何人都可以使用它们。实际的worker操作不在本文的讨论范围之内,因为我们关注的是设置管理它们的系统(并通过注解发现它们)。
您可以查看这个仓库并按照其中介绍的说明在本地Symfony应用程序中设置捆绑包。
Worker
Worker将实现一个接口,该接口需要一个方法:::work()
。在新的WorkerBundle中,让我们创建一个Workers/
目录来保持整洁,并在其中添加接口:
<?php namespace WorkerBundle\Workers; interface WorkerInterface { /** * 执行工作 * * @return NULL */ public function work(); }
注解
每个worker都必须实现上述接口。很清楚。但除此之外,我们还需要它们在类上方有一个注解,以便找到它们并读取有关它们的某些元数据。
Doctrine将文档块注解映射到一个类,该类的属性表示注解本身内部的键。让我们创建我们自己的注解并实际操作一下。
每个WorkerInterface实例在其文档块中都将具有以下注解:
<?php namespace WorkerBundle\Workers; interface WorkerInterface { /** * 执行工作 * * @return NULL */ public function work(); }
我们将保持简单,只有两个属性:唯一名称(字符串)和worker速度(整数)。为了使Doctrine的注解库能够识别此注解,我们必须创建一个匹配的类,不出所料,它自己也有一些注解。
我们将此类放在捆绑包命名空间的Annotation文件夹中,并将其简单地命名为Worker:
/** * @Worker( * name = "唯一的Worker名称", * speed = 10 * ) */
如您所见,我们有两个属性和一些简单的getter方法。更重要的是,我们在顶部有两个注解:@Annotation
(告诉Doctrine此类表示一个注解)和@Target("CLASS")
(告诉它应该在整个类而不是方法或属性上方使用)。就是这样,WorkerInterface类现在可以通过确保在文件顶部使用use
语句导入相应的类来使用此注解:
<?php namespace WorkerBundle\Annotation; use Doctrine\Common\Annotations\Annotation; /** * @Annotation * @Target("CLASS") */ class Worker { /** * @Required * * @var string */ public $name; /** * @Required * * @var int */ public $speed; /** * @return string */ public function getName() { return $this->name; } /** * @return int */ public function getSpeed() { return $this->speed; } }
管理器
接下来,我们需要一个管理器,我们的应用程序可以使用它来获取所有可用worker的列表并创建它们。在与WorkerInterface相同的命名空间中,我们可以使用这个简单的管理器类:
use WorkerBundle\Annotation\Worker;
发现器
我们注解演示的关键部分实际上是发现过程的一部分。为什么?因为我们使用Worker注解来确定是否应将相应的类视为Worker。在此过程中,我们在实际实例化对象之前使用了元数据。让我们看看我们的WorkerDiscovery类:
<?php namespace WorkerBundle\Workers; class WorkerManager { // ... (代码与原文相同) ... }
连接起来
现在我们有了主要组件,是时候将所有内容连接起来了。首先,我们需要服务定义,因此在捆绑包的Resource/config文件夹中,我们可以有这个services.yml文件:
<?php namespace WorkerBundle\Workers; // ... (代码与原文相同) ... }
为了使我们的服务定义能够被容器集中拾取,我们需要编写一个小的扩展类。因此,在捆绑包的DependencyInjection文件夹中,创建一个名为WorkerExtension的类。位置和名称对于Symfony自动拾取它都很重要。
services: worker_manager: class: WorkerBundle\Workers\WorkerManager arguments: ["@worker_discovery"] worker_discovery: class: WorkerBundle\Workers\WorkerDiscovery arguments: ["%worker_namespace%", "%worker_directory%", "%kernel.root_dir%", "@annotation_reader"]
最后,我们需要注册我们的捆绑包。在我们的AppKernel中:
<?php namespace WorkerBundle\DependencyInjection; // ... (代码与原文相同) ... }
就是这样。
让我们工作!
现在我们准备工作了。让我们在中央parameters.yml文件中配置我们的worker将在哪里找到:
public function registerBundles() { return array( // ... new WorkerBundle\WorkerBundle(), // ... ); }
这些参数从容器传递到WorkerDiscovery类,如上所述。
位置由您决定,但现在让我们将我们的worker放在应用程序的主要AppBundle捆绑包中。我们的第一个worker可能如下所示:
<?php namespace WorkerBundle\Workers; interface WorkerInterface { /** * 执行工作 * * @return NULL */ public function work(); }
我们的注解在那里,use
语句到位,因此没有任何东西可以阻止某些业务逻辑找到它并实例化它。让我们假设在一个Controller方法内部:
/** * @Worker( * name = "唯一的Worker名称", * speed = 10 * ) */
或者
<?php namespace WorkerBundle\Annotation; use Doctrine\Common\Annotations\Annotation; /** * @Annotation * @Target("CLASS") */ class Worker { /** * @Required * * @var string */ public $name; /** * @Required * * @var int */ public $speed; /** * @return string */ public function getName() { return $this->name; } /** * @return int */ public function getSpeed() { return $this->speed; } }
结论
我们现在可以使用注解来表达有关类(或方法和属性)的元数据。在本教程中,我们构建了一个小型包,它使应用程序(或其他外部捆绑包)能够通过定义有关它们的某些元数据来声明能够执行某些工作的worker。此元数据不仅使它们易于发现,而且还提供有关是否应该实际使用它们的信息。
您在自己的项目中使用自定义注解吗?如果是这样,您是如何实现它们的?也许还有第三种方法?让我们知道!
(感兴趣了解更多关于Symfony、Doctrine、注解以及各种企业级PHP内容的信息?加入我们,参加为期三天的动手研讨会,WebSummerCamp——唯一一个完全手动的会议,并且还会照顾您想带去的任何人!)
(原文的FAQ部分已省略,因为其内容与已翻译和改写的内容高度重复。)
以上是您自己的自定义注释 - 不仅仅是评论!的详细内容。更多信息请关注PHP中文网其他相关文章!

使用数据库存储会话的主要优势包括持久性、可扩展性和安全性。1.持久性:即使服务器重启,会话数据也能保持不变。2.可扩展性:适用于分布式系统,确保会话数据在多服务器间同步。3.安全性:数据库提供加密存储,保护敏感信息。

在PHP中实现自定义会话处理可以通过实现SessionHandlerInterface接口来完成。具体步骤包括:1)创建实现SessionHandlerInterface的类,如CustomSessionHandler;2)重写接口中的方法(如open,close,read,write,destroy,gc)来定义会话数据的生命周期和存储方式;3)在PHP脚本中注册自定义会话处理器并启动会话。这样可以将数据存储在MySQL、Redis等介质中,提升性能、安全性和可扩展性。

SessionID是网络应用程序中用来跟踪用户会话状态的机制。1.它是一个随机生成的字符串,用于在用户与服务器之间的多次交互中保持用户的身份信息。2.服务器生成并通过cookie或URL参数发送给客户端,帮助在用户的多次请求中识别和关联这些请求。3.生成通常使用随机算法保证唯一性和不可预测性。4.在实际开发中,可以使用内存数据库如Redis来存储session数据,提升性能和安全性。

在无状态环境如API中管理会话可以通过使用JWT或cookies来实现。1.JWT适合无状态和可扩展性,但大数据时体积大。2.Cookies更传统且易实现,但需谨慎配置以确保安全性。

要保护应用免受与会话相关的XSS攻击,需采取以下措施:1.设置HttpOnly和Secure标志保护会话cookie。2.对所有用户输入进行输出编码。3.实施内容安全策略(CSP)限制脚本来源。通过这些策略,可以有效防护会话相关的XSS攻击,确保用户数据安全。

优化PHP会话性能的方法包括:1.延迟会话启动,2.使用数据库存储会话,3.压缩会话数据,4.管理会话生命周期,5.实现会话共享。这些策略能显着提升应用在高并发环境下的效率。

thesession.gc_maxlifetimesettinginphpdeterminesthelifespanofsessiondata,setInSeconds.1)它'sconfiguredinphp.iniorviaini_set().2)abalanceIsiseededeedeedeedeedeedeedto to to avoidperformance andununununununexpectedLogOgouts.3)

在PHP中,可以使用session_name()函数配置会话名称。具体步骤如下:1.使用session_name()函数设置会话名称,例如session_name("my_session")。2.在设置会话名称后,调用session_start()启动会话。配置会话名称可以避免多应用间的会话数据冲突,并增强安全性,但需注意会话名称的唯一性、安全性、长度和设置时机。


热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

Video Face Swap
使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

热工具

螳螂BT
Mantis是一个易于部署的基于Web的缺陷跟踪工具,用于帮助产品缺陷跟踪。它需要PHP、MySQL和一个Web服务器。请查看我们的演示和托管服务。

SecLists
SecLists是最终安全测试人员的伙伴。它是一个包含各种类型列表的集合,这些列表在安全评估过程中经常使用,都在一个地方。SecLists通过方便地提供安全测试人员可能需要的所有列表,帮助提高安全测试的效率和生产力。列表类型包括用户名、密码、URL、模糊测试有效载荷、敏感数据模式、Web shell等等。测试人员只需将此存储库拉到新的测试机上,他就可以访问到所需的每种类型的列表。

mPDF
mPDF是一个PHP库,可以从UTF-8编码的HTML生成PDF文件。原作者Ian Back编写mPDF以从他的网站上“即时”输出PDF文件,并处理不同的语言。与原始脚本如HTML2FPDF相比,它的速度较慢,并且在使用Unicode字体时生成的文件较大,但支持CSS样式等,并进行了大量增强。支持几乎所有语言,包括RTL(阿拉伯语和希伯来语)和CJK(中日韩)。支持嵌套的块级元素(如P、DIV),

SublimeText3汉化版
中文版,非常好用

MinGW - 适用于 Windows 的极简 GNU
这个项目正在迁移到osdn.net/projects/mingw的过程中,你可以继续在那里关注我们。MinGW:GNU编译器集合(GCC)的本地Windows移植版本,可自由分发的导入库和用于构建本地Windows应用程序的头文件;包括对MSVC运行时的扩展,以支持C99功能。MinGW的所有软件都可以在64位Windows平台上运行。