目录

PHP 编码标准插件

无规矩不成方圆,无规范不能协作。每门编程语言几乎都有一套合乎自有章法的规矩,当然了,PHP 也不例外。即使是你独立撸码,也自有一套自己的章法,可惜,你不是一个人在战斗,在这个讲究编程效率的兵团作战时代,没有人愿意忍受别人糟糕的代码风格。是的,代码的的确确是机器去执行的,但是因更改需求而维护的代码可不是机器人干的。为了让代码可维护,看起来像那么回事,还是尽力码好吧,不然,下一个遭吐槽可能就是你了,我已经不至一次听过『这特么谁写的』这句话了,反正听起来不像是在夸谁。

PHP CodeSniffer

CodeSniffer 安装

PHP CodeSniffer 有两个脚本,phpcs 脚本用来检测违反定义的编码标准,phpcbf 脚本用来自动纠正编码标准违规。

Curl 安装 CodeSniffer

1
2
3
4
5
6
7
8
// 下载 phpcs 和 phpcbf 文件并移动到相应目录
$ cd ~
$ curl -OL https://squizlabs.github.io/PHP_CodeSniffer/phpcs.phar
$ sudo mv phpcs.phar /usr/local/bin/phpcs
$ sudo chmod +x /usr/local/bin/phpcs
$ curl -OL https://squizlabs.github.io/PHP_CodeSniffer/phpcbf.phar
$ sudo mv phpcbf.phar /usr/local/bin/phpcbf
$ sudo chmod +x /usr/local/bin/phpcbf

Composer 安装 CodeSniffer

安装完成后,将 $HOME/.composer/vendor/bin 添加到系统 PATH 中。

  • 全局安装
1
2
// 安装 squizlabs/php_codesniffer
$ composer global require "squizlabs/php_codesniffer=*"
  • 项目安装
1
2
3
4
5
6
7
$ vim composer.json
{
    "require-dev": {
        "squizlabs/php_codesniffer": "3.*"
    }
}
$ composer install

工具设置

设置编码标准

支持的编码标准:PEAR, Zend, PSR2, MySource, Squiz, PSR1,默认:PEAR

1
2
// 设置编码标准为 PSR2
$ phpcs --config-set default_standard PSR2

设置报告格式

支持的报告格式:full, xml, checkstyle, csv, json, junit, emacs, source, summary, diff, svnblame, gitblame, hgblame, notifysend,默认:full

1
2
// 设置报告格式为 full
$ phpcs --config-set report_format full

设置警告隐藏

1
2
// 默认值为 1
$ phpcs --config-set show_warnings 0

设置运行进度

1
2
// 默认值为 0
$ phpcs --config-set show_progress 1

设置输出时使用润色

1
2
// 默认值为 0
$ phpcs --config-set colors 1

设置报告的宽度

1
2
// 设置报告的宽度为 120
$ phpcs --config-set report_width 120

设置文本字符编码

1
2
// 设置文本字符编码为 UTF-8
$ phpcs --config-set encoding UTF-8

设置制表符宽度

默认情况下,PHP_CodeSniffer 不会将检查文件中的制表符转换为空格,指定一个制表符宽度将使 PHP_CodeSniffer 用空格替换制表符。也可以通过设置 tab_width 配置选项来强制 PHP_CodeSniffer 将制表符替换为空格。

1
2
// 设置制表符宽度为 4 个空格
$ phpcs --config-set tab_width 4

设置 PHP 版本

1
2
// 使用 PHP_VERSION_ID 获取后再设置
$ phpcs --config-set php_version 70114

设置 PHP 路径

1
2
// 使用 which 命令获取后再设置
$ phpcs --config-set php_path /usr/local/bin/php

查看设置

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
// 查看刚才的一些设置
$ phpcs --config-show
Using config file: /Users/user/.composer/vendor/squizlabs/php_codesniffer/CodeSniffer.conf

colors:           1
default_standard: PSR2
encoding:         UTF-8
php_path:         /usr/local/bin/php
php_version:      70114
report_format:    full
report_width:     120
show_progress:    1
show_warnings:    1
tab_width:        4

工具使用

指定编码标准

也可以自定义编码标准,然后加入到检查标准中。

