swift网络库Moya使用

1. 概述

Moya是对请求库Alamofire的抽象封装,相当于YTKNetwork和AFNetworking的关系

image.png

2. 基本使用

假设有如下api用于订单相关逻辑,使用Moya来实现

  • 请求地址:http://127.0.0.1:8080
  • 公共请求头:devtype:iOS,devid
  • 公共请求参数:token:"Gz1qYLXeBW8MZuUfDlr9wsAYuVS1cZFMJY9BbaF842L2gRps747o4w=="
API 参数 说明
order/list pageNO:订单列表开始页码,默认从1, pageSize:每页记录数 订单列表
order/findById sn:订单id 根据id查询订单

使用Moya接入API,它的设计风格就是利用枚举来实现,特别很好的利用枚举绑定值这个特性

// 生成请求封装类
let orderProvider = MoyaProvider<OrderApi>()

/// 订单相关api
enum OrderApi {
    case list(pageNO: Int = 1, pageSize: Int = 10)
    case findOne(sn: String)
}

/// 实现TargetType协议
extension OrderApi: TargetType {
    
    /// url
    var baseURL: URL {
        return URL(string: "http://127.0.0.1:8080/order")!
    }
    
    /// 请求路径
    var path: String {
        switch self {
        case .list:
            return "list"
        case .findOne(_):
            return "findById"
        }
    }
    
    /// 请求方式
    var method: Moya.Method {
        return .post
    }
    
    /// 解析格式
    var sampleData: Data {
        return "{}".data(using: String.Encoding.utf8)!
    }
    
    var task: Task {
        // 公共参数
        var params: [String: Any] = ["token": "Gz1qYLXeBW8MZuUfDlr9wsAYuVS1cZFMJY9BbaF842L2gRps747o4w=="]
        
        // 收集参数
        switch self {
        case let .list(pageNO, pageSize):
            params["pageNO"] = pageNO
            params["pageSize"] = pageSize
        case .findOne(let sn):
            params["sn"] = sn
        }
        
        // 发起请求
        return .requestParameters(parameters: params, encoding: URLEncoding.default)
    }
    
    /// 公共请求头
    var headers: [String : String]? {
        return ["devtype": "iOS", "devid": UIDevice().identifierForVendor?.uuidString ?? "unknow"]
    }

}

调用发送请求

orderProvider.request(OrderApi.findOne(sn: "DJKRE3248DFHJEW23")) { (result) in
            let json = try! JSON(data: result.value!.data)
        }

3. 高级使用

实际使用中可能有很多需求,并不像上述这么简单,比如显示网络指示器、统一加密解密后回调等等...Moya中的解决方案是在MoyaProvider创建的时候,可以传入回调闭包和插件

import Foundation
import Moya
import SwiftyJSON
import KRProgressHUD
//import Alamofire
import enum Result.Result

let endpointClosure = { (target: LawApi) -> Endpoint in
    var endpoint: Endpoint = MoyaProvider.defaultEndpointMapping(for: target)
    endpoint = endpoint.adding(newHTTPHeaderFields: ["appName": appName()])
    let request = try! endpoint.urlRequest()
    DLog("\n请求地址:\(request.url!.absoluteString)\n" + "请求头:\(request.allHTTPHeaderFields!)\n" + "请求参数:\(String(describing: String(data: request.httpBody!, encoding: .utf8)?.components(separatedBy: "&")))")
    return endpoint
}

let requestClosure = { (endpoint: Endpoint, done: MoyaProvider.RequestResultClosure) -> Void in
    
//    done(.success(<#T##URLRequest#>))
}

// 插件写法
class LoadingPlugin: PluginType {
    
    func prepare(_ request: URLRequest, target: TargetType) -> URLRequest {
        return request
    }
    
    func willSend(_ request: RequestType, target: TargetType) {
        KRProgressHUD.show()
    }
    
    func didReceive(_ result: Result<Response, MoyaError>, target: TargetType) {
        KRProgressHUD.dismiss()
    }
    
    func process(_ result: Result<Response, MoyaError>, target: TargetType) -> Result<Response, MoyaError> {

        // 在这里对请求进行统一处理(比如有加密,可以统一进行解密)
        if let value = result.value, let _ = try? JSONSerialization.jsonObject(with: value.data, options: JSONSerialization.ReadingOptions.allowFragments) as? [String: Any] {

            let ob = ["code": 666, "description": "牛逼大发了"] as [String : Any]
            let data = try! JSONSerialization.data(withJSONObject: ob, options: JSONSerialization.WritingOptions.fragmentsAllowed)
            
            let response = Response(statusCode: value.statusCode, data: data, request: value.request, response: value.response)
            
            let res = Result<Response, MoyaError>.init(value: response)
            return res
        }
        return result
    }
}

// 生成请求封装类
let lawProvider = MoyaProvider<LawApi>(endpointClosure: endpointClosure, plugins: [LoadingPlugin()])

enum LawApi {
    case getUserInfo
    case getNewsList(pageNo: Int = kPAGENO, pageSize: Int = kPAGESIZE)
    case getNewsDetail(id: String)
}

extension LawApi: TargetType {
    
    var baseURL: URL {
        return URL(string: "http://xxoo/parse/rest.q4w")!
    }
    
    var path: String {
        return ""
    }
    
    var method: Moya.Method {
        return .post
    }
    
    var sampleData: Data {
        return "{}".data(using: String.Encoding.utf8)!
    }
    
    var task: Task {
        // 公共参数
        var params: [String: Any] = ["appsid": "Gz1qYLXeBW8MZuUfDlr9wsAYuVS1cZFMJY9BbaF842L2gRps747o4w=="]
        
        // 收集参数
        switch self {
        case .getUserInfo:
            params["cfg"] = "com.lawyee.lam.web.parse.dto.LamUserDto@getUserInfo"
        case let .getNewsList(pageNo, pageSize):
            params["cfg"] = "com.lawyee.lam.web.parse.dto.LamNewsDto@getNewsList"
            params["pageNo"] = pageNo
            params["pageSize"] = pageSize
        case .getNewsDetail(let id):
            params["cfg"] = "com.lawyee.lam.web.parse.dto.LamNewsDto@getNewsDetail"
            params["id"] = id
        }
        
        // 发起请求
        return .requestParameters(parameters: params, encoding: URLEncoding.default)
    }
    
    var headers: [String : String]? {
        return ["devtype": "wechat", "devid": "af533cbd0168f046e60817b04fd5db7f2057"]
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352