面向对象编程实践--迭代器模式

关键字: 迭代器模式, 关注点分离

返回引言目录

最近工作中,要处理一个生成查询条件问题,代码写的很乱. 认真思考后觉得可以重构一下,让它更加面向对象.
如果大家有更好的建议,欢迎提出讨论。如果有类似问题也可以在评论去提出。谢谢

需求####

需求: 用户通过QueryBuilder,构建出筛选条件, 筛选可能使用几张表内字段, 根据筛选所使用的字段生成几张表的Join关系. 如果某个筛选条件是选项查询,则需要查出该字段所有的选项值。

需求不复杂,只需要找出用的所有字段,查出对应的表名,根据规则构建出SQL的Join条件就好。 以及根据查询条件确定选项字段,查出对应的选项值即可。 问题是QueryBuilder生成的对象是一个树状结构,需要遍历整颗树来找出所有字段. 这个也不复杂.

代码初稿#####
public static void GetAllDistinctFieldID(List<QueryBuilderRuleViewModel> rules, List<string> fieldIDs, bool isOption)
{
    if(rules != null)
    {
        foreach (var item in rules)
        {
            if (item.rules == null)
            {
                if(string.IsNullOrWhiteSpace(item.field)== false && fieldIDs.Contains(item.field) == false)
                                {
                                       if(isOption)
                                       {
                                              if(item.input == "select”) fieldIDs.Add(item.field);
                                        }else
                                        {
                                            fieldIDs.Add(item.field);
                                        }
                                  }
            }
            else
            {
                GetAllDistinctFieldID(item.rules, fieldIDs, isOption);
            }
        }
    }
}

上面代码很简单,好像也没有什么问题, 递归查找整个树,找出叶子节点, 根据条件把叶子节点的字段名取出来,去重加入到字段列表就好.
实际代码比这个复杂点,这是整理过后的版本
这个版本有什么问题了,太过面向过程了,不够面向对象.:-)

  • 混合两种处理取字段逻辑,相互干扰
  • 混合遍历字段的技术需求和取字段的业务需求
  • 代码阅读起来很累,需要同时考虑两种逻辑(遍历和选择业务)
  • 当业务需求变更时,维护困难

简单思考后,决定采用下面的设计思路来改造上面的代码

OO原则:####

关注点分离(Separation of concerns,SOC)是对只与“特定概念、目标”(关注点)相关联的软件组成部分进行“标识、封装和操纵”的能力,即标识、封装和操纵关注点的能力。是处理复杂性的一个原则。由于关注点混杂在一起会导致复杂性大大增加,所以能够把不同的关注点分离开来,分别处理就是处理复杂性的一个原则,一种方法。

设计模式:####

迭代器模式是一种设计模式,是一种最简单也最常见的设计模式。它可以让用户通过特定的接口巡访容器中的每一个元素而不用了解底层的实现。

改造后的代码:
迭代器,只用来遍历所有规则,采用了堆栈来避免递归。但是这个不是重点,用递归也一样可以实现迭代器。

public static IEnumerable<QueryBuilderRuleViewModel> FlatRule(this List<QueryBuilderRuleViewModel> qbrs)
{
     if (qbrs != null){
         Stack<QueryBuilderRuleViewModel> ruleStack = new Stack<QueryBuilderRuleViewModel>();
         qbrs.ForEach(p => ruleStack.Push(p));                
         while (ruleStack.Count > 0)
         {
            var ruleItem = ruleStack.Pop();
              if (ruleItem.rules == null)
                  yield return ruleItem;
             else
                 ruleItem.rules.ForEach(p => ruleStack.Push(p));
          }
    }
}

查出所有使用的字段

queryFieldIDs = qbr.rules.FlatRule()
                    .Select(p => p.id)//选出字段ID
                    .Where(p => string.IsNullOrWhiteSpace(p) == false)//排除空ID,无效ID
                    .Distinct()//去重
                    .ToList();

查出所有需要检索选项值的字段。

queryFieldIDs = qbr.rules.FlatRule()
                    .Where(p=>p.Input == "select")//选出选项查询条件
                    .Select(p => p.id)//选出字段ID
                    .Where(p => string.IsNullOrWhiteSpace(p) == false)//排除空ID,无效ID
                    .Distinct()//去重
                    .ToList();
点评######

改造后的代码,看起来还比以前多一点。但是代码逻辑已经分开,不同的部分处理不同的逻辑,每处的关注点都不同。
在迭代器主要是技术细节,如何遍历整课树。这个逻辑可以一直使用,只和QueryBuilder的存储结构相关。
而后面两段逻辑,则是业务逻辑,代码具有极强的可读性,和需求提供的业务规则基本是用同一种语言描述的。当业务规则变化,就可以很快的修改这段逻辑。调试,维护时,也可以很明白的看出这段逻辑有没有问题。
这就是关注点分离原则的作用。
迭代器模式是一个简单模式,但是简单的模式用的好一样都可以起大作用。

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

推荐阅读更多精彩内容