程序员, 不要过度封装

过度封装是程序员最容易犯的错, 因为是否过度并没有固定的数值标准, 只能是有经验的程序员基于科学的判断. 过度封装的危害十分严重, 所以必须重视, 并且极力避免.

首先, 什么是"过度封装"? 我们知道, 计算机科学领域最称为经典也是最强大的思想便是递归, 分而治之. 但是, 递归本身最重要的因素是: 结束条件. 一生二, 二生三, 三生万物, 总要有终止的条件吧? 不然一辆车不停地急驰, 你怎样上车?

新手程序员, 特别是非科班出身的文科程序员, 最容易犯了过度封装的错误的原因就是不知道何时适可而止, 脑筋往往不断地分解分解再分解问题, 最后, 你看到的就是这个调用那个, 那个引用这个, 乱七八糟毫无头绪, 三两行代码的函数比比皆是, 自以为封装抽象, 其实是拿捏不好这个度.

一旦过度封装, 直接的危害往往是写了无数行代码, 封装了N多个类, 就是看不出一个完整的功能, 因为分裂起来收不住. 即使最终把功能实现了, 代码维护性也让人不忍直视.

如何判断是否过度封装? 如何避免过度封装? 我认为, 只要掌握了一个原则, 时刻以这个原则为准绳, 那便几乎不会犯过度封装的错误.

这个原则便是"直观优先"原则, 代码逻辑以直观为最优先的原则, 其它的原则均低于此原则, 即使有一百个考虑, 如果违反了直观原则, 都必须放弃掉而优先保证直观原则.

举一个例子:

function closeDialog() {
    login.close();
    message.close();
    ...close();
}

var action = {
    close: function () {
        closeDialog();
    },
    backout: function () {
        if (BACKOUT_ID) {
            ...
        }
    }
};

$(document).on('click', function (e){
    var $that = $(e.target);
    for ( var k in action) {
        if ( $that.hasClass('action-'+k) ) {
            action[k].call(e.target);
        }
    }
});

这样的代码就是过度封装的经典体现!

首先, closeDialog() 这个函数不应该存在, 而应该被 展开 , 因为函数只被调用一次, 而且没有逻辑上的独立性的必要.

其次, 上面的函数通过跳转表的方式来逃避直接的函数调用, 这是对跳转表技术的极大滥用! if-else 加函数调用是最直观的分支跳转方式, 完全没有必要在这个例子里使用跳转表技术.

先解释一下什么是跳转表技术, 这是一个非常古老而基础的计算机编程领域的技术. 跳转表包含很多层含义, 简单的说, 就是将逻辑代码块放到一个索引表中, 代码编写过程通过传递索引项来最终调用相应的逻辑代码块(因为多处引用).

第一次听到或者见到跳转表技术的程序员的心情, 如果你想体验的话, 你想想当你写了无数的 if-else 之后, 你发现了 switch-case 时的心情, 或者是你第一次听到函数指针这个事物时的心情.

但是, 像 js 之类高级语言, 已经把跳转表技术, 直接融合到了语言的语法本身, 如类(class). 你不需要实现自己的跳转表技术, 你只需要用好语言本身的特性即可.

所以, 上面的代码应该改成:

$(document).on('click', function (e){
    var $that = $(e.target);
    if ( $that.hasClass('action-close') ) {
        login.close();
        message.close();
        ...close();
    }
    if ( $that.hasClass('action-backout') ) {
        if (BACKOUT_ID) {
            ...
        }
    }
});
这是最直观, 最正常, 人人都应该写成这样的代码. 如果你还想显式地使用查找表技术以体现自己牛逼, 我是说, 如果你真想装逼的话, 你可以这样写:

var action = {
    close: function () {
        login.close();
        message.close();
        ...close();
    },
    backout: function () {
        if (BACKOUT_ID) {
            ...
        }
    }
};

$(document).on('click', function (e){
    var $that = $(e.target);
    var func_name = $(that).attr('action');
    if(func_name && action[func_name]){
        action[func_name]();
    }
});
但我劝你最好不要装逼也别装傻, 乖乖地有理有据有逻辑地写代码, 让代码的逻辑直观起来, 不要过度封装. 如果你内心封装的欲望太强的话, 那我劝你学学 Go 语言, 把自己的思想强加到一种自己新发明的语言上来(我估计你没那种本事). 不然, 直观优先, 不要过度封装.


转载 http://www.ideawu.net/blog/archives/959.html?utm_source=tuicool&utm_medium=referral

  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值