Dart 编码规范:`多态` 和 `控制反转`最佳实践

前言

我们通常都知道程序设计要依赖抽象,提高复用性,做到对扩展开放,对修改关闭。贯彻SOLID五大原则的最重要法宝就是抽象继承多态是一种手段,下面,通过简单 demo 介绍 flutter 开发中常用的最佳实践。

举栗子🌰

/// 不推荐,避免把逻辑放在公共底层处理
class TestWidget extends StatefulWidget {
  const TestWidget({Key? key}) : super(key: key);

  @override
  TestWidgetState createState() => TestWidgetState();
}

class TestWidgetState extends State<TestWidget> {
  dynamic param;
  Widget childWidget = Container();

  ///提供给外部调用
  void update(dynamic value) {
    setState(() {
      param = value;
      _updateBody();
    });
  }
  
  void _updateBody(){
    if(param == null){
      childWidget = Container();
      return;
    }
    switch(param.runtimeType){
      case A:
        childWidget = AWidget(a: param as A,);
        break;
      case B:
        childWidget = BWidget(b: param as B,);
        break;
      default:
        childWidget = Container();
        break;
    }
  }

  @override
  Widget build(BuildContext context) {
    return Container(
      child: childWidget,
    );
  }
}

如上的写法是不推荐的,应该进行依赖倒置,将可变的部分放上层处理,保证底层干净,如下:

  • 方式一:通过 传参构造器 进行控制反转
typedef ChildBuilder = Widget Function(dynamic param);
class TestWidget extends StatefulWidget {
  final ChildBuilder builder;

  const TestWidget({Key? key, required this.builder}) : super(key: key);

  @override
  TestWidgetState createState() => TestWidgetState();
}

class TestWidgetState extends State<TestWidget> {
  dynamic param;

  void update(dynamic value) {
    setState(() {
      param = value;
    });
  }
  
  @override
  Widget build(BuildContext context) {
    return Container(
      child: widget.builder(param),
    );
  }
}


///外部使用 TestWidget
Widget _builder(dynamic param){
  if (param == null) {
    return Container();
  }
  switch (param.runtimeType) {
    case A:
      return AWidget(
        a: param as A,
      );
    case B:
      return BWidget(
        b: param as B,
      );
    default:
      return Container();
  }
}

return TestWidget(builder: _builder);
  • 方式二:通过 继承 + 泛型 进行解耦
/// 将 TestWidget 改成抽象类并指定泛型
abstract class TestWidget<T> extends StatefulWidget {
  const TestWidget({Key? key}) : super(key: key);

  Widget childBuilder(T param);

  @override
  TestWidgetState<T> createState() => TestWidgetState<T>();
}

class TestWidgetState<T> extends State<TestWidget> {
  T? param;
  Widget childWidget = Container();

  void update(T value) {
    setState(() {
      param = value;
    });
  }

  @override
  Widget build(BuildContext context) {
    return Container(
      child: widget.childBuilder(param),
    );
  }
}

/// 实例A
class ATestWidget extends TestWidget<A> {
  const ATestWidget({Key? key}) : super(key: key);

  @override
  Widget childBuilder(A param) {
    return AWidget(a: param);
  }
}

/// 实例B
class BTestWidget extends TestWidget<B> {
  const BTestWidget({Key? key}) : super(key: key);

  @override
  Widget childBuilder(B param) {
    return BWidget(b: param);
  }
}
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 最近为项目组整理相关项目相关的规范文档,并整理了项目中,代码中的一些问题,识别项目代码中的一些可以优化的代码...
    Android开发_Hua阅读 4,212评论 0 1
  • 关于重构 为什么要重构 代码重构漫画 项目在不断演进过程中,代码不停地在堆砌。如果没有人为代码的质量负责,代码总是...
    陈二狗想吃肉阅读 5,005评论 0 7
  • 测试代码质量的唯一方式:别人看你代码时说 f * k 的次数。 代码质量与其整洁度成正比。干净的代码,既在质量上较...
    零星小雨_c84a阅读 1,819评论 0 0
  • 变量 函数 对象和数据结构 类 SOLID 测试 异步 错误处理 代码风格 注释 1. 用有意义且常用的单词命名变...
    16manman阅读 6,228评论 1 1
  • Clean Code PHP github地址 目录 介绍 变量使用见字知意的变量名同一个实体要用相同的变量名使用...
    code_nerd阅读 3,297评论 0 3

友情链接更多精彩内容