意图导向编程

1.1 基本思想

“意图导向编程”,Programming by Intention,也称目的导向编程/自顶向下编程.
其基本思想是:
每一个问题都可以分解成一系列的功能性步骤,在写代码的过程中,会按照顺序有意识的去完成每一个步骤;而意图导向编程则是先假设每一个步骤都有一个理想的方法来完成,而不关注每个步骤的具体实现,在这种情况下,需要关心的是每个方法的输入参数,返回值以及什么样的名字最符合它的含义

例如,创建一个服务,它接受一个业务交易,然后提交,具体的需求如下:

  • 交易信息开始于一串标准的ASCII字符串。
  • 这个信息字符串必须转换成一个字符串的数组,数组存放的值是此次交易用到的领域语言(domain language)中所包含的词汇元素(token)。
  • 每一个词汇元素必须标准化(第一个字母大写,其余字母小写,空格和非字符数字的符号都要删掉)。
  • 超过150个词汇元素的交易,应该采用不同于小型交易的方式(不同的算法)来提交,以提高效率。
  • 如果提交成功,API方法返回true,否则返回false。

基于“意图导向编程”的思想,我们假设有一个类,类中有一个API实现上面的服务:

    public Boolean commit(String command){
        Boolean result = true;
        String[] tokens = tokenize(command);
        normalizeTokens(tokens);
        if(isALargeTransaction(tokens)){
            result = processLargeTransaction(tokens);
        }else{
            result = processSmallTransaction(tokens);
        }

        return result;
    }
}

commit()方法是程序API,用于提供服务,而其他方法(tokenize()、normalizeTokens()、isALargeTransaction()、processLargeTransaction()、processSmallTransaction())都不属于这个对象API,仅仅是实现过程中的功能性步骤,称之为“辅助方法”(helper methods),暂时可以将他们视为私有方法。

通过这样的编码方式,可以将精力集中在如何分解最终目标,以及那些全局性的问题上。
并且这种实现方式,与直接把所有代码写到一个很长的方法里相比并没有增加工作量,不同点在于思考的方式和编码的顺序(先理清整体流程,再深入细节)。

1.2 优点

如果遵循意图导向编程,那么代码将会:

  • 更加内聚(职责单一)。
  • 更加可读和清晰
  • 更易于调试
  • 更易于重构和优化
  • 更易于单元测试
  • 更易于维护
  • 创建的方法更容易从一个类移动到另一个类
  • 更容易应用设计模式

1.2.1 方法的内聚性

代码的质量标准之一就是内聚性。
以类为例,每个类都应该根据职责来定义,并且应该只有一个职责。类内部包括方法、状态以及与其他对象之间的关系,如果各个方面都紧密相关,并且围绕着这个类的唯一职责,则说这个类的内聚性很强。

如果一个方法只实现整体职责中一个单独的功能点,则说这个方法的内聚性很强。

人的思维方式是单线程的,当进行“多任务”的时候,实际上是在多个任务之间快速切换而已,人们仍旧习惯于一次只思考一件事情。意图导向编程正是利用这一事实,用思维链条单一性的特定去创建同样具备单一性的内聚方法。

1.2.2 可读性和表达性

通过阅读最初的实例代码:

public class Transaction{
    public Boolean commit(String command){
        Boolean result = true;
        String[] tokens = tokenize(command);
        normalizeTokens(tokens);
        if(isALargeTransaction(tokens)){
            result = processLargeTransaction(tokens);
        }else{
            result = processSmallTransaction(tokens);
        }

        return result;
    }
}

可以发现该服务的实现流程是:
获取到一个指令,然后对指令进行分词,再把分词后得到的指令标准化,判断需要进行交易处理的类型,根据判断结果来决定进行大型事务还是小型事务的处理,最后返回结果。

因为上面的代码只涉及到“做什么”,而不是具体的“如何做”,这种情况下,不需要注释也能读懂代码的基本逻辑,这得益于规范的命名和步骤的清晰界定。

考虑下面的实现方式:

public class Transaction{
    public Boolean commit(String command){
        Boolean result = true;

        // tokenize the string
        some code here
        some more code here
        even some more code here that sets tokens

        // normalize the tokens
        some code here that normalize tokens
        some more code here that normalize tokens
        even some more code here that normalize tokens

        // see if you have a large transaction
        code that determines if you have a large transaction
        set lt = true if you do
            if(lt){
                // process large transaction
                some code here to process large transaction
                some more code here to process large transaction
            }else{
                // process small transaction
                some code here to process small transaction
                some more code here to process small transaction
            }

        return result;
    }
}

