草庐IT

iOS 链式编程

f8d1cf28626a 2023-03-28 原文

链式编程

主要介绍链式编程原理,以及如何创建链式编程

编程范式

  • 在介绍链式编程之前,首先来了解下什么是编程范式。
  • 编程范式是编程语言的一种分类,是指从事软件工程的一类典型的编程风格

常见的编程范式

常见的编程范式主要有以下几种

  • 面向过程编程(Process Oriented Programming,POP):属于典型的程序流程思想,即按照一定的顺序,按部就班的工作,特别适合解决线性的问题,其中过程化编程语言主要包含机器语言、C等支持过程化的语言

  • 面向对象编程(Object Oriented Programming,OOP):包含3个基本概念:封装、继承、多态。通过类、方法、对象和消息传递,其相关的语言包含Java、Objective-C等

  • 面向切面编程(AOP):是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行分离,降低业务间的耦合度,提升程序的可重用性。例如OC中的Method Swizzling、消息转发就是采用AOP的典型

  • 函数式编程(FP):是一种结构化编程,即如何编写程序的方法论。其核心思想就是将运算过程分解成一系列可复用函数的调用,其中函数是重中之重。也是比较火热且推崇的一种编程范式。

  • 响应式编程:简单理解为就是一点触发,多点响应,例如OC中的KVO、通知等,触发者只负责触发,不理会结果

  • 链式编程:运用点语言将很多函数串联起来,例如OC中的Masory和Swift中的Snapkit

POP、OOP、AOP优劣对比

POP 面向过程

优点

  • 流程化的编程,任务明确,即在开发前就已经明确了最终实现和最终效果

  • 开发效率高,代码短小精悍,适合结合数据结构来开发高效率的程序,例如算法等

  • 流程明确,具体步骤清晰,便于节点分析

缺点

  • 需要深入思考,耗费精力
  • 代码重用性地,基本是用于解决一种固定的问题,且不易扩展,维护难度大
  • 对于复杂业务,面向过程的模块化难度高,耦合度高

OOP 面向对象

优点

  • 结构清晰,不同类承担不同的职责

  • 封装性,将事务进行抽象,便于流程中的行为分析、操作

  • 易扩展,代码复用性高,可继承、覆盖

  • 实现简单,维护相对简单

缺点

  • 在面向过程的基础上高度抽象,和底层代码交互少,不适合底层开发和游戏开发

  • 对于事务而言,本身是面向过程的,过度的封装会导致事务本身的复杂性提高

AOP 面向切面

优点

  • 简单、易用、易扩展

  • 降低模块间的耦合度

  • 设计决定的迟邦定(即运行时绑定)

  • 提升代码的复用性

缺点

  • 增加额外的重复代码,且紧耦合

  • 每个业务逻辑都需要一个装饰器实现或代理

  • 使用麻烦,必须增加容器

综上所述,三者是一个相互补充和完善的逻辑

  • POP是以功能为中心来思考和组织程序的,注重功能的实现

  • OOP是以对象为中心,强调整体性,注重封装,代码整洁且规范

  • AOP是以业务解耦为中心,解决OOP中业务间高度耦合的问题

函数式编程

函数式编程是一种结构化编程,即如何编写程序的方法论。其核心思想就是将运算过程分解成一系列可复用函数的调用,其中函数是重中之重。也是比较火热且推崇的一种编程范式。

简单理解为就是函数和数据类型是一致的,也是可以作为函数的参数、返回值。例如OC、Swift中的map、filter、reduce函数等,每个函数的处理结果给到下一个函数,最后的结果由自身函数调出。

如下所示

计算: (1+2)*3/4
f1(a, b) = a + b
f2(c) = c * 3
f3(d) = d / 4

所以整个计算等价于
f(x) = f3( f2( f1(1, 2) ) )

对应到OC中,其核心点就是Block,如下所示

@interface Test: NSObject

- (Test *(^)(NSString *str))handle;

@end

@implementation Test

- (Test *(^)(NSString *str))handle{
    return ^(NSString *str){
        return self;
    };
}

@end

<!--调用-->
Test *t = [[Test alloc] init];
t.handle(@"1111").handle(@"22222");

