Swift 轻量级网络层设计
前言
普遍我们的网络层设计的时候直接是如下结构APIManager.post(url, parameter,completeHandle),服务器配置在APIManager.m文件中进行配置。这样一个简单便捷网络请求类便写好了,但细心思考我们会发现如下一些问题:
相同API可能分散各处导致每次需要填写的参数key值。回调处理代码也可能会存在冗余。
//例如登录功能 APIManager.post("登录APIUrl",{"name":"","pwd":""},completeHandle)
//登录功能首先必然存在于LoginVc中
LoginVC ==> 登录功能
//此时要求实现自动登录功能,那么在主控制器生成时应该判断是否可以自动登录(当然也可以推出LoginVc后再去自动登录)
MainVc ==> 自动登录(登录功能)
此时便需要在两处来写入url、parameter及回调方法,当此接口有更新时就需要在两处进行修改,若工程中用到的此网络功能越多则需要修改的地方也越多!
部分网络请求需要随着页面的pop而取消,手动进行管理task显得笨拙而麻烦。
关于网络请求起飞后回调没有着陆点是很危险的一件事,因此正常的做法是让APIManager.post返回task任务,并有当前请求发起类持有,并在当前请求类析构并且task任务未完成时取消。这样的操作方式一看就繁琐,若一个大工程处处这样维护怕是相当麻烦了!
因此我们便需要设计出一个能解决上述问题的网络层设计
设计模式
此种设计主要是请求类持有自定义Request,并向中间件BQAPIManager传入Request调用URLSession。此过程中Request会持有task任务,当请求类被释放时,Request也会释放,此时判断其是否有任务执行如有任务执行则停止执行。从而达到自动取消任务的功能。而此时针对相同的url及其参数便可封装与一个Request中,可有效的减少冗余代码!
关键类说明
BQRequest.Swift
主要在于封装参数key值和设定url并处理网络请求类容及持有并按需取消或释放task任务
//关键代码如下
var method: HTTPMethod = .post
var result: Any?//如成功处理网络请求内容后result有值
weak var task: URLSessionTask?
public func url() -> String { return "" }//API接口
//模型转字典
public func toDiction() -> [String: Any] {
let mir = Mirror(reflecting: self)
var dict = [String: AnyObject]()
for p in mir.children {
if p.label == "method" || p.label == "result" || p.label == "task"{
continue
}
dict[p.label!] = (p.value as AnyObject)
}
return dict
}
//请求内容处理
public func responseAction(data: Data?, response: URLResponse?, error: Error?) {
if let data = data {
do {
self.result = try JSONSerialization.jsonObject(with: data, options: .allowFragments)
} catch let err as NSError {
print(err.localizedDescription)
}
}
if let error = error {
print(error.localizedDescription)
}
}
//保证析构时(请求类不存在时)取消正在执行的任务
deinit {
if let task = self.task{
if task.state == .running || task.state == .suspended {
print("cancel \(task)")
task.cancel()
}
self.task = nil
}
}
BQAPIManager.Swift
主要作用为配置服务器环境(方便本地、测试服、正式服的切换),并通过传入的Request发起网络请求。
其中主要的关键点在与避免Request与task的循环持有。
//关键代码如下
public class func sendRequest(request: BQRequest, completionHandler:@escaping () -> ()) {
weak var req = request
let task = BQNetWork.sendRequest(urlstr: baseUrl + request.url(), parameter: request.toDiction(), method: request.method, time: 10, headers: nil) { (data, response, error) in
//对返回内容进行处理,如果处理成功结果赋予Request.result之中
req?.responseAction(data: data, response: response, error: error)
DispatchQueue.main.async {
if req?.task != nil {
completionHandler()
//任务完成后释放task任务
req?.result = nil//此处最好让网络调用类来决定是否置空result
req?.task = nil
}
}
}
request.task = task
}
其中BQNetWork中的内容是基于URLSession的一个简单封装,这里便不再详述
使用方式
以上述登录功能为示例,此时需设计LoginRequest继承与BQRequest并重写其url 及 responseAction方法
public name: String?
public pwd: String?
override public func url() -> String { return "登录APIUrl" }
override public func responseAction(data: Data?, response: URLResponse?, error: Error?) {
super.responseAction(data: data, response: response, error:error);
if let result = self.result {
//登录处理通用功能实现
}
}
接着让LoginVc持有LoginRequest对name及pwd赋值(减少因参数key字段出错而失败的情况),并通过BQAPIManager发起请求即可。
后记
这种设计方式主要是通过对Moya的模仿变化而来,其做到了最简单、方便的管理。当然在实际运用中可能也会出现问题,如发现问题或有何不妥之处望指正!谢谢!最后附上笔者自己的Swift工具库
Swift 轻量级网络层设计的更多相关文章
- 【转】Swift 语言的设计错误
Swift 语言的设计错误 在『编程的智慧』一文中,我分析和肯定了 Swift 语言的 optional type 设计,但这并不等于 Swift 语言的整体设计是完美没有问题的.其实 Swift 1 ...
- iOS网络层设计感想
App的开发无外乎从网络端获取数据显示在屏幕上,数据做些缓存或者持久化,所以网络层极为重要.原来只是把AFNetwork二次封装了一下,使得调用变得很简单,并没有深层次的考虑一些问题. 前言 参考: ...
- Swift 路由机制设计
设计模式 APP设计模式多种多样,从最初的MVC到MVVM,再到MVP,VIPER等.越来越多的设计模式被开发出来并得以应用,但不论我们用到哪种设计模式,只需要记住高内聚.低耦合那边是好的设计模式.在 ...
- Swift基础之设计折线坐标图
最近添加了折线视图的样式,所以在这里用Swift语言重新再使用设计一下 首先设置纵坐标的数值是:体重 //体重 let weightLabel = UILabel.init(frame: ...
- 轻量级RPC设计与实现第五版(最终版)
在最近一段时间里,通过搜集有关资料加上自己的理解,设计了一款轻量级RPC,起了一个名字lightWeightRPC.它拥有一个RPC常见的基本功能.主要功能和特点如下: 利用Spring实现依赖注入与 ...
- 轻量级RPC设计与实现第三版
在前两个版本中,每次发起请求一次就新建一个netty的channel连接,如果在高并发情况下就会造成资源的浪费,这时实现异步请求就十分重要,当有多个请求线程时,需要设计一个线程池来进行管理.除此之外, ...
- Kafka Broker源码:网络层设计
一.整体架构 1.1 核心逻辑 1个Acceptor线程+N个Processor线程(network.threads)+M个Request Handle线程(io threads) 多线程多React ...
- Swift - 让StoryBoard设计视图,程序运行时都使用横屏形式
1,运行时横屏 将项目属性“General”->“DeviceOritentation”的Portrait复选框去掉 2,storyboard设计视图横屏 在storyboard中,单击中间界面 ...
- 轻量级RPC设计与实现第四版
在本版本中引入了SPI机制,关于Java的SPI机制与Dubbo的SPI机制在以前的文章中介绍过. 传送门:Dubbo的SPI机制与JDK机制的不同及原理分析 因为设计的RPC框架是基于Spring的 ...
随机推荐
- Vulkan Tutorial 12 Fixed functions
操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Visual Studio 2017 早起的图形API在图形渲染管线的许多阶段提供了默认的状态.在Vulkan中,从vie ...
- 华为A199:近期不会再买华为的手机了
为了支持国货,也省点钱,买了个华为A199: 缺点: 没有google play market很复杂的刷机后才能装Gmail 不过也有亮点: 自带录音功能,老htc通过软件也只能录单方向的哦关机闹 ...
- PHP的SQL注入技术实现以及预防措施
SQL 攻击(SQL injection,台湾称作SQL资料隐码攻击),简称注入攻击,是发生于应用程序之数据库层的安全漏洞.简而言之,是在输入的字符串之中注入SQL指 令,在设计不良的程序当中忽略了检 ...
- Spring学习(1)----入门学习(附spring-framework下载地址)
(一)Spring是什么 Spring是一个开源框架,为了解决企业应用开发的复杂性而创建的,但现在已经不止应用于企业应用 是一个轻量级的控制反转(IOC)和面向切面(AOP)的容器框架- 从大小和开销 ...
- 第一次使用idea从SVN什么checkout项目,一直都是用的eclipse
IntelliJ IDEA 14 拉取SVN maven 多模块项目 部署tomcat 详细图解! 二话不说 进入主题 我们创建空项目实际上是项目空间 进入主界面 想用svn必须先启用它 选择Su ...
- Bottle源码阅读笔记(二):路由
前言 程序收到请求后,会根据URL来寻找相应的视图函数,随后由其生成页面发送回给客户端.其中,不同的URL对应着不同的视图函数,这就存在一个映射关系.而处理这个映射关系的功能就叫做路由.路由的实现分为 ...
- [codeforces631E]Product Sum
E. Product Sum time limit per test: 1 second memory limit per test: 256 megabytes input:standard inp ...
- Web压力测试软件webbench
官方网站:http://home.tiscali.cz/~cz210552/webbench.html下载地址:http://home.tiscali.cz/~cz210552/distfiles/w ...
- js数组及数组应用(冒泡和二分,遍历输出)
一.定义:1)var arr=new Array(); 加数据:arr[0]=1; 2)定义同时赋值:var arr=new Array(1,2,3,4,5); 3)调用:var arr=new Ar ...
- canvas学习总结三:绘制虚线
上一章节我们说到,线性路径的绘制,主要利用movoTo(),lineTo()等方法,当然 Canvas 2D API 也提供了虚线的绘制方法,CanvasRenderingContext2D.setL ...