UIAlertView是iOS开发过程中最常用的控件之一,是提醒用户做出选择最主要的工具.在iOS8及后来的系统中,苹果更推荐使用UIAlertController来代替UIAlertView.所以本文也并不提倡开发者再使用UIAlertView,
本文的目的是探讨如何将原来的给变量赋值和通过Delete来回调的方式变成链式编程风格和通过Block来回调.通过学习对UIAlertView的改造让各位iOS开发者能够学会这种更加便捷的开发方式

 

什么是链式编程

对于有一定开发经验的开发者来说,链式编程并不陌生.有很多知名的开源库使用了这种编程风格,比如大名鼎鼎的JQuery,而iOS库中,Masory和Snapkit也是典型的使用链式编程的例子.
大体来说,链式编程就是将多个操作(多行代码)通过某种操作符(通常是点号.)链接成一句的代码.便代码更加紧凑,可读性也更好,降低了代码的重复度.比如以下代码:

 //使用普通的赋值模式
txtUserName.placeHolder = "请输入用户名"
txtUserName.font = UIFont.systemFont()
txtUserName.textColor = UIColor.GrayColor()
//使用链式编程的语句
txtUserName.setPlaceHolder("请输入用户名").setFont().setTextColor(UIColor.GrayColor())

通过这个例子读者应该可以更清楚的了解什么是链式编程风格.简单来说,链式编程风格要有以下特点

  • 通常是都是调用一个函数来给属性赋值.也就是说,该函数封装了赋值的语句,还可以在里面加入自己的判断和一些逻辑.
  • 该函数必须有一个返回值,通常是它本身.也可以是处理后的数据或者对象.
  • 可以用静态函数作为起始函数,但是后面的全是实例函数.
  • 可以设定一个最终函数,该函数不返回任何对象,用它来完全最后所有操作.

链式编程的利弊

使用链式编程最主要的好处是可以使代码更简洁,写起来一种"爽快"感.设计优秀的链式编程可以大大降低重复的代码,增强逻辑感.不足之处就是对开
发者的业务逻辑能力要求较高,同时因为链式编程都是调用函数,所以有可能会造成过深的函数调用栈.稍微影响性能.

使用Block来回调UIAlertView的Delegate

iOS开发中有很多回调都是使用Delegate完成的,相信各位读者已经写过很多次了.相对来说,使用Delegate比较繁琐,有时还需要在Delegate里
判断是哪个对象的Delegate,优点是运行效率较高.而Block则相反,写起来更直观,开发效率更高.苹果也是逐步使用Block来代替Delegate,iOS
8最新的UIViewController里的Action已经全部使用Block来实现.而且Swift里的闭包,匿名函数更上比比皆是.所以大胆地使用Block来代替
Delegate和Target-Action吧,苹果会帮我们处理好各种性能问题的.但是需要注意的是retian-circle问题,初识Block很容易出现这种问题.

使用链式编程和Block来改造UIAlertView

目前有两种方式可以实现将UIAlertView的Delegate改成Block的回调,一是使用运行时给UIAlertView加上Block属性.二是再写一个新的类继承
UIAlertView,本文使用的是第二种方式,写起来更简单一点.

先定义一个新的类来继承UIAlertView,并且实现UIAlertViewDelegate协议
 class BlockAlert:UIAlertView,UIAlertViewDelegate{
var completion:((buttonIndex:Int,alert:UIAlertView)->Void)? //定义各个Block用来回调,其参数名和Delegate回调的函数参数名一至
var willDismissBlock:((buttonIndex:Int,alert:UIAlertView)->Void)?
var didDismissBlock:((buttonIndex:Int,alert:UIAlertView)->Void)?
var didPresentBlock:((alert:UIAlertView)->Void)?
var willPresentBlock:((alert:UIAlertView)->Void)?
var alertWithCancelBlock:((alert:UIAlertView)->Void)?
override init(frame: CGRect) {
super.init(frame: frame)
self.delegate = self //将delegate设为自己
}
func alertView(alertView: UIAlertView, clickedButtonAtIndex buttonIndex: Int) {
if let block = completion{ //最主要的Block,当用户点了按钮时回调,先要判断这个Block存在不?
block(buttonIndex: buttonIndex, alert: alertView)
}
}
func alertView(alertView: UIAlertView, didDismissWithButtonIndex buttonIndex: Int) {
if let block = didDismissBlock{//其他根据相应的Delegate回调函数依次调用各个Block
block(buttonIndex: buttonIndex, alert: alertView)
}
}
func alertView(alertView: UIAlertView, willDismissWithButtonIndex buttonIndex: Int) {
if let block = willDismissBlock{
block(buttonIndex: buttonIndex, alert: alertView)
}
}
// 其他几个委托方法也省略了,原理都是一样的
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
}

参考上面的代码,总体思路是在构造器里面将委托设为自己,再将Delegate里面所有的函数用Block来实现,Block里面的参数和委托回调的函数参数一至.当有委托函数回调时,先判断这个
Block是不是nil,如果不是,则执行该Block
接下来的操作就很容易了

