设计模式漫谈

开门见山一句话,我认为设计模式的核心就是“封装变化点”,用古早软件工程的话语体系说就是“低耦合,高内聚”,用糙一点的话说就是既不要重复代码,又要好扩展。

比如:

这些东西吧,就是理解的人不用多说,不理解的人多说也没用。我是一个极度不擅长记忆的人,尤其年龄大了,n 年前看过的设计模式,早忘的一干二净了。所以基本当别人扯到设计模式的时候,我一般都敬而远之,因为很多时候别人说一个设计模式的时候,我都不记得这个设计模式到底是干嘛用的了。这里还刚妄言设计模式,纯粹是想表述一下我对设计的理解与实践,话不多说,直接 show code。

举例

以 Android 中的列表举例,我们可以先把元素列出,看看哪些属于模版代码

  1. url
  2. 返回的数据结构
  3. 网络请求框架
  4. 列表展示的 UI
  5. 分页逻辑
  6. 下拉刷新
  7. 网络请求失败展示错误提示
  8. 列表条目点击事件处理

暂时只列这几个比较公共的逻辑,我们可以挨个分析一下这些元素哪些是“公共”的,哪些是独有的

url

以 restful api 为例,格式为 ${domain}/${version}/${targetObj}?offset=${offsetNum}&limit=${limitNum} 我们可以看到其实其中的五个参数,只有 ${targetObj} 是与本次业务相关,其他都是公共代码

返回的数据结构

返回的数据如下:

{
  "errorCode": "",
  "errorMsg": "",
  "results": [
    {
      "id": "xxxxxx1",
      ...
    },
    {
      "id": "xxxxxx2",
      ...
    }
  ]
}

其中 errorCode、errorMsg、results 也都是样式代码,都可以通过 json 解析一次性解决问题,只有 results 中的数据是不同的

在 kotlin 中基本都是一行代码解决问题:

data class DataAItem(val id: String, val others: String, ...)

网络请求框架

这个不多说,Android 端现在的最佳实践就是 retrofit + okhttp

列表展示的 UI

在 Android 中,可以简单理解为单条目的 UI 对应的其实就是 holder

class DataAItemHolder(context: Context, root: ViewGroup) : BaseViewHolder<DataAItem>(context, root, R.layout.layout_data_a_item) {

    override fun bindData(item: DataItem) {
        binding.idView.text = item.id
        binding.othersView.setContent(item.others)

        binding.idView.setOnClickListener {
            context.startActivity(...)
        }
    }
}

为了篇幅,这里就不列 R.layout.layout_data_a_item 了,相信 Androider 都明白

分页逻辑、下拉刷新、网络请求失败展示错误提示

与 url 结合,只要当时约定的 api 是格式化的,那么这里的分页逻辑与下拉刷新其实也都是公共的,因为所有类似列表页面的形式都是相同的 至于错误展示,基本逻辑也都是公共的

不同点:

我们最后说这些问题的处理

列表条目点击事件处理

这个是根据内容不同事件是不同的,但是这部分的逻辑是可以些在 DataItemAHolder 中的,见上文中定义的 DataItemAHolder

理想完整形态

所以一个单独的列表页面理论上的所有代码便如下:

data class DataAItem(val id: String, val others: String, ...)

class DataAItemHolder(context: Context, root: ViewGroup) : BaseViewHolder<DataAItem>(context, root, R.layout.layout_data_a_item) { ... }

class DataAListFragment : BaseListFragment<CommonListBinding>() {

    init {
        setPageUrlTarget("${targetObj}");
        registerHolder(DataAItemHolder::class.java)
    }
}

// 如果用注解形式,则更简洁
@Endpoint("targetObj")
@Holder(DataAItemHolder::class.java)
class DataAListFragment : BaseListFragment<CommonListBinding>() {}

还有一个 layout_data_a_item.xml

所有变化点都在代码里了,以这种形式去实现一个列表,就只有 layout_data_a_item.xml 会稍微费点时间,总共加起来也不会超过 1 小时,而且逻辑清晰、代码简洁、便于维护。

不需要 adapter,不需要 LayoutManager,不需要 ItemDecoration,甚至,这个 DataAListFragment.kt 都是模版代码,既然是模版代码,那就可以动态生成。 哪怕上述代码只能够替代 50% 的真实列表需求,其实都是极大的劳动力的解放。

至于为什么是理想完整形态,是因为我也没有完全实现上述逻辑,主要是以往写 sdk 居多,少写 UI,以上逻辑都是在我大概五六年前写过的一个框架的基础上优化而来。

问题

