模板模式 一时失言乱红尘 2022-11-27 06:46 75阅读 0赞 # 模板模式 # > 模板模式准备一个抽象类,将部分逻辑以具体方法以及具体构造形式实现,然后声明一些抽象方法来迫使子类实现剩余的逻辑。不同的子类可以以不同的方式实现这些抽象方法,从而对剩余的逻辑有不同的实现。先制定一个顶级逻辑框架,而将逻辑的细节留给具体的子类去实现。 > 讲到模板模式,我不得不提的就是我们著名的[PSR-3][]规范,`PSR-3`为我们提供了模板的接口 <?php namespace Psr\Log; /** * Describes a logger instance. * * The message MUST be a string or object implementing __toString(). * * The message MAY contain placeholders in the form: {foo} where foo * will be replaced by the context data in key "foo". * * The context array can contain arbitrary data, the only assumption that * can be made by implementors is that if an Exception instance is given * to produce a stack trace, it MUST be in a key named "exception". * * See https://github.com/php-fig/fig-standards/blob/master/accepted/PSR-3-logger-interface.md * for the full interface specification. */ interface LoggerInterface { /** * System is unusable. * * @param string $message * @param array $context * @return void */ public function emergency($message, array $context = array()); /** * Action must be taken immediately. * * Example: Entire website down, database unavailable, etc. This should * trigger the SMS alerts and wake you up. * * @param string $message * @param array $context * @return void */ public function alert($message, array $context = array()); /** * Critical conditions. * * Example: Application component unavailable, unexpected exception. * * @param string $message * @param array $context * @return void */ public function critical($message, array $context = array()); /** * Runtime errors that do not require immediate action but should typically * be logged and monitored. * * @param string $message * @param array $context * @return void */ public function error($message, array $context = array()); /** * Exceptional occurrences that are not errors. * * Example: Use of deprecated APIs, poor use of an API, undesirable things * that are not necessarily wrong. * * @param string $message * @param array $context * @return void */ public function warning($message, array $context = array()); /** * Normal but significant events. * * @param string $message * @param array $context * @return void */ public function notice($message, array $context = array()); /** * Interesting events. * * Example: User logs in, SQL logs. * * @param string $message * @param array $context * @return void */ public function info($message, array $context = array()); /** * Detailed debug information. * * @param string $message * @param array $context * @return void */ public function debug($message, array $context = array()); /** * Logs with an arbitrary level. * * @param mixed $level * @param string $message * @param array $context * @return void */ public function log($level, $message, array $context = array()); } > 以上代码提供了记录各个级别日志的接口,下面我们使用模板模式做一个日志的实现。 <?php /** * Describes log levels. */ class LogLevel { const EMERGENCY = 'emergency'; const ALERT = 'alert'; const CRITICAL = 'critical'; const ERROR = 'error'; const WARNING = 'warning'; const NOTICE = 'notice'; const INFO = 'info'; const DEBUG = 'debug'; } /** * Class ILogger * @datetime 2020/7/17 7:16 PM * @author roach * @email jhq0113@163.com */ abstract class ILogger { /** * @param string $message * @param array $context * @return string * @datetime 2020/7/17 7:17 PM * @author roach * @email jhq0113@163.com */ public static function interpolate($message, array $context = array()) { // build a replacement array with braces around the context keys $replace = array(); foreach ($context as $key => $val) { // check that the value can be cast to string if (!is_array($val) && (!is_object($val) || method_exists($val, '__toString'))) { $replace['{' . $key . '}'] = $val; } } // interpolate replacement values into the message and return return strtr($message, $replace); } /** * System is unusable. * * @param string $message * @param array $context * @return void */ public function emergency($message, array $context = array()) { $this->log(LogLevel::EMERGENCY, $message, $context); } /** * Action must be taken immediately. * * Example: Entire website down, database unavailable, etc. This should * trigger the SMS alerts and wake you up. * * @param string $message * @param array $context * @return void */ public function alert($message, array $context = array()) { $this->log(LogLevel::ALERT, $message, $context); } /** * Critical conditions. * * Example: Application component unavailable, unexpected exception. * * @param string $message * @param array $context * @return void */ public function critical($message, array $context = array()) { $this->log(LogLevel::CRITICAL, $message, $context); } /** * Runtime errors that do not require immediate action but should typically * be logged and monitored. * * @param string $message * @param array $context * @return void */ public function error($message, array $context = array()) { $this->log(LogLevel::ERROR, $message, $context); } /** * Exceptional occurrences that are not errors. * * Example: Use of deprecated APIs, poor use of an API, undesirable things * that are not necessarily wrong. * * @param string $message * @param array $context * @return void */ public function warning($message, array $context = array()) { $this->log(LogLevel::WARNING, $message, $context); } /** * Normal but significant events. * * @param string $message * @param array $context * @return void */ public function notice($message, array $context = array()) { $this->log(LogLevel::NOTICE, $message, $context); } /** * Interesting events. * * Example: User logs in, SQL logs. * * @param string $message * @param array $context * @return void */ public function info($message, array $context = array()) { $this->log(LogLevel::INFO, $message, $context); } /** * Detailed debug information. * * @param string $message * @param array $context * @return void */ public function debug($message, array $context = array()) { $this->log(LogLevel::DEBUG, $message, $context); } /** * Logs with an arbitrary level. * * @param mixed $level * @param string $message * @param array $context * @return void */ abstract public function log($level, $message, array $context = array()); } /** * Class File * @datetime 2020/7/17 7:23 PM * @author roach * @email jhq0113@163.com */ class File extends ILogger { /** * @var string * @datetime 2020/7/17 7:20 PM * @author roach * @email jhq0113@163.com */ protected $_fileName; /** * @param $fileName * @datetime 2020/7/17 7:21 PM * @author roach * @email jhq0113@163.com */ public function setFileName($fileName) { $this->_fileName = $fileName; } /** * @param string $level * @param string $message * @param array $context * @datetime 2020/7/17 7:23 PM * @author roach * @email jhq0113@163.com */ public function log($level, $message, array $context = array()) { $message = '{datetime} {level} {ip} {url} '.$message.PHP_EOL; $context['datetime'] = date('Y-m-d H:i:s'); $context['level'] = $level; $context['ip'] = $_SERVER['REMOTE_ADDR']; $context['url'] = $_SERVER['REQUEST_URI']; $message = self::interpolate($message, $context); file_put_contents($this->_fileName, $message, FILE_APPEND | LOCK_EX); } } class Db extends ILogger { /** * @var \PDO * @datetime 2020/7/17 7:24 PM * @author roach * @email jhq0113@163.com */ protected $_pdo; /** * @param PDO $pdo * @datetime 2020/7/17 7:24 PM * @author roach * @email jhq0113@163.com */ public function setPdo(\PDO $pdo) { $this->_pdo = $pdo; } /** * @param mixed $level * @param string $message * @param array $context * @datetime 2020/7/17 7:28 PM * @author roach * @email jhq0113@163.com */ public function log($level, $message, array $context = array()) { $message = self::interpolate($message, $context); $stmt = $this->_pdo->prepare('INSERT INTO `roach_log`(`level`,`ip`,`url`,`msg`,`add_time`)VALUES(?,?,?,?,?)'); $stmt->execute([ $level, $_SERVER['REMOTE_ADDR'], $_SERVER['REQUEST_URI'], self::interpolate($message, $context), time() ]); } } > 分析以上代码,`ILogger`是抽象的模板类,实现了部分逻辑,但是`log`方法没有实现,交给子类实现,作者实现了`File`和`Db`子类,如果您感兴趣可以实现其他类,如`Redis`、`Kafka`等。 > 看过策略模式文章的同学也许有疑问了,这不是策略模式吗?把`log`方法封装了起来,从本质上来讲算是解决了不同算法的问题,但是严格上来讲,缺少了一个`Context`对象,不过不重要,只要我们的代码符合开闭原则,我们以设计模式六大原则来编写代码就可以了,严格按照模式来编写代码是没有意义的。 ### [学习更多内容: https://404.360tryst.com][https_404.360tryst.com] ### ### [我的视频课程: https://edu.csdn.net/course/detail/9933][https_edu.csdn.net_course_detail_9933] ### [PSR-3]: https://www.php-fig.org/psr/psr-3 [https_404.360tryst.com]: https://404.360tryst.com [https_edu.csdn.net_course_detail_9933]: https://edu.csdn.net/course/detail/9933
相关 模板模式 模板模式 > 模板模式准备一个抽象类,将部分逻辑以具体方法以及具体构造形式实现,然后声明一些抽象方法来迫使子类实现剩余的逻辑。不同的子类可以以不同的方式实现这些抽象方法, 一时失言乱红尘/ 2022年11月27日 06:46/ 0 赞/ 76 阅读
相关 模板模式 在模板模式(Template Pattern)中,一个抽象类公开定义了执行它的方法的方式/模板。它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进行。这种类型的设计 偏执的太偏执、/ 2022年11月10日 10:44/ 0 赞/ 122 阅读
相关 模板模式 模板模式 定义: 模板模式是行为型模式的一种。它的定义是一个抽象类公开定义了执行它的方法的方式/模板。它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进 小鱼儿/ 2022年07月13日 12:54/ 0 赞/ 120 阅读
相关 模板模式 介绍 对于某一个业务逻辑(算法实现)在不同的对象中有不同的细节实现,但是逻辑(算法) 的框架(或通用的应用算法)是相同的。`Template`提供了这种情况的一个实现框架 ╰+攻爆jí腚メ/ 2022年05月22日 04:11/ 0 赞/ 119 阅读
相关 模板模式 前言 模板方法模式(TemplateMethod):定义一个操作中的算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某 蔚落/ 2021年12月09日 00:45/ 0 赞/ 198 阅读
相关 模板模式 模板模式 在模板模式(Template Pattern)中,一个抽象类公开定义了执行它的方法的方式/模板。它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进行 谁践踏了优雅/ 2021年09月29日 16:06/ 0 赞/ 220 阅读
相关 模板模式 本质:子类重写父类方法,并且使用父类引用指向子类对象,调用此方法是子类的方法。 -------------------- 模板抽象类,定义一些模板方法 publ 浅浅的花香味﹌/ 2021年09月23日 13:48/ 0 赞/ 231 阅读
相关 模板模式 模板模式 定义 模板方法模式在一个方法中定义一个算法骨架,并将某些步骤推迟到子类中实现。模板方法模式可以让子类在不改变算法整体结构的情况下,重新定义算法中的某些步骤 向右看齐/ 2021年09月18日 07:26/ 0 赞/ 313 阅读
相关 模板模式 模板模式:定义一个操作中的算法的骨架,而将一些步骤延迟到子类中,模板模式可以使得子类可以不改变一个算法的结构就可以重新定义改算法的某些特定的步骤。 当我们要完成在某一细节层次 你的名字/ 2021年09月17日 01:10/ 0 赞/ 219 阅读
相关 模板模式 模板模式(Template Pattern)中,一个抽象类公开定义了执行它的方法的方式/模板。它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进行。这种类型的... 小灰灰/ 2020年06月13日 05:46/ 0 赞/ 631 阅读
还没有评论,来说两句吧...