70c3878c4c8ea4e4a7946b75759662c4
Swift - 当Moya遇上RxSwift(网络架构优化)

一、Moya 面向协议网络编程

1️⃣:Moya初探

如果你上面的POP面向协议编程已经看得差不多了,那么这个模块内容是非常简单的!

常规网络层在iOS应用程序中很常见。它们不好有几个原因:

  • 让编写新的应用程序变得困难(“我从哪里开始?”)
  • 很难维护现有的应用程序(“哦,天哪,这一团糟……”)
  • 使编写单元测试变得困难(“我如何再做一次?”)

Moya 的基本思想是:我们需要一些网络抽象层,能够充分封装直接调用Alamofire。它应该足够简单,普通的事情很容易,但是足够全面,复杂的事情也很容易。

  • Moya 的特点有以下几点:
  • 编译时检查正确的API端点访问。
  • 允许您定义具有关联枚举值的不同端点的明确用法。
  • 将测试存根视为一等公民,因此单元测试非常容易。

2️⃣:Moya开发使用

1:接口枚举

public enum LGLoginAPI {
    case login(String, String, String)  // 登录接口
    case smscode(String)                // 登录,发送验证码
    case otherRequest                   // 其他接口,没有参数
}
  • 这个接口枚举提供这个登录注册模块的所有接口
extension LGLoginAPI: TargetType {
    //服务器地址
    public var baseURL: URL {
        return URL(string:"http://127.0.0.1:5000/")!
    }
    // 各个请求的具体路径
    public var path: String {
        switch self {
        case .login:
            return "login/"
        case .smscode:
            return "login/smscode/"
        case .otherRequest:
            return "login/otherRequest/"
        }
    }
    // 请求方式
    public var method: Moya.Method {
        switch self {
        case .login:
            return .post
        case .smscode:
            return .post
        default:
            return .get
        }
    }
    //这个就是做单元测试模拟的数据,只会在单元测试文件中有作用
    public var sampleData: Data {
        return "{}".data(using: String.Encoding.utf8)!
    }
    //请求任务事件(这里附带上参数)
    public var task: Task {
        var param:[String:Any] = [:]
        switch self {
        case .login(let username,let password,let smscode):
            param["username"] = username
            param["password"] = password
            param["smscode"] = smscode
        case .smscode(let username):
            param["username"] = username
        default:
            return .requestPlain
        }
        return .requestParameters(parameters: param, encoding: URLEncoding.default)
    }
    //设置请求头
    public var headers: [String: String]? {
        return nil
    }
}
  • baseURL:服务器地址host 处理
  • path:根据不同的接口,确定各个请求的具体路径
  • method:根据不同的接口,设置请求方式
  • headers:统一配置的请求头信息配置
  • task:配置内部参数,以及task信息

2:登录模块网络管理者

class LGLoginClient: NSObject {
    static let manager = LGLoginClient()

    //MARK: - 验证码事件
    func smscode(username:String,complete:@escaping ((String) -> Void)) {
        let provide = MoyaProvider<LGLoginAPI>()
        provide.request(.smscode(username)) { (result) in
            switch result{
            case let .success(response):
                let dict = LGLoginClient.lgJson(data: response.data)
                complete(dict["smscode"] as! String)
            case let .failure(error):
                print(error)
                complete("")
            }
        }
    }
}
  • MoyaProvider 是此次网络请求的信息提供者
  • MoyaProvider 根据模块 LGLoginAPI 设置的信息绑定数据请求
  • MoyaProvider 通过调用 request 方法传出此次请求的接口,但是参数需要应用层提供!
  • 获取回调信息,然后进行 json 序列化!
  • 最后利用函数式编程思想回调 携带信息的闭包 给应用层

3:应用层调用

@IBAction func didClickCodeBtn(_ sender: Any) {
    LGLoginClient.manager.smscode(username: username) { [weak self](smscode) in
        self?.smscodeTF.text = smscode
    }
}
  • 应用层只需要为此次网络提供信息参数
  • 在回调闭包拿到信息,处理其他业务就OK!
top Created with Sketch.