再直接扩展UIAlertView,加上自己想要的链式编程函数
 extension UIAlertView {
static func setMessage(msg:String)->UIAlertView{//在这里设定这个为静态函数,在里面实例化一个BlockAlert对象,再返回该对象
let alert = BlockAlert()
alert.message = msg
return alert
} func addAlertStyle(style:UIAlertViewStyle)->UIAlertView{ //直接在各个函数中封装一些操作.再返回自身
self.alertViewStyle = style
return self
} func addTitle(title:String)->UIAlertView{
self.title = title
return self
} func addFirstButton(btnTitle:String)->UIAlertView{
self.addButtonWithTitle(btnTitle)
return self
} func addSecondButton(btnTitle:String)->UIAlertView{
self.addButtonWithTitle(btnTitle)
return self
} func addButtons(btnTitles:[String])->UIAlertView{
for title in btnTitles{
self.addButtonWithTitle(title)
}
return self
} func addButtonClickEvent(clickButton:((buttonIndex:Int,alert:UIAlertView)->Void)?)->UIAlertView{
if let alert = self as? BlockAlert{//对于block,先要判断它是不是BlockAlert,如果是的话才添加该block,后面都是这种操作
alert.completion = clickButton
}
return self
} //这里也是省略了部分函数,因为原理都是一样的
func addAlertCancelEvent(event:((alert:UIAlertView)->Void)?)->UIAlertView{
if let alert = self as? BlockAlert{
alert.alertWithCancelBlock = event
}
return self
} func alertWithButtonClick(clickButton:((buttonIndex:Int,alert:UIAlertView)->Void)?){
if let alert = self as? BlockAlert{ //这个为终结函数,没有返回值,在里面直接调用了show()方法.
alert.completion = clickButton
alert.show()
}
}
}

在这里面我选择了用UIAlverView来扩展,而不是BlockAlert,这样可能会导致开发过程中踩进陷阱.因为在首个静态函数返回的对象是一个BlockAction对象,而不是UIAlertView对象
在后面添加Block时先要判断本身是不是BlockAction对象,如果是的话再将设定Block,所以在开发过程中要小心这个陷阱.当然读者也可以直接扩展BlockAlert,这样会更安全一些.

最后直接使用静态函数加链式编程来弹出Alert
UIAlertView.setMessage("这里要设置信息").addTitle("我是标题").addFirstButton("取消").addSecondButton("确认").alertWithButtonClick { (buttonIndex, alert) -> Void in
print("你按下了第\(buttonIndex)个按键")
//你按下了第1个按键
//你按下了第0个按键
}

如果要使用其他的委托方法也很简单

    UIAlertView.setMessage("这里要设置信息").addTitle("我是标题").addFirstButton("取消").addSecondButton("确认").addWillPresentEvent { (alert) in
print("the alertView will present")
}.addDidPresentEvent { (alert) in
print("the alertView did present")
}.alertWithButtonClick { (buttonIndex, alert) in
print("你按下了第\(buttonIndex)个按键")
}

当然其他的一些自定义样式,或者是带用户名输入和密码输入框的也很简单

  UIAlertView.setMessage("这里要设置信息").addTitle("我是标题").addFirstButton("取消").addSecondButton("确认").addAlertStyle(.LoginAndPasswordInput).addDidPresentEvent { (alert) in
print("the alertView did present")
}.alertWithButtonClick { (buttonIndex, alert) in
let txt1 = alert.textFieldAtIndex()?.text
let txt2 = alert.textFieldAtIndex()?.text
print("userName:\(txt1) password:\(txt2)")
//打印出the alertView did present
//userName:Optional("111") password:Optional("222")
}

带用户名密码的Alert

可见,使用链式编程加Block的方式基本只需要短短几行代码就可以满足大部分使用UIAlertView的使用场景,写起来很方便快捷.对于UIActionSheet也一样,可以将其完全改造成使用链式编程加Block的形式来使用.这个便留给读者
来完成了.相信看完上面的代码并实践过的读者很快就可以写出来.以上所有代码都可以在我的Github看到DuckDeck-GrandCue
但是目前Apple还是更推荐开发者使用最新的UIAlertController,它的API设计和UIAlertView是完全不同的.改成了基于Block的回调方式,使用起来更方便了.在这里这篇文章给大家详细地介绍了
UIAlertController详解,有兴趣的读者可以去看看

总结

本文用一个足够简单的例子和大家探讨了怎么将原先使用UIAlertView编程方式改造成基于链式编程和block的编程方式.但是在实际的开发过程中,情况要比这个复杂得多,首先要考虑该业务和逻辑适不适合使用链式编程,其次设计出良好的
链式编程API也是需要花费很大精力的.函数返回值的设计也是很需要功力的,因为它直接影响了下一个调用函数.

