资讯专栏INFORMATION COLUMN

PHP-PSR 现代PHPer的开发规范

zebrayoung / 2218人阅读

摘要:的使命是实现框架之间的互操作性。个人和官方都认为开发者应该遵循更为严格的代码标准,在现代的生态系统中,风格统一,可以更好的让其他开发者理解代码。记录的消息用于诊断检查和排除应用中的操作稳定性和性能方面的问题。

原文是在我自己博客中,小伙伴也可以点阅读原文进行跳转查看,还有好听的背景音乐噢~

    PSR是PHP Standards Recommendation的简称,意为PHP推荐标准。要想了解PSR,首先得知道制定这一标准的人/组织是谁————PHP-FIG。

PHP-FIG

PHP-FIG全称为PHP Framework Interop Group,是一个组织,这个组织的成员由一些PHP框架的代表组成,这些人聚在一起“讨论框架之间的共性,寻找可以合作的方式”。PHP-FIG制订了推荐规范,PHP框架可以自愿实现这些规范,改进其他框架的通信和共享功能。

PHP-FIG的使命是实现框架之间的互操作性。

PSR-1:基本的代码风格

    在文章的最开始,我们就已经简单介绍过什么是PSR,PSR是PHP标准,而PSR-1是PHP最基本也是最简单的标准。

PHP标签

必须把代码放在标签中。不得使用其它的标签句法

这点相信很多PHPer都很容易遵守,而且在现实撸代码中一般都是采用正常的标签

编码

所有PHP文件都必须使用UTF-8字符集编码,而且不能有字节顺序标记(Byte Order Mark,BOM)

这个也很常见,就是无BOM和有BOM格式,记得刚开始敲PHP代码的时候,前辈总是很关照,一定要用IDE调成无BOM格式啊,当时表示懵懂,然后就跟着做了,现在看到这里,又重新查了资料,找到了为什么不能使用有BOM格式的原因,BOM会产生多余的输出,就像平白无故多了一个空行:

php在处理BOM头的时候,有时候存在错误,可能造成你在使用 header 或 session_start 之类的函数时,出现 文件已经输出的错误,多数都是因为BOM头送出去了。。因为在php看来,成了一个空格。所以使用无BOM的格式

目的

一个PHP文件可以定义符号(类、性状、函数、常量等),或者执行有副作用的操作(生成结果或者处理数据),但不能同时做两件事

这个规定的意思差不多就是一个变量、方法或者一个类,只能相应完成一个操作、做一件事情,这样保证了代码的清晰易懂,也保证了方法、变量的单一性,各司其职。其实也是为了方便,我们在以后项目/应用较大时,可以很好的解耦

自动加载

PHP的命名空间和类必须遵守PSR-4自动加载器标准

后续看PSR-4的具体解释

类的名称

PHP类的名称必须使用驼峰式,又名标题式

驼峰式和分词式(每个单词用_隔开)这两种写法,记得以前存在很大的争议,有人支持驼峰(GirlFriend),有人支持分词式(girl_friend),现在好了,统一规定出来了,为了PHP更好的发展,那就委屈支持分词式的兄弟,统一驼峰了。记得公司的CI2项目,用的就是这种分词式,不过也是框架规定,后来在自己的项目中,自己有预感的使用了驼峰式,哈哈,为自己的眼光点赞~

常量的名称

PHP的常量名称必须大写;

这点应该是毋庸置疑的吧,最开始写PHP的时候,这个写法已经根深蒂固了。

方法的名称

使用驼峰式(boyFriend)

方法的命名和类的命名方式有些相似,不过还是有些区别:类的命名规定首字母大写(BoyFriendMoney),而方法的命名规定首字母小写(boyFriendMoney)

PSR-2:严格的代码风格

PSR-2 相较于PSR-1是更为严格的代码规范。个人和官方都认为开发者应该遵循更为严格的代码标准,在现代的PHP生态系统中,风格统一,可以更好的让其他开发者理解PHP代码。

贯彻PSR-1

使用PSR-2 之前先要贯彻PSR-1

缩进

使用四个空格缩进。

关于缩进这个问题,相信有很多争议。我在真正正视这个问题之前,一直使用的都是IDE的tab键。然后当同事和自己在编写同一文件的时候,就会出先代码缩进不一的情况,导致代码结构非常乱。所以在出现这个问题之后,就统一了一下文件缩进的标准,以四个空格为缩进。这样的话,就算是用不一致的编辑器打开,效果也是一样的。
很多IDE都可以设置tab键,百度一下就可以搜到。

文件和代码行

PHP文件必须使用UNIX风格的换行符(LF),最后要有一个空行,而且不能使用PHP关闭 ?> 标签。

