In the previous article "What is the state mode in PHP? Learn it through examples"We introduced the state pattern in PHP. The following article will take you to understand the visitor pattern in PHP design patterns.
Visitors, just like when we visit other people’s homes, or when others come to our home to visit us. Each of us is like an entity, and everyone who visits greets us one by one. After all, our Chinese nation is a nation that pays great attention to etiquette and hospitality. Visitor is the most complex pattern among GoF's 23 design patterns, and it is also the pattern that is placed last in various design pattern textbooks. Regardless of the difficulty, let's first look at its definition and implementation.
Gof class diagram and explanation
GoF definition: Represents an operation that acts on each element in an object structure. It allows you to define new operations that act on each element without changing its class
GoF Class Diagram
Code implementation
interface Visitor { public function VisitConcreteElementA(ConcreteElementA $a); function VisitConcreteElementB(ConcreteElementB $b); } class ConcreteVisitor1 implements Visitor { public function VisitConcreteElementA(ConcreteElementA $a) { echo get_class($a) . "被" . get_class($this) . "访问", PHP_EOL; } public function VisitConcreteElementB(ConcreteElementB $b) { echo get_class($b) . "被" . get_class($this) . "访问", PHP_EOL; } } class ConcreteVisitor2 implements Visitor { public function VisitConcreteElementA(ConcreteElementA $a) { echo get_class($a) . "被" . get_class($this) . "访问", PHP_EOL; } public function VisitConcreteElementB(ConcreteElementB $b) { echo get_class($b) . "被" . get_class($this) . "访问", PHP_EOL; } }
Abstract visitor interface and two specific implementations. It can be regarded as a young couple visiting our home!
interface Element { public function Accept(Visitor $v); } class ConcreteElementA implements Element { public function Accept(Visitor $v) { $v->VisitConcreteElementA($this); } public function OperationA() { } } class ConcreteElementB implements Element { public function Accept(Visitor $v) { $v->VisitConcreteElementB($this); } public function OperationB() { } }
Element abstraction and implementation can also be regarded as entities to be accessed. Of course it’s me and my wife.
class ObjectStructure { private $elements = []; public function Attach(Element $element) { $this->elements[] = $element; } public function Detach(Element $element) { $position = 0; foreach ($this->elements as $e) { if ($e == $element) { unset($this->elements[$position]); break; } $position++; } } public function Accept(Visitor $visitor) { foreach ($this->elements as $e) { $e->Accept($visitor); } } }
This is an object structure used to save element entities and make access calls. Everyone met in the living room and exchanged greetings. This is just a living room.
$o = new ObjectStructure(); $o->Attach(new ConcreteElementA()); $o->Attach(new ConcreteElementB()); $v1 = new ConcreteVisitor1(); $v2 = new ConcreteVisitor2(); $o->Accept($v1); $o->Accept($v2);
The call of the client finally allowed everyone to meet formally and introduce each other and shake hands. One visit was completed happily.
- Let visitors call the specified element. It should be noted here that the behavior of visitors calling elements is generally fixed and rarely changes. That is, the two methods VisitConcreteElementA() and VisitConcreteElementB(). That is to say, the class that defines the object structure rarely changes, but when it is often necessary to define new operations on this structure, the visitor pattern is used.
- It is necessary to perform many different and unrelated operations on the objects in an object structure. When you want to avoid having these operations "pollute" the classes of these objects, the visitor pattern is suitable
- The visitor pattern is suitable for situations where the data structure does not change. Therefore, it is a mode that you cannot use normally, but you can only use this mode when needed. GoF: "Most of the time you don't need visitor mode, but when you do, you really need it." Because there are few situations where the data structure does not change
- Some advantages and disadvantages of the visitor pattern: easy to add new operations; concentrate related operations and separate irrelevant operations; it is difficult to add new ConcreteElement classes; through Access at class level; accumulate status; destroy encapsulation
Our company’s accounts only have two items (Element): income and expenditure, but different departments (Visitor) will give produce different content. For example, when I check, I only need to check the monthly or quarterly summary data. The financial director needs detailed income and expenditure records, and the accountant needs complete details when doing accounting. It can be seen that the operation of the company really requires a very wide range of knowledge, not only management capabilities, but also accounting knowledge is necessary to understand! !
Full code: https://github.com/zhangyue0503/designpatterns-php/blob/master/23.visitor/source/visitor.php
Example
The last example of the pattern is back to our message sending. The same applies to multiple service providers. As visitors, they need to use their own SMS sending and APP push interfaces. At this time, you can use the visitor mode to operate and realize all the operations of these visitors.
Visitor mode information sending
Full source code: https://github.com/zhangyue0503/designpatterns- php/blob/master/23.visitor/source/visitor-msg.php
<?php interface ServiceVisitor { public function SendMsg(SendMessage $s); function PushMsg(PushMessage $p); } class AliYun implements ServiceVisitor { public function SendMsg(SendMessage $s) { echo '阿里云发送短信!', PHP_EOL; } public function PushMsg(PushMessage $p) { echo '阿里云推送信息!', PHP_EOL; } } class JiGuang implements ServiceVisitor { public function SendMsg(SendMessage $s) { echo '极光发送短信!', PHP_EOL; } public function PushMsg(PushMessage $p) { echo '极光推送短信!', PHP_EOL; } } interface Message { public function Msg(ServiceVisitor $v); } class PushMessage implements Message { public function Msg(ServiceVisitor $v) { echo '推送脚本启动:'; $v->PushMsg($this); } } class SendMessage implements Message { public function Msg(ServiceVisitor $v) { echo '短信脚本启动:'; $v->SendMsg($this); } } class ObjectStructure { private $elements = []; public function Attach(Message $element) { $this->elements[] = $element; } public function Detach(Message $element) { $position = 0; foreach ($this->elements as $e) { if ($e == $element) { unset($this->elements[$position]); break; } $position++; } } public function Accept(ServiceVisitor $visitor) { foreach ($this->elements as $e) { $e->Msg($visitor); } } } $o = new ObjectStructure(); $o->Attach(new PushMessage()); $o->Attach(new SendMessage()); $v1 = new AliYun(); $v2 = new JiGuang(); $o->Accept($v1); $o->Accept($v2);
Description
- We assume sending SMS and sending push They are two unchanged behaviors, that is, their data structures are stable and unchanged
- This way we can easily add ServiceVisitor. When adding Baidu Cloud or other SMS providers, it will be very easy Just add visitors conveniently
- The visitor mode is more suitable for structures with stable data structures. For example, bills only have income and expenses, people’s gender is only male and female, etc.
Original address: https://juejin.cn/post/6844903993240453133
Author: Hardcore Project Manager
recommends learning: "PHP Video Tutorial"
The above is the detailed content of An in-depth analysis of the visitor pattern in PHP. For more information, please follow other related articles on the PHP Chinese website!

