欢迎光临
我们一直在努力

聊聊代码的割裂感

早些年,我特别喜欢下围棋,每天都会下几盘。那时候日本围棋不仅高手林立,而且风格迥异,比如:小林光一的地铁流,武宫正树的宇宙流等等,不过我最喜欢的棋手当属大竹英雄,他下棋时追求美感,如果棋形不漂亮,那么他宁可认输也绝不玷污棋盘。后来,我成为了一名程序员,每天都要写不少代码,可惜写了不少丑陋的代码,本文筛选了几个例子,希望大家看过之后都能写出更具美感的代码来。

 

在聊之前,我们不妨想想割裂到底是什么意思。其实所谓割裂,说白了就是指把原本应该在一起的东西分开了,比如两地分居的夫妻,亦或者留守儿童和爸爸妈妈。

普通的语言描述总是苍白无力的,让我们看看有割裂感的代码到底长啥样:

<?php

$i = 0;

while ($i <= 10) {
    echo $i;
    $i++;
}

?>

代码似乎很常见,如果你没有意识到割裂感的存在,请看看下面的改进版:

<?php

for ($i = 0; $i <= 10; $i++) {
    echo $i;
}

?>

怎么样?理解了吧!在我们的努力下,变量「i」的三口之家终于团聚了!当然,这只是命令式语言的写法,如果你想更酷一点,还可以用函数式语言的写法:

<?php

array_map(function($i) { echo $i; }, range(0, 10));

?>

接着看另一个例子,如果想统计程序的运行时间,那么最简单的方法无疑是开头记录一下起始时间,最后记录一下结束时间,然后相减,就好像下面这样:

<?php

// head
$begin = microtime(true);

// mock
sleep(1);

// foot
$end = microtime(true);
echo $end - $begin;

?>

可惜如此一来前后时间相关代码无疑就两地分居了,想想看如何让它们团聚:

<?php

$begin = microtime(true);

register_shutdown_function(function() use($begin) {
    $end = microtime(true);
    echo $end - $begin;
});

sleep(1);

?>

BTW:当然,这用到了一下 PHP 本身的语言特性,如果放到其它语言里,也可以利用装饰器模式等方法来达到类似的效果,具体实现这里就不多说了。

最后再看一个例子,很多项目都会搞一个类统一存放错误码,如果业务出错了就返回相应的错误码,然后抛出相应的异常信息,大致代码如下所示:

<?php

class Result
{
    const ERROR_USERNAME = 1;
    const ERROR_PASSWORD = 2;

    public static $msg = array(
        1 => '用户名错误',
        2 => '密码错误',
    );
}

?>

可惜,错误码和错误信息割裂了。如此说来,改成数组怎么样:

<?php

class Result
{
    public static $msg = array(
        'ERROR_USERNAME' => '用户名错误',
        'ERROR_PASSWORD' => '密码错误',
    );
}

?>

不过这样数字标识就没有了,很多时候我们需要它,而且放弃了 const,也就意味不能再享受到对应的工具提示,错误检查。其实新版 PHP 的 const 可以赋值数组:

<?php

class Result
{
    const ERROR_USERNAME = [1, '用户名错误'];
    const ERROR_PASSWORD = [2, '密码错误'];
}

?>

两地分居的代码终于又团聚了!使用的时候只要抛出一个精心构造的异常就行了:

<?php

class ResultException extends Exception
{
    public function __construct(array $result = array())
    {
        if ($result) {
            parent::__construct($result[1], $result[0]);
        }
    }
}

throw new ResultException(Result::ERROR_USERNAME);

?>

结尾顺便说一句,搞一个类单独存放错误码是否可取存争议,通常我们认为这样的类是哑巴类,相对应的,错误码应该分散到它原本所属的业务对象之中去。不过这个问题和本文的主题关系不大,说起来就没完没了了,索性留到以后有空再聊。

赞(0) 打赏

评论 抢沙发

评论前必须登录!

 

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

微信扫一扫打赏