一. 前言

  • 对于在 MVC 的定义中,view 层是不引用 model 层,view 和 model 是不相往来的

  • 一般开发中,我们都写过 在自定义 view 中增加一个 model 的属性,外接直接传个 model 来,在 view 中 model 的 set 方法里对 view 的控件赋值的代码,例如在自定义 UITableViewCell 时用的很多,此时 view 是直接引用了 model

  • 基于封装的思想,我们需要竟可能的复用我们的代码,复用我们的 view,这时我们需要进行解耦,不依赖于某个特殊的 model。另外,如果对于很特殊的 view,整个项目中没有什么重用的,可以按之前情况处理

  • 本文简要介绍自己常用的两种写法来解耦 view 和 model,使之更符合 MVC 的定义

二.定义 ViewModel 对象及 ViewModelType (协议) 的形式

  • 说明

    • 将 view 中所有控件需要的数据依次列出
    • 定义 ViewModelType 一种协议,定义协议方法 ,旨在通过协议方法,返回 view 需要的数据
    • view 中控件赋值,需要外接传入遵守 ViewModelType 协议的 ViewModel,在 ViewModel 的 set 方法里进行控件赋值
    • 该 ViewModel 的创建是依赖具体的 model 的,需要遵守 ViewModelType 协议
    • view 和 ViewModelType 为一整体,通过更换不同的 ViewModel 来达到重用。更换不同 ViewModel,也相当于更换了数据 model。
  • 示例
    • 如一个 cell (有大标题,几个子标题和标题对应的内容,状态,图片等)
      @interface LXReservationCell : UITableViewCell
      
      @property (nonatomic, weak) id<LXReservationCellViewModelType> viewModel;
      
      @end
    • LXReservationCellViewModelType 协议
      @protocol LXReservationCellViewModelType <NSObject>
      
      // 标题
      @property (nonatomic, copy, readonly) NSString *title;
      // 第一个子标题
      @property (nonatomic, copy, readonly) NSString *firstItemTitle;
      // 第一个子标题对应的内容
      @property (nonatomic, copy, readonly) NSString *firstItemContent; @end
    • LXReservationCellViewModel 中

      #import <Foundation/Foundation.h> @class LXReservationItem; @interface LXReservationCellViewModel : NSObject - (instancetype)initWithItem:(LXReservationItem *)item; @end
      @interface LXReservationCellViewModel () <LXReservationCellViewModelType>
      // 标题
      @property (nonatomic, copy) NSString *title;
      // 第一个子标题
      @property (nonatomic, copy) NSString *firstItemTitle;
      // 第一个子标题对应的内容
      @property (nonatomic, copy) NSString *firstItemContent;
      @end
      @implementation LXReservationCellViewModel - (instancetype)initWithItem:(LXReservationItem *)item {
      if (self = [super init]) {
      self.title = item.title;
      self.firstItemTitle = item.orderCategory;
      self.firstItemContent = item.orderdate; }
      return self; }