php把负数转为正整数的方法:1、使用abs()函数将负数转为正数,使用intval()函数对正数取整,转为正整数,语法“intval(abs($number))”;2、利用“~”位运算符将负数取反加一,语法“~$number + 1”。

实现方法:1、使用“sleep(延迟秒数)”语句,可延迟执行函数若干秒;2、使用“time_nanosleep(延迟秒数,延迟纳秒数)”语句,可延迟执行函数若干秒和纳秒;3、使用“time_sleep_until(time()+7)”语句。

php除以100保留两位小数的方法:1、利用“/”运算符进行除法运算,语法“数值 / 100”;2、使用“number_format(除法结果, 2)”或“sprintf("%.2f",除法结果)”语句进行四舍五入的处理值,并保留两位小数。

php字符串有下标。在PHP中,下标不仅可以应用于数组和对象,还可应用于字符串,利用字符串的下标和中括号“[]”可以访问指定索引位置的字符,并对该字符进行读写,语法“字符串名[下标值]”;字符串的下标值(索引值)只能是整数类型,起始值为0。

判断方法:1、使用“strtotime("年-月-日")”语句将给定的年月日转换为时间戳格式;2、用“date("z",时间戳)+1”语句计算指定时间戳是一年的第几天。date()返回的天数是从0开始计算的,因此真实天数需要在此基础上加1。

在php中,可以使用substr()函数来读取字符串后几个字符,只需要将该函数的第二个参数设置为负值,第三个参数省略即可;语法为“substr(字符串,-n)”,表示读取从字符串结尾处向前数第n个字符开始,直到字符串结尾的全部字符。

方法:1、用“str_replace(" ","其他字符",$str)”语句,可将nbsp符替换为其他字符;2、用“preg_replace("/(\s|\ \;||\xc2\xa0)/","其他字符",$str)”语句。

php判断有没有小数点的方法:1、使用“strpos(数字字符串,'.')”语法,如果返回小数点在字符串中第一次出现的位置,则有小数点;2、使用“strrpos(数字字符串,'.')”语句,如果返回小数点在字符串中最后一次出现的位置,则有。


Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

AI Hentai Generator
Generate AI Hentai for free.

Hot Article

Hot Tools

SublimeText3 Chinese version
Chinese version, very easy to use

Dreamweaver Mac version
Visual web development tools

WebStorm Mac version
Useful JavaScript development tools

Notepad++7.3.1
Easy-to-use and free code editor

SecLists
SecLists is the ultimate security tester's companion. It is a collection of various types of lists that are frequently used during security assessments, all in one place. SecLists helps make security testing more efficient and productive by conveniently providing all the lists a security tester might need. List types include usernames, passwords, URLs, fuzzing payloads, sensitive data patterns, web shells, and more. The tester can simply pull this repository onto a new test machine and he will have access to every type of list he needs.