最开始我也不懂为什么在纯PHP页面中不使用关闭 ?> 标签,后来在书中找到了答案

为了避免意料之外的输出错误,如果加上关闭标签,而且在关闭标签后有空行,那么这个空行也被当成输出,导致错误(例如,设定http首部时)

关键字

关键字,要使用小写;

之前不知道在哪儿看的PHP的教程,上面写的PHP代码像truefalse这样的关键字都使用的是大写TRUEFALSE,我也一直在这样使用,后来看到PSR-2的规范,才知道应该要使用小写,心累~

命名空间

每个命名空间语句后必须跟着一个空行。类似的,使用use关键字导入命名空间或为命名空间创建别名时,在一系列use声明语句后要加一个空行

类似于:



类定义体的起始括号应在类名之后另起一行写;
类定义体的结束括号必须在定义体之后新起一行写;

例:


方法

方法定义体的起始括号应在方法名之后另起一行写;
方法定义体的结束括号必须在方法定义体之后新起一行写;

请参考上面类示例中方法的例子。

可见性

1、类中的每个属性和方法都要声明可见性。可见性由public、protected或者private指定,其作用是决定在类的内部和外部如何访问属性的方法。
2、私有方法的名称前加上下划线
3、如果类属性声明为abstractfinal,这两个限定符必须放在可见性关键字之前
4、如果属性、方法声明为static,这个限定符必须放在可见性关键字之后

例子:

// 1、2
public $sex;
private $_sex;
protected $sex;

// 3、
abstract public $sex;
final public $sex;

// 4、
public static $sex;
public static $age;
控制结构

所有控制结构关键字后面都要有一个空格。控制结构关键字包括:ifelseifelseswitchcasewhiledo whileforforeachtrycatch。如果控制结构关键字后面有一对圆括号,起始圆括号后面不能有空格,结束圆括号之前不能有空格。与类和方法的定义体不同,控制结构关键字后面的起始括号应该和控制机构关键字写在同一行。控制结构关键字后面的结束括号必须写在多带带一行。

例:

/** 
* 错误的示例:
* 这里有4个错误:
* 1、if关键词后面和圆括号之前没有空格
* 2、圆括号前后有空格
* 3、后圆括号和起始括号之前没有空格
* 4、else关键词前后没有空格
**/
if( 1 == true ){
    // do something
}else{
    // do something
}

/** 
* 正确的示例:
**/
if (1 == true) {
    // do something
} else {
    // do something
}
PSR-3:日志记录器接口 日志记录器

PHP-FIG发布的第三个推荐规范和前两个不同,这个有点特殊,是一个接口。规定PHP日志记录器组件可以实现的方法。

日志记录器是对象,用于把不同重要程度的消息写入指定的输出。记录的消息用于诊断、检查和排除应用中的操作、稳定性和性能方面的问题。例如:开发的时候把调试信息写入到文本文件,把网站的流量统计信息记录到数据库等。

相信基本上所有的框架中都实现了日志功能,那么如果想要使用PSR-3规范的日志记录器,该怎么做呢?首先要满足两点:

日志功能委托给第三方库实现

最终用户能选择他们喜欢的日志记录器组件

编写PSR-3日志记录器

符合PSR-3推荐规范的PHP日志记录器组件,必须包含一个实现PsrLogLoggerInterface接口的PHP类。PSR-3接口复用了RFC 5424系统日志协议,规定要实现9个方法:

下面的代码是我从PHP-FIG的官网上拿过来的,想了解详细信息的可以 点我


这个类中的每一个方法都对应RFC 5424协议的一个日志级别,而且都接受两个参数。第一个参数必须是一个字符串,或者有一个__toString()方法的对象。第二个参数为数组,可选参数;

如果要编写符合PSR-3规范的日志记录器,那么就要创建一个实现PsrLogLoggerInterface接口的PHP类,而且要提供这个接口中每个方法的具体实现

使用PSR-3日志记录器

PSR-3规范出来之后,达到这种效果的组件太多了,这里就不介绍,如何实现这个接口的类了。现在有成熟的日志记录器组件,推荐monolog/monolog。这个组件完全上线了PSR-3的接口,而且可以使用自定义的消息格式化程序和处理程序扩展功能
如果monolog满足不了日常的使用,我们可以在此基础上拓展自己的方法,也非常简单;

使用monolog示例:

pushHandler(new StreamHandler("logs/development.log", Logger::DEBUG));
$logger->pushHandler(new StreamHandler("logs/production.log", Logger::WARNING));
// 使用日志记录器
$logger->debug("This is debug message");
$logger->debug("This is warning message");
PSR-4:自动加载器

PHP-FIG发布的第四个推荐规范描述了一个标准的自动加载器策略;自动加载器的意思就是指在程序运行时按需查找PHP类、接口(interface)或性状(trait)并将其载入加载器。