三、定义 view 对应的 config,使用链式编程的形式进行对 view 控件赋值

  • 说明

    • 将 view 中所有控件需要的数据依次列出
    • 定义一个 config 对象,使用链式编程的形式进行获取 view 需要的各种数据,在 config 中弱引用 view,把各种数据在赋值给 view
    • view 中需要定义一个配置数据的方法,方法的参数是个 block,block 传一个 config 对象给外界使用
    • view 和 config 为一整体,并不引用 model,因此脱离的 model 的限制,具有重用性
  • 示例
    • 如一个简单的展示标题、内容,还有一个按钮的 view
      #import <UIKit/UIKit.h>
      
      @class LXIntroduceViewConfig;
      
      @interface LXIntroduceView : UIView
      // 标题
      @property (nonatomic, copy) NSString *title;
      // 内容
      @property (nonatomic, copy) NSString *content;
      // 按钮标题
      @property (nonatomic, copy) NSString *btnTitle; // 配置 view 对应的数据的方法
      - (void)lx_makeWithConfig:(void(^)(LXIntroduceViewConfig *config))block;
      - (void)lx_makeWithConfig:(void (^)(LXIntroduceViewConfig *))block {
      LXIntroduceViewConfig *config = [[LXIntroduceViewConfig alloc]initWithIntroduceView:self];
      !block? : block(config);
      }
    • LXIntroduceViewConfig.h
      #import <Foundation/Foundation.h>
      
      @class LXIntroduceView;
      
      @interface LXIntroduceViewConfig : NSObject
      
      - (instancetype)initWithIntroduceView:(LXIntroduceView *)introduceView;
      // 设置标题
      - (LXIntroduceViewConfig *(^)(NSString *))setupTitle;
      // 设置要显示的内容
      - (LXIntroduceViewConfig *(^)(NSString *))setupContent;
      // 设置按钮的标题
      - (LXIntroduceViewConfig *(^)(NSString *))setupBtnTitle; @end
    • LXIntroduceViewConfig.m
      @interface LXIntroduceViewConfig ()
      // 弱引用 view
      @property (nonatomic, weak) LXIntroduceView *introduceView;
      @end @implementation LXIntroduceViewConfig
      // init
      - (instancetype)initWithIntroduceView:(LXIntroduceView *)introduceView {
      if (self = [super init]) {
      self.introduceView = introduceView;
      }
      return self;
      }
      // 把标题传给 view,view 中 set 方法接收
      - (LXIntroduceViewConfig *(^)(NSString *))setupTitle {
      return ^(NSString *tmp) {
      self.introduceView.title = tmp;
      return self;
      };
      }
      // 把内容传给 view,view 中 set 方法接收
      - (LXIntroduceViewConfig *(^)(NSString *))setupContent {
      return ^(NSString *tmp) {
      self.introduceView.content = tmp;
      return self;
      };
      }
      // 把按钮标题传给 view
      - (LXIntroduceViewConfig *(^)(NSString *))setupBtnTitle {
      return ^(NSString *tmp) {
      self.introduceView.btnTitle = tmp;
      return self;
      };
      }
      @end
    • 外界使用
      LXIntroduceView *introduceView = [[LXIntroduceView alloc]init];
      [introduceView lx_makeWithConfig:^(LXIntroduceViewConfig *config) {
      config
      .setupTitle(self.resultItem.title)
      .setupContent(self.resultItem.content)
      .setupBtnTitle(@"test");
      }];

四. 结尾

  • 一点一滴,仅此记录。