链式编程

  • 链式编程是函数式编程的一种体现。

  • 链式编程的中心思想:方法的返回值必须是方法的调用者

  • 链式编程的核心语法:点语法 + Block

  • 链式编程的特点:使用点语法将对象的多个函数连起来调用

首先说点语法,在OC中,我们常应用于getter、setter方法,是一种特殊的语法糖,OC中是通过 [receiver message] 来调用方法的,所以getter、setter的点语法最终会调用对应属性的getter、setter方法

其次来说Block,在OC中,Block既是匿名函数,也是对象,具体的可参考iOS-Block底层原理

最后回到我们的焦点:链式编程,我们要如何实现呢?其实很简单,只需要在返回值作相应改动即可,如下所示

@interface Test: NSObject
- (Test *)a;
- (Test *)b;
- (Test *)c;
@end

<!--调用-->
Test *t = [[Test alloc] init];
t.a.b.c;

可是通过上面的例子发现,点语法是有了,确实连起来,但是并不能传参呀,此时就需要借助Block了,在OC中,常用的传值方式主要由代理、通知、Block等,其中满足点语法的就当属Block了。其次回想函数式编程,当返回值是带参block的getter方法时就实现了参数的传递。如下所示

<!--.h文件-->
@interface Test : NSObject

@property (nonatomic, strong) Test *(^block1)(NSString *name);
@property (nonatomic, strong) Test *(^block2)(NSInteger age);
@property (nonatomic, strong) Test *(^block3)(void);

- (Test *(^)(void))handleData;
@end

<!--.m文件-->
@interface Test ()

@property (nonatomic, strong) NSString *name;
@property (nonatomic, assign) NSInteger age;

@end

@implementation Test

- (Test * _Nonnull (^)(NSString * _Nonnull))block1{
    return ^(NSString *name){
        self.name = name;
        return self;
    };
}

- (Test * _Nonnull (^)(NSInteger))block2{
    return ^(NSInteger age){
        self.age = age;
        return self;
    };
}

- (Test * _Nonnull (^)(void))block3{
    return ^(void){
        return self;
    };
}


- (Test *(^)(void))handleData{
    
    return ^(void){
        NSLog(@"处理数据");
        return self;
    };
}
@end

<!--链式调用-->
Test *t = [[Test alloc] init];
t.block1(@"Tom").block2(3).block3().handleData();

所以思考实现链式编程,也是逐步递进的过程:方法调用 -> 方法通过点语法调用 -> 手写getter方法 -> 点语法调用属性 -> 实现点语法+block的链式编程

通过属性实现的链式编程,在getter方法中既完成了setter方法的赋值,也处理了逻辑关系,还能通过getter方法完成链式编程,正所谓一举三得嗷嗷嗷啊!