1
2
3
4
5
$ phpcs --standard=PEAR /path/to/code/Demo.php
$ phpcs --standard=/path/to/MyStandard /path/to/code/Demo.php
$ phpcs --standard=PEAR,Squiz,/path/to/MyStandard /path/to/code/Demo.php

打印安装编码标准的列表

1
2
$ phpcs -i
The installed coding standards are PEAR, Zend, PSR2, MySource, Squiz and PSR1

编码标准内的嗅探列表

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
$ phpcs --standard=PSR2 -e

The PSR2 standard contains 42 sniffs

Generic (12 sniffs)
-------------------
  Generic.ControlStructures.InlineControlStructure
  Generic.Files.ByteOrderMark
  Generic.Files.LineEndings
  Generic.Files.LineLength
  Generic.Formatting.DisallowMultipleStatements
  Generic.Functions.FunctionCallArgumentSpacing
  Generic.NamingConventions.UpperCaseConstantName
  Generic.PHP.DisallowShortOpenTag
  Generic.PHP.LowerCaseConstant
  Generic.PHP.LowerCaseKeyword
  Generic.WhiteSpace.DisallowTabIndent
  Generic.WhiteSpace.ScopeIndent

PEAR (1 sniff)
---------------
  PEAR.Functions.ValidDefaultValue

PSR1 (3 sniffs)
---------------
  PSR1.Classes.ClassDeclaration
  PSR1.Files.SideEffects
  PSR1.Methods.CamelCapsMethodName

PSR2 (12 sniffs)
----------------
  PSR2.Classes.ClassDeclaration
  PSR2.Classes.PropertyDeclaration
  PSR2.ControlStructures.ControlStructureSpacing
  PSR2.ControlStructures.ElseIfDeclaration
  PSR2.ControlStructures.SwitchDeclaration
  PSR2.Files.ClosingTag
  PSR2.Files.EndFileNewline
  PSR2.Methods.FunctionCallSignature
  PSR2.Methods.FunctionClosingBrace
  PSR2.Methods.MethodDeclaration
  PSR2.Namespaces.NamespaceDeclaration
  PSR2.Namespaces.UseDeclaration

Squiz (14 sniffs)
-----------------
  Squiz.Classes.ValidClassName
  Squiz.ControlStructures.ControlSignature
  Squiz.ControlStructures.ForEachLoopDeclaration
  Squiz.ControlStructures.ForLoopDeclaration
  Squiz.ControlStructures.LowercaseDeclaration
  Squiz.Functions.FunctionDeclaration
  Squiz.Functions.FunctionDeclarationArgumentSpacing
  Squiz.Functions.LowercaseFunctionKeywords
  Squiz.Functions.MultiLineFunctionDeclaration
  Squiz.Scope.MethodScope
  Squiz.WhiteSpace.ControlStructureSpacing
  Squiz.WhiteSpace.ScopeClosingBrace
  Squiz.WhiteSpace.ScopeKeywordSpacing
  Squiz.WhiteSpace.SuperfluousWhitespace

使用 CodeSniffer 格式化代码

请自行创建工程去测试。

CodeSniffer 格式化前的不规范代码

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
<?php

namespace Nilnice\Acme;
class Finder{
    public const IGNORE_VCS_FILES =1;
public const IGNORE_DOT_FILES=2;
    private $ignore = 0;
public function __construct(){
    $this-> ignore=static::IGNORE_VCS_FILES|static::IGNORE_DOT_FILES ;
}
public function getIterator() {
        if (0===count($this->dirs)&&0===count($this->iterators)){
            throw new \LogicException('You must call one of in() or append() methods before iterating over a Finder.');
        }
        if(1===count( $this->dirs ) &&0 ===count($this->iterators)){
            return   $this ->searchInDirectory( $this->dirs[0]);
        }
        $iterator=new \AppendIterator ();
        foreach ($this-> dirs        as $dir)
        {
            $iterator->append($this->searchInDirectory( $dir )) ;
        }
        foreach (  $this->iterators as $it) {
            $iterator->append(    $it);
        }
        return$iterator;
    }}

经过 phpcs 检查后的报告

https://inotes.oss-cn-beijing.aliyuncs.com/php/201812/php-codesniffer-phpcs.png