小记:iOS 中一般对于 view 不依赖 model 的的两种代码书写形式的更多相关文章

  1. vs中发布WebSever时启用HTTP-POST和HTTP-GET这两种协议

    一.问题介绍 在vs中建立一个websever项目时候默认是禁用HTTP-POST和HTTP-GET这两种协议的.但是如果你是在本机上去调试或者是在iis中浏览都会有HTTP-POST这种方式,因为这 ...

  2. Android代码中设置字体大小,字体颜色,显示两种颜色.倒计时效果

    Android代码中设置字体大小,字体颜色,显示两种颜色 在xml文件中字体大小用的像素 <TextView android:id="@+id/uppaid_time" an ...

  3. 【Java面试真题】剑指Offer53.2——0~n-1中缺失的数字(异或、二分两种解法)

    [Java实现]剑指Offer53.2--0~n-1中缺失的数字:面试真题,两种思路分享 前面有另一道面试题[Java实现]剑指offer53.1--在排序数组中查找数字(LeetCode34:在排序 ...

  4. WebGIS中解决使用Lucene进行兴趣点搜索排序的两种思路

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/. 1.背景 目前跟信息采集相关的一个项目提出了这样的一个需求:中国银行等 ...

  5. 关于实现自定义Dialog和实现Dialog里view的事件监听的两种方法

    一.自定义dialog. 二.实现dialog里view的事件监听 1.自定义dialog比较简单.在实例化new的时候,加入样式,布局就行了.或者重写dialog. 2.实现dialog里view的 ...

  6. 用Java集合中的Collections.sort方法对list排序的两种方法

    用Collections.sort方法对list排序有两种方法第一种是list中的对象实现Comparable接口,如下: ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 ...

  7. 【转】java中byte数组与int类型的转换(两种方式)----不错

    原文网址:http://blog.csdn.net/piaojun_pj/article/details/5903009 java中byte数组与int类型的转换,在网络编程中这个算法是最基本的算法, ...

  8. springMVC中对HTTP请求form data和request payload两种数据发送块的后台接收方式

    最近在做项目中发现,前台提交数据时,如果通过form表单提交和ajax发送json时,springMVC后台接收不能都通过@ModelAttribute方式处理,经过一番查找后,ajax发送json请 ...

  9. iOS同一项目多个Target的快速实现方法 - 两种使用场景详解

    我们项目中,默认建好是只有一个target的,但是,一些场景中,多target能帮助我们更好的使用项目. 场景1: 同一项目,一般会分不同环境:开发环境.测试环境.正式(生产)环境. 这就涉及到一个请 ...

随机推荐

  1. 盒模型的auto值

    浮动在盒模型的auto值 属性 常规流 浮动 margin-left:auto 尽量撑满包含块 0px margin-right:auto 尽量撑满包含块 0px margin-top:auto 0p ...

  2. fs-hasher工具介绍

    工具列表 hf-stat hf-simdex fs-hasher hf-stat工具 使用方法 以-f参数为例 Usage: ./hf-stat -f hashfile 参数用途 -f : 显示.ha ...

  3. Java Web之分页的实现(通用)

    一.用到的工具类的封装 为了实现代码的重用性,我们将经常用到的代码封装到工具类中,以便在任何地方都可以调用 1.获取路径工具 在jsp页面中,我们经常会向Servlet发送请求,并通过反射,实现通过传 ...

  4. 设计模式UML模型图

    1.抽象工厂(Abstract Factory)模式 意图:为特定的客户(或情况)提供特定系列的对象. 2.类的适配器(Adapter)模式 意图:将一个类的接口转换成客户希望的另外一个接口. 3.对 ...

  5. nodejs安装及使用步骤详解

    就一段小小的时间不用,就忘得差不多了,果然好记性不如乱笔头. 1.必须要安装node环境(建议装在C盘,这是一个系统盘)+安装mongoose数据库  +Robot 3T之于mongodb就相当于so ...

  6. javascript数组常用的遍历方法

    本篇文章给大家带来的内容是关于javascript数组常用的遍历方法(代码示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助. 前言 本文主要介绍数组常见遍历方法:forEach.m ...

  7. Django 01 django基本介绍及环境搭建

    Django 01 django基本介绍及环境搭建 #http服务器 #用来接收用户请求,并将请求转发给web应用框架进行处理 #Web应用框架 #处理完请求后在发送给http服务器,http服务器在 ...

  8. yii2 AR模型使用exists添加子查询与父查询关联

    有A,B两个表对应A_AR,B_AR两个模型B表interval_id对应A表id现在要查a表的数据,且没有code为a的子数据要求使用yii2的AR模型写查询: A_AR::find()->w ...

  9. python接口自动化(四十一)- 发xml格式参数的post请求(超详解)

    简介 最近在工作中,遇到一种奇葩的接口,它的参数数据是通过xml,进行传递的,不要大惊小怪的,林子大了什么鸟都有,每个人的思路想法不一样,开发的接口也是各式各样的,如果想要统一的话,必须是提前团队已经 ...

  10. phpcms文件结构

    主要目录部分 /admin 管理后台目录       -- /skin/    后台样式       -- /templates/    后台样式模板 /api api接口 /corpandresiz ...