自动加载器策略

PSR-4推荐规范不要求改变代码的实现方式,只建议如何使用文件系统目录结构和PHP命名空间组织代码。PSR-4依赖PHP命名空间和文件系统目录结构查找并加载PHP类、性状和接口

为什么自动加载器很重要

举一个很常见的场景,我们引入文件通常都是采用requireinclude这样的方法,这样的方式简单也可靠,但是如果我们引入一两个还好说,但是当我们一个项目运行时需要引入几十个文件呢,那我们岂不是要写几十个require或者include?这样既不方便,又不美观,所以PHP-FIG在此基础上考虑,规范了一个统一的自动加载器策略;

如何使用自动加载器

建议使用依赖管理器Composer自动生成的PSR-4自动加载器。
现代的PHP框架,laravel、Yii、TP5等都使用了依赖Composer的自动加载器策略,方便我们下载组件和引入合适的类。

PSR-ME:制定自己的PHP规范

遵循PSR-1、PSR-2的使用规范

合适、精简的变量、方法、类命名。能让人看一眼就清楚是做什么的

尽量编写出高内聚、低耦合的代码

保持代码结构整洁、美观

总结

以上内容均来自《Mordern PHP》,中文名为现代PHP,书中详细的介绍了在现代化PHP开发中的新特性和良好实践,非常好的一本书。
PHP-FIG推出的PHP规范,并不一定说所有的PHP开发者必须遵守。制定这一规范的目的就是为了,在全世界的PHP开发者在查看代码的时候,能更加简单和轻松。造出来的组件/轮子可以很容易的就被所有开发者熟知和使用,同时也减少了我们的工作投入率,使产出大于投入,效率更高更快。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/25866.html

相关文章

  • PHPer也刷《剑指Offer》之链表

    摘要:剑指中链表相关题目俗话说光说不练假把式,既然学习了链表的基础概念和基本操作那我们一定要找些题目巩固下,下面来看剑指中的相关题目。题目分析合并两个排序的链表,需要分别比较两个链表的每个值,然后改变指针。 温故知新 链表由一个一个的作为节点的对象构成的,每一个节点都有指向下一个节点的指针,最后一个节点的指针域指向空。每个节点可以存储任何数据类型。 根据类型可以分为单链表、双链表、环形链表、...

    daydream 评论0 收藏0
  • tastphp,为现代 phper 准备 PHP 框架

    摘要:大家好,推荐下我们团队自己研发的框架为现代化的准备的。可拔插,扩展性强。借鉴了等优秀框架。有兴趣的可以关注下。最渴望有人给我们提交。中文文档基础已经写完,剩下努力写中。。。 大家好,推荐下我们团队自己研发的框架:tastphp 为现代化的phper准备的。可拔插,扩展性强。借鉴了Symfony、Laravel、Silex等优秀框架。 有兴趣的可以关注下 tastphp。最渴望有人给...

    meteor199 评论0 收藏0
  • 2017 —— 一个 PHPer 自白

    摘要:大会年,我去了。小会值得一提的是,今年月份,我参加了一个的分享会。出游沙巴这是部门组织的出游,获得了最佳团队,拿到了一笔经费,于是有了这次出游。于是,我的下个目的地是西藏。 转眼间 2017 年过去了。我已经不能说自己是去年的毕业生了,时光匆匆,感觉自己越来越老了。 这一年,我所经历的,让我收获很多,让我懂得很多,让我明白了很多。也许是明确了某一个目标,也许是其它的什么,我觉得,201...

    BenCHou 评论0 收藏0
  • PHPer书单

    摘要:想提升自己,还得多看书多看书多看书下面是我收集到的一些程序员应该看得书单及在线教程,自己也没有全部看完。共勉吧当然,如果你有好的书想分享给大家的或者觉得书单不合理,可以去通过进行提交。讲师温铭,软件基金会主席,最佳实践作者。 想提升自己,还得多看书!多看书!多看书!下面是我收集到的一些PHP程序员应该看得书单及在线教程,自己也没有全部看完。共勉吧!当然,如果你有好的书想分享给大家的或者...

    jimhs 评论0 收藏0
  • 2018 PHP 面试总结

    摘要:最近面试了不少公司,正好把记得的问题做个总结。抽象类的接口的区别,不在于编程实现,而在于程序设计模式的不同。一般来讲,抽象用于不同的事物,而接口用于事物的行为。 最近面试了不少公司,正好把记得的问题做个总结。 本文 github 会持续更新 公众号 搜索 苏生不惑 或者扫二维码关注,每周更新。 showImg(https://segmentfault.com/img/bVbsYyM?w...

    aboutU 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<