怎么样,这个检查结果明码标价,童叟无欺矣!

使用 phpcbf 修理代码

1
2
// 指定文件修理
$ phpcbf ./src/Test.php

修理后的代码样式

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
<?php

namespace Nilnice\Acme;
class Finder
{
    public const IGNORE_VCS_FILES = 1;
    public const IGNORE_DOT_FILES = 2;
    private $ignore = 0;


    public function __construct()
    {
        $this-> ignore = static::IGNORE_VCS_FILES | static::IGNORE_DOT_FILES ;

    }//end __construct()


    public function getIterator()
    {
        if (0 === count($this->dirs)&&0 === count($this->iterators)) {
            throw new \LogicException('You must call one of in() or append() methods before iterating over a Finder.');
        }

        if (1 === count($this->dirs) &&0 === count($this->iterators)) {
            return   $this ->searchInDirectory($this->dirs[0]);
        }

        $iterator = new \AppendIterator();
        foreach ($this-> dirs        as $dir) {
            $iterator->append($this->searchInDirectory($dir));
        }

        foreach ($this->iterators as $it) {
            $iterator->append($it);
        }

        return$iterator;

    }//end getIterator()


}//end class
// 此处多一空行

PHP Coding Standards Fixer

Fixer 安装

Curl 或 Wget 方式安装

1
2
3
4
5
6
7
$ wget http://cs.sensiolabs.org/download/php-cs-fixer-v2.phar -O php-cs-fixer
$ wget https://github.com/FriendsOfPHP/PHP-CS-Fixer/releases/download/v2.10.2/php-cs-fixer.phar -O php-cs-fixer
$ curl -L http://cs.sensiolabs.org/download/php-cs-fixer-v2.phar -o php-cs-fixer
$ sudo chmod +x php-cs-fixer
$ sudo mv php-cs-fixer /usr/local/bin/php-cs-fixer

Composer 安装 Fixer

1
2
3
4
$ composer global require friendsofphp/php-cs-fixer

// 如果之前设置过,就不需要
$ export PATH="$PATH:$HOME/.composer/vendor/bin"

Homebrew 安装 Fixer

1
2
// 先搜索 fixer 以确定名称后再安装
$ brew install homebrew/php/php-cs-fixer

使用 Fixer 格式化代码

使用 Fixer 格式化前的不规范代码

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
<?php
declare(strict_types=1);
namespace A ;
use SebastianBergmann\CodeCoverage\InvalidArgumentException ;
abstract class Foo extends FooBaseClass implements
    Bar1 ,
    Bar2 ,
    Bar3{
    protected const FIRST='first';
    protected const SECOND=0;
    protected const Z = -1;
    public const E_FOO = 200;
    public const E_BAR = 400;
    private $isNull = null ;
    protected $hello,     $world;
    public $numbers=array('one','two','three','four','five','six');
    public $v = 0;
    public $path = 'root';
    public function bar(
        $v, $w = 'a'
    )
    {
        $y = $w;
        $result = foo(
            'arg1',
            'arg2',
            10
        );
        switch ($v) {
            case 0:
                return 1;
            case 1      :
                echo '1';

                break;
            case  2:
                break;
            default:
                $result = 10;
        }
        return$result;
    }

    static public function fOne(
        $argA,
        $argB,
        $argC,
        $argD,
        $argE,
        $argF,
        $argG,
        $argH){
        $x = $argA + $argB + $argC
            + $argD + $argE + $argF
            + $argG + $argH;
        list($field1, $field2,
            $field3, $filed4,
            $field5, $field6) = explode(',', $x);
        fTwo(
            $argA,
            $argB,
            $argC,
            fThree(
                $argD,
                    $argE,
                    $argF,
                    $argG,
                $argH
            )
        );
        $z = $argA === 'Some string'
            ? 'yes' : 'no';
        $colors = [
            'red',
            'green',
              'blue',
            'black'  ,
            'white',
            'gray'  ,
        ];
        $count=count($colors);
        for ($i = 0; $i < $count; $i++) {
            $colorString
                = $colors[$i];
        }
    }

    public function fTwo(
        $strA,
        $strB,
        $strC,
        $strD
    ){
        if ($strA === 'one'
        || $strB === 'two'
        || $strC === 'three') {
            return $strA + $strB
            + $strC;
        }
        if ($strA==='hello'){$strB = 'world';}
        if($strB==='second') {
        $strA=null;
            throw new \InvalidArgumentException('Invalid argument.');
        }
        $x = $foo->one('a', 'b')
            ->two('c', 'd',             'e')
                ->three('fg')->  four();
        $y = a()    -> b()   ->c            ();
    return $strD;
    }

    public function fThree(
        $strA,
        $strB,
        $strC,
        $strD,
        $strE
    ) {
        try {

        } catch (Exception $e) {
        foo();
        } finally {
            // do something
        }
        return $strA + $strB + $strC
            + $strD + $strE;}
    abstract protected function fFour();
}

