iOS开发-关于自定义控件很值得一看的文章( 四)

2023-05-16

2019独角兽企业重金招聘Python工程师标准>>> hot3.png

我么知道 Objective-C 是采用 消息机制 (messaging)调用方法的,例如我们调用 UIView 的 init 方法

UIView * simpleView = [[UIView alloc] init];

简单的描述一下其中的过程:

  1. 程序一运行,所有的类方法(‘+’开头)和实例方法(‘-’开头)的接口内存地址都被写入一张hash表中

  2. 我们向 UIView 发送类方法 alloc 消息,runtime(运行时环境)根据前面说的hash表,查找对应类(UIView)的对应类方法(alloc)的内存地址,然后调用

  3. 如果UIView并未实现alloc方法,runtime会转而查找UIView的父类是否实现了alloc方法,如果实现了,就将消息投递给父类的alloc方法;如果没有实现,转而查找UIView父类的父类是否实现,重复这一过程直到将消息投递出去

  4. 如果最终发现投递不出去,则会抛出一个最常见的异常 unrecognized selector sent to instance + 内存地址 ,也就是你调用了一个没有实现的方法

不过,我们今天遇到的问题单单依靠 消息机制 并不能很好的解决。

需求我们需要将 Demo 中 XXXSegmentView 获取的触摸事件,反馈给当前的UIViewContoller,应该怎么做?

1. 直接调用

我们从最蠢的做法说起,虽然是蠢,但是是可行的,不过不要模仿啊,单纯为了讲原理和作对照

@interface ViewController ()@property (strong, nonatomic) XXXSegmentView *segmentView;
- (void)segmentDidSelectIdx:(NSInteger)idx;@end

@interface XXXSegmentView : UIView@property (weak, nonatomic) ViewController *viewController;@end

我们在给 XXXSegmentView 加上一个 viewController 属性,然后就可以在获取触摸事件时,通过调用 ViewController 的 segmentDidSelectIdx 方法,传递选择标签这个事件。

这样是可行的,但是缺点十分明显:耦合性太高, XXXSegmentView 需要引用ViewController 头文件,不符合低耦合这个基本原则。

2. delegate(委托)模式

objc 的delegate设计模式,可以解决上面的问题。但根据 objc 的设计初衷,这个问题用delegate解决真的有种杀鸡用牛刀的感觉。

@interface XXXSegmentView : UIView@property (nonatomic, weak) id delegate;@end

这里的delegate属性,是一个 id 类型的属性, id 这个类型就是 objc 的动态类型,编译器不关心它是什么类型,所以 id 类型的对象,可以调用所有声明过的类方法和实例方法,而编译器不会报错。

这样我们就可以个把 viewController 作为 XXXSegmentView 的 delegate属性传入, XXXSegmentView 无需知道自己的 delegate 是什么类,便可以直接调用 delegate 的实例方法。

if (self.delegate && [self.delegate respondsToSelector:@selector(segmentDidSelectIdx:)]) {
        [self.delegate segmentDidSelectIdx:idx];
}

注意我们在调用之前,首先检查 self 的 delegate 是否赋值,然后通过respondsToSelector 确认 delegate 实现了 segmentDidSelectIdx 方法,最后才传递消息。这两步十分重要, delegate 作为动态类型,编译器编译阶段是无法发现问题的,所以运行时要进行确认。

注:标准的委托模式是要结合协议(Protocol)一起使用的,这里就不多讲了。

3. Target模式

这要从一个类说起,他叫 UIControl 。

UIControl 是 UIView 的子类,是 UIKit 框架中可交互的控件的基类,一般不直接使用。我们用的比较多的例如 UIButton 、 UISwitch 、 UITextField 等都是他的子类。

UIControl 为 iOS 的人机交互制定了一系列的标准:

例如最常见的 UIControlEvents 枚举,定义了 iOS 交互中的交互方式

typedef NS_OPTIONS(NSUInteger, UIControlEvents) {
    UIControlEventTouchDown                                         = 1 <<  0,      // on all touch downs
    UIControlEventTouchDownRepeat                                   = 1 <<  1,      // on multiple touchdowns (tap count > 1)
    UIControlEventTouchDragInside                                   = 1 <<  2,    UIControlEventTouchDragOutside                                  = 1 <<  3,    UIControlEventTouchDragEnter                                    = 1 <<  4,    UIControlEventTouchDragExit                                     = 1 <<  5,    UIControlEventTouchUpInside                                     = 1 <<  6,    UIControlEventTouchUpOutside                                    = 1 <<  7,    UIControlEventTouchCancel                                       = 1 <<  8,    UIControlEventValueChanged                                      = 1 << 12,     // sliders, etc.
    UIControlEventPrimaryActionTriggered NS_ENUM_AVAILABLE_IOS(9_0) = 1 << 13,     // semantic action: for buttons, etc.

    UIControlEventEditingDidBegin                                   = 1 << 16,     // UITextField
    UIControlEventEditingChanged                                    = 1 << 17,    UIControlEventEditingDidEnd                                     = 1 << 18,    UIControlEventEditingDidEndOnExit                               = 1 << 19,     // 'return key' ending editing

    UIControlEventAllTouchEvents                                    = 0x00000FFF,  // for touch events
    UIControlEventAllEditingEvents                                  = 0x000F0000,  // for UITextField
    UIControlEventApplicationReserved                               = 0x0F000000,  // range available for application use
    UIControlEventSystemReserved                                    = 0xF0000000,  // range reserved for internal framework use
    UIControlEventAllEvents                                         = 0xFFFFFFFF};

