在写代码的风雨进程中,我们如何来衡量代码是否很棒,很流弊呢?
我们先来看一个概念:
高内聚低耦合,是软件工程中的概念,是判断设计好坏的标准,主要是面向对象的设计,主要是看类的内聚性是否高,耦合度是否低。所谓高内聚是指一个软件模块是由相关性很强的代码组成,只负责一项任务,也就是常说的单一责任原则。对于低耦合,粗浅的理解是:一个完整的系统,模块与模块之间,尽可能的使其独立存在。也就是说,让每个模块,尽可能的独立完成某个特定的子功能。模块与模块之间的接口,尽量的少而简单。如果某两个模块间的关系比较复杂的话,最好首先考虑进一步的模块划分。这样有利于修改和组合。
我们了解过高内聚低耦合这个概念后,有人想问是否能很快就掌握这个思想,写出来的代码也能符合这个规则呢?答案是否定的,我们想要熟练的在代码中用到高内聚低耦合这个思想,是需要我们慢慢沉淀的,需要写大量的代码,才能把这个把它掌握,而函数回调就是它的一个简单实现.
接下来我们用c语言的结构体来简单的了解什么是函数回调.首先我们建立一个Student结构体,里面包含姓名,年龄,性别,成绩四个变量.
typedef struct student{ //定义student结构体,结构体名为STUDENT
char name[20];//姓名
char sex;//性别
float score;//成绩
int age;//年龄
}STUDENT;
然后再main.m函数中初始化5个学生
前提工作都做好了,那么现在问题来了,我要求你写一个函数来对这5个学生的姓名进行升序排序.一般的情况下,我们拿到这个需求后,上去直接开干,写了如下函数:
写完之后,在main.m里调用该函数很容易的把排序结果打印到了控制台:
当你看到名字按升序排列,很对,自己感觉也很爽的时候,产品经理又提需求了,让你对年龄进行升序排列.你拿一想,这也挺简单的吗,二话不说,上去就是敲.几分钟后又有了如下代码:
敲完了年龄按升序排列的函数,感觉自己更加流弊了,这简直太简单了吧,正在飘飘然的时候,产品经理又来了,又对你提了一个需求,让你对成绩进行升序排列,瞬间满头黑线.....这个时候,我们就该换条思路来对这些需求进行实现了.
通过观察我们能很容易的发现,三个需求不管是对年龄排序,还是对姓名排序,还是对成绩排序,在上面的排序函数中我们只是改变了函数的if()中的判断条件,其他的基本没有任何变动.那么现在我们该如何做,才能使我们只写一个排序函数,又能同时满足刚才产品经理的三个需求呢?
这个时候,我们就要用到了我们要说的函数回调这个方法.函数回调就是通过函数指针调用的函数.接下来我们来一步步用函数回调的方法来优化上面的代码.
首先,我们需要定义一个函数指针这样方便下面调用:
//定义一个指向参数为2个(STUDENT )类型的指针函数,函数名SortMethod,返回值为布尔类型
typedef BOOL (*SortMethod)(STUDENT , STUDENT);
接下来,我们需要对上面的排序函数进行改进,再传进去一个参数(函数指针),用这个参数来对判断条件进行改变.废话不多说,直接上代码:
写完主要的排序函数,我们就需要对单独的判断条件写出独立的函数(需要符合上面函数指针指向的函数类型).我们先写一个按年龄的排序的判断条件函数.
sortStudent(stus, 5, sortByScore);通过在主函数内调用这个排序函数,就能输出想要的年龄结果.同时如果要按姓名排序,写一个类似的方法,如下:
在主函数中调用排序方法sortStudent,并把按性别排序的函数名当形参传入即可,如下:
sortStudent(stus, 5, sortByName);
通过上面的代码,我们能够感觉到使用函数回调可以优化我们写的代码,减少大量重复的工作.对上面的例子来说,当我们需要对学生的某个变量进行排序,只需要写一个排序方法,利用函数回调,就可以完成,并且有新需求的时候,并不需要在整个的排序函数里面重新改写,或重写排序函数,只需要在外面写一个排序条件的函数,通过指向函数的指针传进去即可.
想要成为一个优秀的程序员,就要具备这样的思维模式,本篇的函数回调只是简单的说明了高内聚低耦合的思想,还需要大家在以后的写代码中多注意,多练习,才能有更大的进步.