final class Bar{
    public final function foo($x,$z){
        global $k, $s1;
        $obj->foo()->bar();
        $arr =array(0 => 'zero', 1 => 'one');
        call_func(function () {
            return 0;
        });
        for ($i = 0; $i < $x; $i++) {
            $y+= ($y^0x123) << 2;
        }
        $k = $x > 15 ? 1 : 2;
        $k = $x ?: 0;
        $k = $x ?? $z;
        $k = $x<=>$z;
        do
        {
            try
            {
                if (!0 > $x && ! $x < 10){
                    while ($x !== $y){
                        $x = f($x * 3 + 5);
                    }
                    $z += 2;
                } elseif ($x > 20) {
                    $z = $x <<1;
                } else {
                    $z = $x|2;
                }
                $j = intval($z);
                switch ($j) {
                    case 0:
                        $s1=strval('100');
                        break;
                    case 2:
                        $s1 = 'two';
                        break;
                    default:
                        $s1 = 'other';
                }
            } catch (\Exception $e) {
                $t = $one[0];
                $u = $one['str'];
                $v = $one[$x[1]];
                echo $val{foo . $num}[$cell{$a}];
            }finally{// do something
            }
        }while ($x <0);
    }
}

function foo() {
    return 0;
}

function bar(
    $x,
    $y,
    int $z = 1
){
    $x = 0;
    // $x = 1
    do{$y += 1;} while ($y < 10);
    if(true){$x = 10;}elseif ($y < 10) {$x=5;}elseif (true) {$x=     5  ;}
    for ($i = 0; $i < 10; $i++){$yy=$x>2?1:2;}
    while( true ) {$x=0;}
    do{$x += 1;}while( true ) ;
    foreach ([
        'a'   => 0,
        'b'  => 1,
        'c'=>2,
    ] as $e1) {echo$e1;}
    $count = 10;
    $x = [
        'x','y',
        [
            1  => 'abc',
            2 =>            'def',
            3    => 'ghi'
        ],
    ];
    $zz = [
        0.1,
        0.2,
        0.3,
        0.4,
    ];
    $x = [
        0 => 'zero',
        123 => 'one two three',
        25 => 'two five'
    ];
    bar(0, bar(
        1,
        'b'
    ));
}

function foobar( $a ):Bar
{
    $a=null ===$a?$a:new $a;
    include('a.php');
}

function emptyreturn()
{
    return;
}

$map = array_map (function( $value )
{
    return$value  +   10;
}, $array   );
?>
<div><?= foo() ?></div>

自定义代码修复格式

创建 .php_cs 文件,并写入以下内容。

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
<?php

require __DIR__ . '/vendor/autoload.php';

use PhpCsFixer\Config;
use PhpCsFixer\Finder;

$finder = Finder::create()->in(__DIR__ . '/src');

