代码大全2 - 表驱动法

2019-11-04  本文已影响0人  前行的乌龟
image

最近入手了 代码大全2,真实程序员的圣经啊,内容好高端啊,书中建议的初级程序员看的章节就是:表驱动法,真 NM NB,听都没听过,但是非常有用,其想法绝对不是一个人通过自我学习能醒悟出来的

表驱动法 的思路是什么:用表来代替那些 if/else 的业务逻辑,你把业务逻辑写在 if/else 里就是硬编码,但是写到表里是可以实现功能分离的,逻辑表可以统一做路径设置或者其他配置,这样的话业务逻辑会变得非常容易阅读和理解了

我不说原理,书光说原理了,有限的代码还是 VB 的,压根没法看,所以我给大家举些例子,大家看代码体会,思想得落地才能变成自己的


今天周几

像几天周几这样的,其实最适合做表驱动了,因为逻辑最清晰,简单,一个数组+下标,轻轻松松搞定

传统写法:

String today = "周日";
Switch( dayForMonth % 7 ){
    case 0 : 
        today = "周日";
    case 1 : 
        today = "周一";   
    case 2 :
        today = "周二";   
    case 3 :
        today = "周三";   
    case 4 :
        today = "周四";   
    case 5 :
        today = "周五";   
    default:
        today = "周六";   
}

表驱动法

String [] weekday = new String[]{"周日","周一","周二","周三","周四","周五","周六"};
String today = weekday [ dayForMonth % 7 ];

每个月对应多少天

传统的写法,我们要写一串 if/else 返回数据

传统写法:

if(1 == iMonth) {iDays = 31;}
  else if(2 == iMonth) {iDays = 28;}
  else if(3 == iMonth) {iDays = 31;}
  else if(4 == iMonth) {iDays = 30;}
  else if(5 == iMonth) {iDays = 31;}
  else if(6 == iMonth) {iDays = 30;}
  else if(7 == iMonth) {iDays = 31;}
  else if(8 == iMonth) {iDays = 31;}
  else if(9 == iMonth) {iDays = 30;}
  else if(10 == iMonth) {iDays = 31;}
  else if(11 == iMonth) {iDays = 30;}
  else if(12 == iMonth) {iDays = 31;}

但是表驱动法就不一样了,把逻辑写成 map 或是 list,一目了然,这里我们搞个2维数组还加上了闰年的逻辑

表驱动法

const monthDays = [
  [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31],
  [31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31]
]

function getMonthDays(month, year) {
  let isLeapYear = (year % 4 === 0) && (year % 100 !== 0 || year % 400 === 0) ? 1 : 0
  return monthDays[isLeapYear][(month - 1)];
}

console.log(getMonthDays(2, 2000))

不同条件执行不同任务

因为是任务,是个方法,不再是数值了,这里我们可以利用 Dart 这样的支持高阶函数的语言特性,把方法当做一个对象存储在表中

表驱动法

  var data = <String, Map>{
    "A": {
      "name": "AA",
      "action": (name) => print(name + "/AA"),
    },
    "B": {
      "name": "BB",
      "action": (name) => print(name + "/BB"),
    },
  };
  
  var action = data["A"]["action"];
  action("kk");
  
  // if( action is Function ) action("KK");

action 可能会包红线,提示不是方法类型,大家强转一下就行了。这个内部可以无限的往下一层层嵌套,只要保证是 map 的就能通过[]统一书写调用了


加减法

加法和减法有不同的计算共公式,那么就像下面这样写,抽象方法当做对象使用,存到 map 表里去

传统写法:

def test(c,a,b):
    if c == '+':
        return a + b
    elif c == '-':
        return a - b

表驱动法:

def add(a,b):
    return a+b

def minus(a,b):
    return a - b

func_dict = {'+':add,'-':minus}
print(func_dict['+'](1,2))

复杂多条件判断

这个例子逻辑最复杂,因为判断的条件多,并且还有不同配置,像这种问题,我们一般用2维数组来做,x 轴是判断条件,y 轴是不同配置

传统写法:

if( (a && !c ) || (a && b && c)){
    category = 1 ;     
}else if( (b && ! a) || (a && c && !b)){
    category = 2 ;     
}else if ( c && !a && !b){
    category = 3 ;     
}else {
    category = 0 ;     
}

表驱动法:

// 把逻辑变为 2维数组,做好注释,这样看是不是很清晰啊
static int categoryTable[2][2]={
    //!b!c   !bc   b!c   bc
        0,   3,     2,    2, // !a    
        1,   2,     1,    1, // a
}
// 使用
category = categoryTable[1][0];

带取值范围的

像某某范围内是啥这样的,我们取两边的端点数值作为依据,在数值判断时使用最简单的方式:for 循环

表驱动法:

int [] grade = {59,79,84,89,94,100}; 
String [] level = {"F","E","D","C","B","A"},

public String getLevel (int grade){
    for( int i = 0 ; i < grade.length ; i++){
        if(grade <= grade[I]){
            return  level[I];
        }
    }
}
上一篇下一篇

猜你喜欢

热点阅读