iOS开发技巧系列---使用链式编程和Block来实现UIAlertView的更多相关文章

  1. iOS开发技巧系列---详解KVC(我告诉你KVC的一切)

    KVC(Key-value coding)键值编码,单看这个名字可能不太好理解.其实翻译一下就很简单了,就是指iOS的开发中,可以允许开发者通过Key名直接访问对象的属性,或者给对象的属性赋值.而不需 ...

  2. iOS端JSON转Model链式编程框架SuperKVC使用方法与原理

    背景 在client编程中.字典转模型是一个极为常见的问题,苹果提供了KVC来实现NSDictionary到Model的注入,可是KVC仅仅能进行单层浅注入.且无法处理类型转换.key与属性名不正确应 ...

  3. iOS 开发之使用链式编程思想实现简单的计算器

    链式编程思想是将多个操作(多行代码)通过点号(.)链接在一起成为一句代码,使代码可读性好.例如 a(1).b(2).c(3). 链式编程思想最为关键的是,方法的返回值是block,block必须返回对 ...

  4. iOS利用block实现链式编程方法(Objective-C链式编程)

    objc利用block实现链式编程方法 因为不好读.block和其他语言的匿名函数一样,很多程序员刚开始很难主动去用他. 本文描述block作为属性的实际使用,看懂block,并讲解如何利用block ...

  5. ios的链式编程笔记

    1.Masonry的语法为啥能连续的使用点语法? >> 底层使用的是:用block当函数的返回参数  > 链式编程思想 2. 使用block当函数的返回参数 // 之前开发中比较习惯 ...

  6. iOS 链式编程探索(Masonry)

    看了几篇关于链式编程的文章,还是理解的不透彻,我想这可能是因为我自己对block掌握的不熟练. 我已经明白了,所以,和大家分享一下我的理解!如有问题,麻烦大家指出! 直接看代码吧!关键的注释都有. 我 ...

  7. JavaScript系列:模块化与链式编程

    模块化:闭包和和函数作用域(JS没有块级作用域ES6之前)构造模块 var man=function(){ var age=12; return { getYear:function(){ retur ...

  8. iOS:用Block写一个链式编程

    一.介绍 链式编程是一个比较新颖的编程方式,简单直观,用起来也比较舒服.目前比较有名的Mansory和BabyBlueTooth就是使用链式编程写的第三方框架. 二.写法 链式编程写法不同于传统方式, ...

  9. 实现iOS中的链式编程

    谈到链式编程,那Masonry几乎就是最经典的代表.如: make.top.equalTo(self.view).offset() 像这样top.equalTo(self.view).offset(6 ...

随机推荐

  1. SGU 134 Centroid

    题意:给出一个树,每个点有一个value,value的意义是去掉这个点之后所有连通分量中点最多的那个连通分量的点数,这棵树的重心为所有点value的最小值,求重心,及重心都有谁. 解法:貌似是个树形d ...

  2. HDU 1394-Minimum Inversion Number(BIT)

    题意: 给你n个数字的序列 每次把第一个数字放到最后 得到一个新序列 一共有n个序列求这些序列中哪个序列含最小的总的逆序数 (输出最小总逆序数) 分析: 用BIT求出初始各数的逆序数,第一个数放最后它 ...

  3. Doing Homework(HDU 1074状压dp)

    题意:给你n个要做的作业,它们的名字.期限.可完成所需天数(必须连续)在规定期限不能完成要扣分(每天一分)求做作业顺序使扣分最少. 分析:作业数量较少,用状态压缩,做到第i种作业花费的天数dp[i]. ...

  4. 软件测试模型汇总-V模型,W模型,X模型,H模型

    V模型 在软件测试方面,V模型是最广为人知的模型,尽管很多富有实际经验的测试人员还是不太熟悉V模型,或者其它的模型.V模型已存在了很长时间,和瀑布开发模型有着一些共同的特性,由此也和瀑布模型一样地受到 ...

  5. 【Java基础】Java网络编程基础知识

    什么是网络编程 网络编程是通过使用套接字来达到进程间通信目的,那什么是套接字呢?其实套接字是支持TCP/IP的网络通信的基本操作单元,可以看做是不同主机之间的进程进行双向通信的端点,简单的说就是通信的 ...

  6. leetcode@ [79/140] Trie树应用 Word Search / Word Search II

    https://leetcode.com/problems/word-search/ class Solution { public: struct Trie{ Trie *next[]; bool ...

  7. 50个Java多线程面试题

    不管你是新程序员还是老手,你一定在面试中遇到过有关线程的问题.Java 语言一个重要的特点就是内置了对并发的支持,让 Java 大受企业和程序员的欢迎.大多数待遇丰厚的 Java 开发职位都要求开发者 ...

  8. python 输出字符串如果遇到字典

    >>> d = {'a':1} >>> print '%s' % 1,d 1 {'a': 1} >>> print '%s %s' % 1,d T ...

  9. Linux web工程部署远程必备软件安装

    一.序 最近在将程序往linux上面部署,特此记录下部署步骤,待以后参考. web工程部署必备软件为:JDK.tomcat.数据库软件(oracle或mysql),远程监控.上传下载必备软件:VNC. ...

  10. 弹出框JBox实例

    前几天做的考试系统的一些后台弹出框的一些模板.主要是因为普通的弹出框样式不是很好,颜色也不能调换.这里我们用的是JBox,还是从师傅那得知的.自己小实验了下,这里就做个小结. JBox 插件说明 - ...