return Config::create()
    ->setRiskyAllowed(true)
    ->setCacheFile(__DIR__ . '/.php_cs.cache')
    ->setRules([
        '@PSR2' => true, // 使用 PSR2 标准
        'array_syntax' => [
            'syntax' => 'short', // 数组使用短语法
        ],
        'binary_operator_spaces' => [
            'align_double_arrow' => false, // 不对齐双箭头操作符
            'align_equals' => false, // 不对齐赋值操作符
        ],
        'blank_line_after_namespace' => true, // 命名空间之后有一个空行
        'blank_line_after_opening_tag' => true, // PHP 打开标记之后有一个空行
        'blank_line_before_return' => true, // return 语句之前有一个空行
        'blank_line_before_statement' => [
            'statements' => [
                'break',
                'continue',
                'declare',
                'return',
                'throw',
                'try',
            ], // 这些声明之前有一个空行
        ],
        'class_attributes_separation' => [
            'elements' => [
                'const',
                'method',
                'property',
            ], // 类的这些元素分开,也就是元素之间加上空行
        ],
        'dir_constant' => true, // 将 dirname(__FILE__) 替换成 __DIR__
        'include' => true, // 将 include('a.php') 替换成 include 'a.php'
        'is_null' => true, // 将 is_null($a) 替换成 null === $a
        'linebreak_after_opening_tag' => true,
        'list_syntax' => [
            'syntax' => 'long', // list 使用 long 语法
        ],
        'lowercase_constants' => true, // 常量 true, false, null 使用小写
        'lowercase_keywords' => true, // PHP 关键字使用小写
        'method_chaining_indentation' => true, // 方法链式调用不需要缩进
        'modernize_types_casting' => true, // 将 *val 函数做对应类型的强制转换
        'new_with_braces' => true, // 实例化类时带上括号
        'no_blank_lines_after_class_opening' => true, // 类左大括号后没有空行
        'no_blank_lines_before_namespace' => false, // 命名空间之前没有穿行
        'no_closing_tag' => true, // 纯 PHP 文件不需要闭合标记
        'no_multiline_whitespace_around_double_arrow' => true, // 箭头操作符前后没有多余的空格
        'multiline_whitespace_before_semicolons' => true, // 移除结束分号之前的多余空行
        'no_null_property_initialization' => true, // 移除属性用 null 初始化是的显式指定
        'no_php4_constructor' => true, // 移除 PHP4 风格的构造方法
        'no_short_bool_cast' => true, // 采用双感叹号表示布尔情况的不应该使用,会转换为 (bool)
        'no_short_echo_tag' => false, // 替换短标记输出为长标记
        'no_singleline_whitespace_before_semicolons' => true, // 移除分号之前的多余空格
        'no_spaces_after_function_name' => true, // 在进行方法或函数调用时,移除方法或函数名称与左括号之间的空格
        'no_spaces_around_offset' => true, // 移除偏移大括号的空格
        'no_spaces_inside_parenthesis' => true, // 移除左括号后的和右括号前的多余空格
        'no_superfluous_elseif' => true, // 用 if 替换多余的 elseif
        'no_trailing_whitespace' => true, // 移除末行的尾随空格
        'no_trailing_whitespace_in_comment' => true, // 移除注释末行的尾随空格
        'no_unneeded_control_parentheses' => [
            'statements' => [
                'break',
                'clone',
                'continue',
                'echo_print',
                'return',
                'switch_case',
                'yield',
            ], // 移除控制语句周围不需要的括号
        ],
        'no_useless_return' => true, // 将 return; 替换成空
        'no_whitespace_in_blank_line' => true, // 移除空白行末尾的空格
        'not_operator_with_space' => false, // 逻辑非操作符前后有一个空格
        'not_operator_with_successor_space' => true, // 逻辑非操作符尾随一个空格
        'object_operator_without_whitespace' => true, // 移除对象操作符前后的空格
        'short_scalar_cast' => true, // 标量使用缩写
        'single_blank_line_at_eof' => true, // 纯 PHP 文件总是以一个空行换行符结束
        'single_blank_line_before_namespace' => true, // 命名空间之前有一个空行
        'single_class_element_per_statement' => [
            'elements' => [
                'const',
                'property',
            ], // 类元素的每个声明有自己的关键字
        ],
        'single_import_per_statement' => true, // 导入的每个声明有自己的关键字
        'single_line_after_imports' => true, // 每个命名空间的 use 独占一行,最后一个 use 后有一个空行
        'single_quote' => true, // 简单字符串的双引号转换为单引号
        'standardize_not_equals' => true, // 将 <> 替换为 !=
        'strict_comparison' => true, // 比较使用严格化
        'strict_param' => true, // 参数使用严格化
        'switch_case_space' => true, // 移除冒号和 case value 之间的多余空格
        'ternary_operator_spaces' => true, // 移除三元运算符周围多余的空格
        'ternary_to_null_coalescing' => true, // 使用 null 合并运算符 ?? 在可以的地方
        'trailing_comma_in_multiline_array' => true, // 多个数组元素的最后一个元素后有一个逗号
        'trim_array_spaces' => true, // 移除数组首尾元素多余的空格
        'unary_operator_spaces' => true, // 一元操作符紧挨操作项,不需要空格分开
        'visibility_required' => true, // 访问修饰符放置在属性方法前,abstract 和 final 放置在访问修饰符前面,static 放置在访问修饰符后
        'whitespace_after_comma_in_array' => true, // 数组的每个元素逗号后面有一个空格
    ])
    ->setFinder($finder);