有关iOS 链式编程的更多相关文章

  1. ruby - 如何验证 IO.copy_stream 是否成功 - 2

    这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下

  2. ruby - 寻找通过阅读代码确定编程语言的ruby gem? - 2

    几个月前,我读了一篇关于ruby​​gem的博客文章,它可以通过阅读代码本身来确定编程语言。对于我的生活,我不记得博客或gem的名称。谷歌搜索“ruby编程语言猜测”及其变体也无济于事。有人碰巧知道相关gem的名称吗? 最佳答案 是这个吗:http://github.com/chrislo/sourceclassifier/tree/master 关于ruby-寻找通过阅读代码确定编程语言的rubygem?,我们在StackOverflow上找到一个类似的问题:

  3. Ruby 文件 IO 定界符? - 2

    我正在尝试解析一个文本文件,该文件每行包含可变数量的单词和数字,如下所示:foo4.500bar3.001.33foobar如何读取由空格而不是换行符分隔的文件?有什么方法可以设置File("file.txt").foreach方法以使用空格而不是换行符作为分隔符? 最佳答案 接受的答案将slurp文件,这可能是大文本文件的问题。更好的解决方案是IO.foreach.它是惯用的,将按字符流式传输文件:File.foreach(filename,""){|string|putsstring}包含“thisisanexample”结果的

  4. Get https://registry-1.docker.io/v2/: net/http: request canceled while waiting - 2

    1.错误信息:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:requestcanceledwhilewaitingforconnection(Client.Timeoutexceededwhileawaitingheaders)或者:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:TLShandshaketimeout2.报错原因:docker使用的镜像网址默认为国外,下载容易超时,需要修改成国内镜像地址(首先阿里

  5. 网络编程套接字 - 2

    网络编程套接字网络编程基础知识理解源`IP`地址和目的`IP`地址理解源MAC地址和目的MAC地址认识端口号理解端口号和进程ID理解源端口号和目的端口号认识`TCP`协议认识`UDP`协议网络字节序socket编程接口`sockaddr``UDP`网络程序服务器端代码逻辑:需要用到的接口服务器端代码`udp`客户端代码逻辑`udp`客户端代码`TCP`网络程序服务器代码逻辑多个版本服务器单进程版本多进程版本多线程版本线程池版本服务器端代码客户端代码逻辑客户端代码TCP协议通讯流程TCP协议的客户端/服务器程序流程三次握手(建立连接)数据传输四次挥手(断开连接)TCP和UDP对比网络编程基础知识

  6. ruby - 我正在学习编程并选择了 Ruby。我应该升级到 Ruby 1.9 吗? - 2

    我完全不是程序员,正在学习使用Ruby和Rails框架进行编程。我目前正在使用Ruby1.8.7和Rails3.0.3,但我想知道我是否应该升级到Ruby1.9,因为我真的没有任何升级的“遗留”成本。缺点是什么?我是否会遇到与普通gem的兼容性问题,或者甚至其他我不太了解甚至无法预料的问题? 最佳答案 你应该升级。不要坚持从1.8.7开始。如果您发现不支持1.9.2的gem,请避免使用它们(因为它们很可能不被维护)。如果您对gem是否兼容1.9.2有任何疑问,您可以在以下位置查看:http://www.railsplugins.or

  7. ruby - 为什么不能使用类IO的实例方法noecho? - 2

    print"Enteryourpassword:"pass=STDIN.noecho(&:gets)puts"Yourpasswordis#{pass}!"输出:Enteryourpassword:input.rb:2:in`':undefinedmethod`noecho'for#>(NoMethodError) 最佳答案 一开始require'io/console'后来的Ruby1.9.3 关于ruby-为什么不能使用类IO的实例方法noecho?,我们在StackOverflow上

  8. ruby - 如何以编程方式删除实例上的 "singleton information"以使其编码(marshal)? - 2

    我创建了一个由于“在运行时执行的单例元类定义”而无法编码的对象(这段代码的描述是否正确?)。这是通过以下代码执行的:#defineclassXthatmyusesingletonclassmetaprogrammingfeatures#throughcallofmethod:break_marshalling!classXdefbreak_marshalling!meta_class=class我该怎么做才能使对象编码正确?是否可以从对象instance_of_x的classX中“移除”单例组件?我真的需要一个建议,因为我们的一些对象需要通过Marshal.dump序列化机制进行缓存。

  9. Ruby 元编程问题 - 2

    我正在查看Ruby日志记录库Logging.logger方法并从sourceatgithub提出问题与这段代码有关:logger=::Logging::Logger.new(name)logger.add_appendersappenderlogger.additive=falseclass我知道类 最佳答案 这实际上删除了方法(当它实际被执行时)。这是确保close不会被调用两次的保障措施。看起来好像有嵌套的“class 关于Ruby元编程问题,我们在StackOverflow上找到一

  10. ruby - Paperclip:以编程方式分配图像并设置其名称 - 2

    使用Paperclip,我想从这样的URL抓取图像:require'open-uri'user.photo=open(url)问题是我最后得到一个像“open-uri20110915-4852-1o7k5uw”这样的文件名。有什么方法可以更改user.photo上的文件名?作为一个额外的变化,Paperclip将我的文件存储在S3上,所以如果我可以在初始分配中设置我想要的文件名就更好了,这样图像就会上传到正确的S3key。像这样:user.photo=open(url),:filename=>URI.parse(url).path 最佳答案

随机推荐