上边看着舒服,但是其实问题还是很多的。如果所有逻辑都往 base 或者 common 中塞,不用我多说,大家也知道是垃圾设计
我们做到了不要重复代码,那好扩展怎么办呢?
像上边 分页逻辑、下拉刷新、网络请求失败展示错误提示 中所述的不同点,还有其他的:

我们拿其中的几个举例:

分页开关

BaseBindingFragment 中:

fun enablePaging(): Boolean {
    return true
}

这就是简单的模版模式。如果 DataAListFragment 是动态生成的,那可以使用 Builder 模式。

Holder 如何创建

这里其实出现了反向依赖。正常来讲,如果要创建具体的 DataAItemHolder,那么模版代码一定要依赖 DataAItemHolder,不然没法调用构造函数。 这里解决方案是固定构造函数:

  class DataAItemHolder(context: Context, root: ViewGroup) : BaseViewHolder<DataAItem>(context, root, R.layout.layout_data_a_item) {}

即所有 holder 的构造函数都是固定的 context: Context, root: ViewGroup,那可以在 Adapter 中

  override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): CommonViewHolder<T> {
      holderTypeMap[viewType]?.let {
          val holder = it.getConstructor(Context::class.java, ViewGroup::class.java).newInstance(parent.context, parent)
          holder.setOnClickListener(viewHolderClickListener)
          return holder
      }
      throw IllegalArgumentException("The type :$viewType create exception")
  }

通过这种形式创建,这里的 holderTypeMap 可以理解成 DataAItem 的缓存,即去看一下是否已经注册过该 Holder 了,因为 DataAItem 与 DataAItemHolder 是 1v1 绑定的关系。

当然还有其他解决方案,不过我个人目前感觉这应该算是比较好的。

其他

其他问题怎么解决大家可以自己想办法,这里不做赘述。

大总结

以上使用了哪些设计模式?我也不知道。其实只要知道了 理想完整形态,那剩下的就是想办法去解决具体细节问题了,这些细节工作量占 80%,但是从设计角度讲大概只占 20%。不管怎么搞,只要能完成既不要重复代码,又要好扩展的目标,其实就不用管啥设计模式了。重意不重形。

番外篇

我这些年大多是做 sdk 开发,呆过的公司不少,亲眼见证了不少公司从 native -> h5 -> RN -> native -> flutter(or kmm) 的技术路线修改,五味杂陈。变得只是技术路线,写代码的依然还是3年+ 初级工程师
再一个例子,前公司为了增效专门聘请了一个敏捷教练,这其实与换技术路线如出一辙。这就像是拿着设计模式往里套,套不进去再换一个。

大部分人不是尽力去解决问题,而是把时光和精力花在绕过问题上。换技术路线并不能解决产品逻辑问题,也不能解决3年+ 初级工程师的问题。这两个问题才是核心。 产品逻辑问题由人去解决,3年+ 初级工程师的问题也是人的问题。而人的问题要从解决,而不应该从外部下手。所谓的无非也是两个,一是能力,二是责任心

责任心问题

正常开发中,一定是前后端协同、rd pm 协同、产研与运营销售等部门协同,我一直认为好的业务(产品的理想完整形态)一定可以引导出代码上的合理架构。如果代码架构乱七八糟,原因无非两个,一是产品逻辑问题,二是程序员能力问题。这种通过代码设计过程中体现出的问题,绝大多数都可以追溯到产品逻辑上。这是产品优化的及其重要的一条渠道,但是遗憾的大多数时候,这条渠道名存实亡。原因无非也是两个,一是程序员的责任心问题、二是 pm 的责任心问题,大多数人本着能少一事就少一事的原则混饭吃,放任不合理的产品设计,自己也写不负责任的代码。当然万方有罪,罪在朕躬

我是亲眼见过运维的同学在月度总结会上把影响公司营收10%以上的事故当成笑话讲,我也亲眼见过只做营销活动而一点不关心产品的事业部总监。其实我想说,程式化对应员工的公司,一定也会收获程式化应对的员工,这就是你糊弄我,我糊弄你,最终双输的局面,这其实是我离职这么多家公司的最核心原因。

能力问题

程序员更应该增加的对产品和业务的感知能力,产品、迭代流程、管理,其实都是可重构的,核心从来都不是设计模式,而是找到理想完整形态并落实。只有锻炼审美能力,才能知道代码的丑、产品的丑以及管理的丑。

关于二者的解决方案其实很简单,就是人为本。公司中其实是员工占主体,但管理层是大脑。管理层建立正向循环机制,逐步剔除混日子员工。你要是还问我怎么建立正向循环机制,那你是没理解人为本

闲庭随笔,大话漫谈,鄙俚浅陋,诸君勿怪。