执行格式化

  • 格式化期间 php-cs-fixer 会自动创建 .php_cs.cache 文件
  • 不同的代码量和复杂程度会影响执行的时长
1
2
// 指定文件进行格式化
$ php-cs-fixer fix ./src/Test.php

格式化后的代码

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
<?php
declare(strict_types=1);

namespace A;

use SebastianBergmann\CodeCoverage\InvalidArgumentException;

abstract class Foo extends FooBaseClass implements
    Bar1,
    Bar2,
    Bar3
{
    protected const FIRST = 'first';

    protected const SECOND = 0;

    protected const Z = -1;

    public const E_FOO = 200;

    public const E_BAR = 400;

    private $isNull;

    protected $hello;
    protected $world;

    public $numbers = ['one', 'two', 'three', 'four', 'five', 'six'];

    public $v = 0;

    public $path = 'root';

    public function bar(
        $v,
        $w = 'a'
    ) {
        $y = $w;
        $result = foo(
            'arg1',
            'arg2',
            10
        );
        switch ($v) {
            case 0:
                return 1;
            case 1:
                echo '1';

                break;
            case  2:
                break;
            default:
                $result = 10;
        }

        return$result;
    }

    public static function fOne(
        $argA,
        $argB,
        $argC,
        $argD,
        $argE,
        $argF,
        $argG,
        $argH
    ) {
        $x = $argA + $argB + $argC
            + $argD + $argE + $argF
            + $argG + $argH;
        list($field1, $field2,
            $field3, $filed4,
            $field5, $field6) = explode(',', $x);
        fTwo(
            $argA,
            $argB,
            $argC,
            fThree(
                $argD,
                    $argE,
                    $argF,
                    $argG,
                $argH
            )
        );
        $z = $argA === 'Some string'
            ? 'yes' : 'no';
        $colors = [
            'red',
            'green',
              'blue',
            'black'  ,
            'white',
            'gray'  ,
        ];
        $count = count($colors);
        for ($i = 0; $i < $count; $i++) {
            $colorString
                = $colors[$i];
        }
    }

    public function fTwo(
        $strA,
        $strB,
        $strC,
        $strD
    ) {
        if ($strA === 'one'
        || $strB === 'two'
        || $strC === 'three') {
            return $strA + $strB
            + $strC;
        }
        if ($strA === 'hello') {
            $strB = 'world';
        }
        if ($strB === 'second') {
            $strA = null;

            throw new \InvalidArgumentException('Invalid argument.');
        }
        $x = $foo->one('a', 'b')
            ->two('c', 'd', 'e')
            ->three('fg')->four();
        $y = a()->b()->c();

        return $strD;
    }

    public function fThree(
        $strA,
        $strB,
        $strC,
        $strD,
        $strE
    ) {
        try {
        } catch (Exception $e) {
            foo();
        } finally {
            // do something
        }

        return $strA + $strB + $strC
            + $strD + $strE;
    }

    abstract protected function fFour();
}