上面的实现方式是将所有逻辑写在一个大方法中,并且加了详尽的注释,但与意图导向编程的实现方式相比,注释显得很没有必要,并且代码太多,给人的心理无形中造成一种压力。

1.2.3 调试

在程序的代码错误修复过程中,寻找错误所在才是最花时间的。在遵循意图导向编程时,由于一个方法只做一件事,这个时候,如果出现错误,则可试试下面的办法:

  • 通读一遍整个方法,看看所有事情是怎么运作的
  • 对无法正常工作的部分,检查辅助代码的细节有什么问题

相比于费力的查阅一大段复杂的代码,这种调试方法发现代码错误的速度要快很多。

1.2.4 重构和增强

重构系统:保持系统行为不变的情况下,更改它的结构。
增强系统:增加或修改系统的行为以符合新的需求。

重构通常认为是“清理”系统中写得糟糕的代码,重构的一个基本实现方式是:把一部分代码从一个巨大的方法中抽取出来,放到一个属于它自己的新方法中,而在原来代码中的那个位置直接调用这个新方法。

由于原来方法的一部分临时变量也需要迁移到新方法中,所以需要多个步骤才能完成一个函数的提炼。

如果采用意图导向编程,一开始就是辅助方法了,只需要把公用的辅助方法迁移到其他类即可。这样的重构是很快的(复制-粘贴)。

当系统实现后,有新需求加进来了,如:与第三方程序交互时,由于第三方程序的原因,不再支持某些旧版词汇,这个时候需要更新一个词汇元素,如:

public class Transaction{
    public Boolean commit(String command){
        Boolean result = true;
        String[] tokens = tokenize(command);
        normalizeTokens(tokens);
    updateTokens(tokens);
        if(isALargeTransaction(tokens)){
            result = processLargeTransaction(tokens);
        }else{
            result = processSmallTransaction(tokens);
        }

        return result;
    }
}

有新需求加进来的时候,只需要在API方法的实现流程中增加updateTokens()方法,其他都不需要修改到,把影响降到了最低。

如果修改了标准化的算法,则更改normalizeTokens()方法即可,其他都无需改动。在修改的过程中,代码能很快定位。

1.2.5 单元测试

设计的基本建议:使用服务的客户端,在设计时应该遵照的是服务的接口定义,而不是服务的具体实现。
在上面的实现中,辅助方法被定义成了私有方法,是为了不想与外部对象发生关联,但这种情况下就不利于方法的单元测试。
我们只能对commit()方法进行单元测试,即测试服务的整体行为。此时测试情况比较复杂,会有很多种因素导致测试失败。
可以有如下解决办法:

  • 如果辅助方法只是实现单个服务的一部分,则没必要单独测试辅助方法,测试这个服务流程即可。
  • 如果某些辅助方法是能被其他服务使用到的,则需要将该辅助方法单独到其他的类中,并且定义成公有的方法,则对原来辅助方法的调用就变成了对新类方法的调用,并且新类的公有方法是能进行单元测试的。

1.2.6 可迁移的方法

为了提高类的内聚性,需要把这个类不应该有的方法迁移到其他类中,这样可以让这个类所关注的东西减少。

意图导向编程创建的方法只完成一个功能,这样避免了迁移方法是经常遇到的问题:包含不能迁走的部分。
当一个方法只做一件事时,要么全部迁移,要么不迁移。

方法迁移难,还可能由于它直接关联到了类中的状态变量,在使用意图导向编程时,习惯于将参数传递到辅助方法,然后获取一个返回结果,而不是让方法直接使用对象的状态。

1.2.7 易于修改和扩展

从之前的重构和增强可看成,当增加需求时,只需要在流程中增加对应的辅助方法;
当需要修改需求时,只需要修改对应的辅助方法。这种修改和扩展容易定位并且不影响其他代码。

1.2.8 在代码中发现模式

上面的例子中,如果有两个不同的交易类型,流程步骤一样(分词、标准化、更新、处理),但每一步的实现方式不一样。
使用意图导向编程时,处理每个辅助方法具体实现不一样,commit()方法是一样的,这个时候,可以很容易的应用模板方法模式

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容