又例如 UIControlState 定义了控件的基本状态

typedef NS_OPTIONS(NSUInteger, UIControlState) {
    UIControlStateNormal       = 0,
    UIControlStateHighlighted  = 1 << 0,                  // used when UIControl isHighlighted is set
    UIControlStateDisabled     = 1 << 1,    UIControlStateSelected     = 1 << 2,                  // flag usable by app (see below)    UIControlStateFocused NS_ENUM_AVAILABLE_IOS(9_0) = 1 << 3, // Applicable only when the screen supports focus
    UIControlStateApplication  = 0x00FF0000,              // additional flags available for application use
    UIControlStateReserved     = 0xFF000000               // flags reserved for internal framework use};

同时提供了给控件反馈交互操作的一系列方法,例如我们今天要讲的

- (void)addTarget:(nullable id)target action:(SEL)action forControlEvents:(UIControlEvents)controlEvents;

比如我们有一个按钮,当他点击时候,我们执行ViewContollr的 -(void)click:(id)sender 方法,可以这么写:

UIButton* button = [UIButton buttonWithType:UIButtonTypeSystem];
[button addTarget:self action:@selector(click:) forControlEvents:UIControlEventTouchUpInside];

这里传入的 UIControlEventTouchUpInside 枚举量,就是在控件frame内按下,然后抬起这样一个事件, UIContol 将这个事件作为key,和目标(target)和目标方法(action)存到了自己私有的字典里。当用户点击按钮时, UIControl 响应了触摸链的 touchesEnded 方法,便会根据私有字典,把对应UIControlEventTouchUpInside 的目标(target)和目标方法(action)调用,这样完成事件的回传。

这是一个很好的设计,从原则上讲,我们的 XXXSegmentView 是一个可交互控件,理应继承于 UIControl 而非 UIView ,但笔者偷懒了,读者有兴趣可以自己尝试改写。

4. block(块语法)

没有继承 UIControl ,笔者只好祭出终极大杀器, block 。block语法特性加入iOS已经有段日子了,因为使用方法篇幅太大,这里就不细说了,推荐一篇相关 教程 。

我们知道block是可以当作对象看待的,所以给 XXXSegmentView 添加下面这个属性

@property (nonatomic, strong) void (^ didSelectBlock)(NSUInteger idx);

在 ViewContoller 中,我们给 XXXSegmentView 的 didSelectBlock 赋值

@property (weak, nonatomic) IBOutlet XXXSegmentView *segment;

[segment setDidSelectBlock:^(NSUInteger idx) {
        NSLog(@"segment select %@",@(idx));
}];

然后在 XXXSegmentView 中加入 block 调用

- (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event{
    [super touchesEnded:touches withEvent:event];    
    //.....其他代码
    
    if (self.didSelectBlock) {
        self.didSelectBlock(touchNumber);
    }
}

block的调用方法类似C语言的方法调用,传参格式也相同,注意使用前也要进行非空检测哦。


转载于:https://my.oschina.net/caijunrong/blog/643726

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

iOS开发-关于自定义控件很值得一看的文章( 四) 的相关文章

随机推荐

  • 树莓派进阶之路 (031) -字符问题(1) - GBK汉字编码表(转)

    转载 xff1a http blog sina com cn s blog 8184e033010109ug html 基本简介 GB码 xff0c 全称是GB2312 80 信息交换用汉字编码字符集基本集 xff0c 1980年发布 xf
  • 外贸盒子x96max+ 搭建Armbian系统 (附国内可用的img镜像下载地址)

    记录一下在x96max 43 外贸盒子上面搭建Armbian的历程 Armbian是什么操作过程 救砖历程 xff1a 国内外相关电视盒子论坛 Armbian是什么 Armbian是轻量级的Debian系统 xff0c 为ARM开发板专门发
  • 手机上输入http://192.168.1.102:8888/FiddlerRoot.cer为什么下载不了证书

    因为之前你的手机可能已经安装了该证书 xff0c 所以再次下载会说找不到证书 解决办法 如果你遇到上面的问题 就可能是证书的问题 我的本地证书是用系统生成证书的一个软件生成的个人证书 所以出现了问题 操作步骤如下 1 将之前的证书删除打开系
  • 区块链上编程:DApp 开发实战——来写个竞猜游戏吧!

    本文旨在引导对 DApp 开发感兴趣的开发者 xff0c 构建一个基于以太坊去中心化应用 xff0c 通过开发一款功能完备的竞猜游戏 xff0c 迈出 DApp 开发的第一步 xff0c 通过实例讲解 Solidity 语言的常用语法 xf
  • pip更改下载源设置

    任何一个信息都可以把人分为两类 xff0c 知道的和不知道的 有些我们已知的信息 xff0c 自己觉得很平常 xff0c 却可能对另一些不知道的人来说很有用处 比如今天要说的这个小技巧 xff0c 我自己原以为不值一提 xff0c 网上也很
  • 让div在屏幕中居中(水平居中+垂直居中)的几种方法

    水平居中方法 1 inline xff0c inline block元素的水平居中 xff0c 在父级块级元素中设置text align center 2 确定宽度的块级元素水平居中方法 margin xff1a 0 auto xff1b
  • 从ELK到EFK

    背景 作为中国最大的在线教育站点 xff0c 目前沪江日志服务的用户包含沪江网校 xff0c 交易 xff0c 金融 xff0c CCtalk xff08 直播平台 xff09 等多个部门的多个产品的日志搜索分析业务 xff0c 每日产生的
  • 证明:实对称矩阵中,属于不同特征值的特征向量相互正交

    2019独角兽企业重金招聘Python工程师标准 gt gt gt 证明 xff1a 实对称矩阵中 xff0c 属于不同特征值的特征向量相互正交 设AP 61 1P xff0c AP 61 2P xff0c 其中A为实对称矩阵 xff0c
  • 宝塔面板一键docker部署

    2019独角兽企业重金招聘Python工程师标准 gt gt gt 推荐 xff1a 宝塔面板一键docker部署 0 购买服务器 1 安装docker root 64 VM 0 2 centos curl fsSL https get d
  • 异步I/O Asynchronous I/O

    原帖地址 xff1a http www ibm com developerworks cn linux l async AIO 简介 Linux 异步 I O 是 Linux 内核中提供的一个相当新的增强 它是 2 6 版本内核的一个标准特
  • Trapping Rain Water II

    Given n x m non negative integers representing an elevation map 2d where the area of each cell is 1 x 1 compute how much
  • QT- label实现滚动字幕 (跑马灯)效果 -像素级

    QT label滚动字幕 原理代码完整源码 43 部件提升 想通过QSS实现更多效果的请搜索QT自带帮助文档 Qt Style Sheets Examples xff0c 使用样式表来定制程序界面 原理 让label字母滚动起来的原理很简单
  • matlab练习程序(TV模型图像修复)

    曾经想要实现过Bertalmio图像修复算法 xff0c 无奈自身实力不够 xff0c 耗费两天时间也没能实现 昨天博客上有人问到TV模型 xff0c 这个模型我过去是没听说过的 xff0c 于是就找来相关论文研究了一下 xff0c 发现T
  • M4——GPIO配置

    1 GPIO 简述 通用输入输出 General Purpose Input Output 的简称 xff0c 就是芯片引脚可以通过他们输出高电平或者低电平 xff0c 也可以通过他们读取引脚的电平状态 以STM32F407ZGT6芯片为例
  • linux 查看服务器序列号(S/N)

    root 64 node1 dmidecode t 查看支持的参数 dmidecode option requires an argument 39 t 39 Type number or keyword expected Valid ty
  • Apache 中文显示乱码的问题

    用Apache 2 2发布一个html网页 xff0c 访问时发现网页中的中文全部显示乱码 之前在IIS上发布显示很正常 xff0c 网页本身编码应该是没问题的 查看源代码确定html文档编码为GB2312 在 etc httpd conf
  • 对centos进行一些个性化设置(如alias等)

    为什么80 的码农都做不了架构师 xff1f gt gt gt 一 xff1a linux xff08 centos5 10 xff09 中alias的设置 amp 常用 查询 xff1a 查看当前系统的所有alias shell gt a
  • Linux编程之有限状态机FSM的理解与实现

    有限状态机 xff08 finite state machine xff09 简称FSM xff0c 表示有限个状态及在这些状态之间的转移和动作等行为的数学模型 xff0c 在计算机领域有着广泛的应用 FSM是一种逻辑单元内部的一种高效编程
  • 用Nohup命令让Linux下的程序真正在后台执行(转载)

    Unix Linux 下一般想让某个程序在后台运行 xff0c 很多都是使用 amp 在程序结尾来让程序自动运行 比如我们要运行 mysql 在后台 xff1a usr local mysql bin mysqld safe user 61
  • iOS开发-关于自定义控件很值得一看的文章( 四)

    2019独角兽企业重金招聘Python工程师标准 gt gt gt 我么知道 Objective xff0d C 是采用 消息机制 xff08 messaging xff09 调用方法的 xff0c 例如我们调用 UIView 的 init