final class Bar
{
    final public function foo($x, $z)
    {
        global $k, $s1;
        $obj->foo()->bar();
        $arr = [0 => 'zero', 1 => 'one'];
        call_func(function () {
            return 0;
        });
        for ($i = 0; $i < $x; $i++) {
            $y += ($y ^ 0x123) << 2;
        }
        $k = $x > 15 ? 1 : 2;
        $k = $x ?: 0;
        $k = $x ?? $z;
        $k = $x <=> $z;
        do {
            try {
                if (! 0 > $x && ! $x < 10) {
                    while ($x !== $y) {
                        $x = f($x * 3 + 5);
                    }
                    $z += 2;
                } elseif ($x > 20) {
                    $z = $x << 1;
                } else {
                    $z = $x | 2;
                }
                $j = (int) $z;
                switch ($j) {
                    case 0:
                        $s1 = (string) '100';

                        break;
                    case 2:
                        $s1 = 'two';

                        break;
                    default:
                        $s1 = 'other';
                }
            } catch (\Exception $e) {
                $t = $one[0];
                $u = $one['str'];
                $v = $one[$x[1]];
                echo $val{foo . $num}[$cell{$a}];
            } finally {// do something
            }
        } while ($x < 0);
    }
}

function foo()
{
    return 0;
}

function bar(
    $x,
    $y,
    int $z = 1
) {
    $x = 0;
    // $x = 1
    do {
        $y += 1;
    } while ($y < 10);
    if (true) {
        $x = 10;
    } elseif ($y < 10) {
        $x = 5;
    } elseif (true) {
        $x = 5;
    }
    for ($i = 0; $i < 10; $i++) {
        $yy = $x > 2 ? 1 : 2;
    }
    while (true) {
        $x = 0;
    }
    do {
        $x += 1;
    } while (true);
    foreach ([
        'a' => 0,
        'b' => 1,
        'c' => 2,
    ] as $e1) {
        echo$e1;
    }
    $count = 10;
    $x = [
        'x', 'y',
        [
            1 => 'abc',
            2 => 'def',
            3 => 'ghi',
        ],
    ];
    $zz = [
        0.1,
        0.2,
        0.3,
        0.4,
    ];
    $x = [
        0 => 'zero',
        123 => 'one two three',
        25 => 'two five',
    ];
    bar(0, bar(
        1,
        'b'
    ));
}

function foobar($a):Bar
{
    $a = null === $a ? $a : new $a();
    include 'a.php';
}

function emptyreturn()
{
}

$map = array_map(function ($value) {
    return$value + 10;
}, $array);
?>
<div><?= foo() ?></div>

结论

原始代码

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
<?php

$result = [];
$array = [
    'x' =>     0  ,
        'a' =>     1  ,
            'b' =>     2  ,
                'c' =>     3  ,
                    'd' =>     4                   ,
                'e' =>     5  ,
            'f' =>     6  ,
        'g' =>     7  ,
    'y' =>     8  ,
'z' => 9,                ];
foreach ($array as $key => $val) {
    if (in_array($key, ['a', 'b', 'c'], true)) {
        continue;
    }
    $result[$key] = $val;
}

PHP_CodeSniffer 格式化

执行 phpcbf ./src/Test.php 后:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
<?php

$result = [];
$array  = [
    'x' => 0  ,
    'a' => 1  ,
    'b' => 2  ,
    'c' => 3  ,
    'd' => 4        ,
    'e' => 5  ,
    'f' => 6  ,
    'g' => 7  ,
    'y' => 8  ,
    'z' => 9,
];
foreach ($array as $key => $val) {
    if (in_array($key, ['a', 'b', 'c'], true)) {
        continue;
    }

    $result[$key] = $val;
}

PHP-CS-Fixer 格式化

执行 php-cs-fixer fix ./src/Test.php 后:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
<?php

$result = [];
$array = [
    'x' => 0  ,
        'a' => 1  ,
            'b' => 2  ,
                'c' => 3  ,
                    'd' => 4        ,
                'e' => 5  ,
            'f' => 6  ,
        'g' => 7  ,
    'y' => 8  ,
'z' => 9,                ];
foreach ($array as $key => $val) {
    if (in_array($key, ['a', 'b', 'c'], true)) {
        continue;
    }
    $result[$key] = $val;
}

经过对两个工具的使用,发现各有所长,PHP_CodeSniffer 囊括代码质量检测和自动修复,而 PHP-CS-Fixer 则只有自动修复功能,无论哪种工具,在我测试的过程中,都没法把用户主动添加的一些空格给利索的去掉,略有点失望。