iOS RxSwift+Moya+ObjectMapper实现网络请求数据与解析

抓取的是有妖气的API

学起来很麻烦、

1、下面是封装好的网络请求部分

项目地址 可以在这里下载代码
目前还在更新代码、、、、、
最近项目上很忙 没来得及继续更新了

//
//  YYQApi.swift
//  YYQCartoon
//
//  Created by kcl on 2018/3/21.
//  Copyright © 2018年 KCL. All rights reserved.
//

import Foundation
import Moya
import ObjectMapper
import RxSwift
import RxCocoa
import Result

final class RequestAlertPlugin: PluginType {

    private let viewController: UIViewController

    init(viewController: UIViewController) {
        self.viewController = viewController
    }

    func willSend(_ request: RequestType, target: TargetType) {

        guard let requestURLString = request.request?.url?.absoluteString else { return }

        let alertViewController = UIAlertController(title: "Sending Request", message: requestURLString, preferredStyle: .alert)
        alertViewController.addAction(UIAlertAction(title: "OK", style: .default, handler: nil))

        viewController.present(alertViewController, animated: true, completion: nil)
    }

    func didReceive(_ result: Result<Moya.Response, MoyaError>, target: TargetType) {


        guard case Result.failure(_) = result else { return }


        let alertViewController = UIAlertController(title: "Error", message: "Request failed with status code: ", preferredStyle: .alert)
        alertViewController.addAction(UIAlertAction(title: "OK", style: .default, handler: nil))

        viewController.present(alertViewController, animated: true, completion: nil)

    }
}

private let YYQEndpointMapping = {(target:YYQApi) -> Endpoint in

    let url = target.baseURL.appendingPathComponent(target.path).absoluteString
    let endpoint = Endpoint(url: url,
                                    sampleResponseClosure: {.networkResponse(200, target.sampleData)},
                                    method: target.method,
                                    task: target.task,
                                    httpHeaderFields: target.headers)

    print("url=\(url),param=\(String(describing: target.task))")
    let token = UserDefaults.standard.string(forKey: "accesstoken") ?? ""
    if token == "" {
        return endpoint.adding(newHTTPHeaderFields: [
            "Accept":"application/json",
            "Content-Type":"application/json",
            "Accept-Language":"en",
//            "Authorization":"key=\(MoveApi.apiKey)"
            ])
    }else{

        return endpoint.adding(newHTTPHeaderFields: [
            "Accept":"application/json",
            "Content-Type":"application/json",
            "Accept-Language":"en",
//            "Authorization":"key=\(MoveApi.apiKey),token=\(token)"
            ])
    }
}
//public final class func defaultRequestMapping(for endpoint: Endpoint, closure: RequestResultClosure) {
//    do {
//        let urlRequest = try endpoint.urlRequest()
//        closure(.success(urlRequest))
//    } catch MoyaError.requestMapping(let url) {
//        closure(.failure(MoyaError.requestMapping(url)))
//    } catch MoyaError.parameterEncoding(let error) {
//        closure(.failure(MoyaError.parameterEncoding(error)))
//    } catch {
//        closure(.failure(MoyaError.underlying(error, nil)))
//    }
//}

//private let YYQRequestTimeoutClosure = { (endpoint: Endpoint, done: @escaping MoyaProvider<YYQApi>.RequestResultClosure) in
//    guard let request = endpoint.urlRequest else { return }
//    request.timeoutInterval = 10    //设置请求超时时间
//    request.httpShouldHandleCookies = false
//    done(.success(request))
//}


/// 网络封装
class YYQRequest{

    private static let dProvider = MoyaProvider<YYQApi>()
    private static let YYQProvider = MoyaProvider<YYQApi>(
        endpointClosure:YYQEndpointMapping
//        plugins:[RequestAlertPlugin(viewController: (UIApplication.shared.keyWindow?.rootViewController)!)]
//        requestClosure:YYQRequestTimeoutClosure,
        )
    //获取Vip列表
    final class func getVipList() ->Observable<VipListModel>{

        return YYQProvider.rx.request(.vipList)
            .mapJSON()
            .asObservable()
            .mapObject(type:VipListModel.self)
    }
    //获取排行列表
    final class func getRankList() ->Observable<RankListModel>{

        return YYQProvider.rx.request(.rankList)
            .mapJSON()
            .asObservable()
            .mapObject(type:RankListModel.self)
    }
    //获取推荐列表
    final class func getRecommendList() ->Observable<RecommendListModel>{

        return YYQProvider.rx.request(.boutiqueList(sexType: 1))
            .mapJSON()
            .asObservable()
            .mapObject(type:RecommendListModel.self)
    }

    //获取订阅列表
    final class func getSubscribeList() ->Observable<SubscribeListModel>{

        return YYQProvider.rx.request(.subscribeList)
            .mapJSON()
            .asObservable()
            .mapObject(type:SubscribeListModel.self)
    }
}


enum YYQApi {

    case searchHot
    case searchRelative(inputText:String)   //相关搜索
    case searchResult(argCon:Int, q:String) //搜索结果
    case boutiqueList(sexType:Int)          //推荐列表
    case special(argCon:Int, page:Int)      //专题
    case vipList //VIP列表
    case subscribeList //订阅列表
    case rankList//排行列表
    case cateList//分类列表
    case comicList(argCon: Int, argName: String, argValue: Int, page: Int)//漫画列表
    case guessLike//猜你喜欢
    case detailStatic(comicid: Int)//详情(基本)
    case detailRealtime(comicid: Int)//详情(实时)
    case commentList(object_id: Int, thread_id: Int, page: Int)//评论
    case chapter(chapter_id: Int)//章节内容
}

extension YYQApi:TargetType
{
    private struct YYQApiKey {
        static var key = "fabe6953ce6a1b8738bd2cabebf893a472d2b6274ef7ef6f6a5dc7171e5cafb14933ae65c70bceb97e0e9d47af6324d50394ba70c1bb462e0ed18b88b26095a82be87bc9eddf8e548a2a3859274b25bd0ecfce13e81f8317cfafa822d8ee486fe2c43e7acd93e9f19fdae5c628266dc4762060f6026c5ca83e865844fc6beea59822ed4a70f5288c25edb1367700ebf5c78a27f5cce53036f1dac4a776588cd890cd54f9e5a7adcaeec340c7a69cd986:::open"
    }
    var baseURL: URL {
        return URL(string: "http://app.u17.com/v3/appV3_3/ios/phone")!

    }

    /// The path to be appended to `baseURL` to form the full `URL`.
    var path: String {
        switch self {

        case .searchHot: return "search/hotkeywordsnew"
        case .searchRelative: return "search/relative"
        case .searchResult: return "search/searchResult"
        case .boutiqueList: return "comic/boutiqueListNew"
        case .special: return "comic/special"
        case .vipList: return "list/vipList"
        case .subscribeList: return "list/newSubscribeList"
        case .rankList: return "rank/list"
        case .cateList: return "sort/mobileCateList"
        case .comicList: return "list/commonComicList"
        case .guessLike: return "comic/guessLike"
        case .detailStatic: return "comic/detail_static_new"
        case .detailRealtime: return "comic/detail_realtime"
        case .commentList: return "comment/list"
        case .chapter: return "comic/chapterNew"

        }
    }

    /// The HTTP method used in the request.
    var method: Moya.Method { return .get }

    /// Provides stub data for use in testing.
//    var sampleData: Data { get }

    /// The type of HTTP task to be performed.
    var task: Task {
        var parameters = ["time":Int32(Date().timeIntervalSince1970),
                          "device_id":UIDevice.current.identifierForVendor!.uuidString,
                          "key":YYQApiKey.key,
//                          "model": UIDevice.current.modelName,
                          "target": "U17_3.0",
                          "version": Bundle.main.infoDictionary!["CFBundleShortVersionString"]!]
        switch self {

            case .searchRelative(let inputText):
                parameters["inputText"] = inputText
            case .searchResult(let argCon, let q):
                parameters["argCon"] = argCon
                parameters["q"] = q
            case .boutiqueList(let sexType):
                parameters["sexType"] = sexType
                parameters["v"] = 3320101
            case .special(let argCon,let page):
                parameters["argCon"] = argCon
                parameters["page"] = max(1, page)
            case .cateList:
                parameters["v"] = 2

            case .comicList(let argCon, let argName, let argValue, let page):
                parameters["argCon"] = argCon
                if argName.count > 0 { parameters["argName"] = argName }
                parameters["argValue"] = argValue
                parameters["page"] = max(1, page)

            case .detailStatic(let comicid),
                 .detailRealtime(let comicid):
                parameters["comicid"] = comicid
                parameters["v"] = 3320101

            case .commentList(let object_id, let thread_id, let page):
                parameters["object_id"] = object_id
                parameters["thread_id"] = thread_id
                parameters["page"] = page

            case .chapter(let chapter_id):
                parameters["chapter_id"] = chapter_id
                default:break
        }

        return .requestParameters(parameters: parameters, encoding: URLEncoding.default)
    }

    /// The type of validation to perform on the request. Default is `.none`.
    var sampleData: Data { return "".data(using: String.Encoding.utf8)! }
    var headers: [String : String]? { return nil }
}

2、需要将ObjectMapper转换成所需要的Observer类型

//
//  YYQApi+Observable.swift
//  YYQCartoon
//
//  Created by kcl on 2018/3/21.
//  Copyright © 2018年 KCL. All rights reserved.
//

import Foundation
import RxCocoa
import RxSwift
import ObjectMapper

// 该文件将ObjectMapper->Observable

extension Observable {

    public func mapObject<T:Mappable>(type:T.Type) -> Observable<T> {

        return self.map{ response in

            //if response is a dictionary, then use ObjectMapper to map the dictionary
            //if not throw an error
            guard let dict = response as? [String:Any] else {

                throw RxSwiftMoyaError.ParseJSONError
            }
            if let error = self.parseError(response: dict){
                throw error
            }
            print("response=\(dict)")
            return Mapper<T>().map(JSON: dict)!
        }
    }
    public func mapArray<T:Mappable>(type:T.Type) ->Observable<[T]>{

        return self.map{ response in

            guard let array = response as? [[String:Any]] else{

                throw RxSwiftMoyaError.ParseJSONError
            }

            for dict:[String :Any] in array {

                if let error = self.parseError(response: dict){

                    throw error
                }
            }
            return Mapper<T>().mapArray(JSONArray: array)
        }
    }

    final fileprivate func parseError(response:[String :Any]?) -> NSError?{

        var error:NSError?

        if let value = response{
            var code:Int?
            var msg:String?

            if let errorDic = value["error"] as? [String:Any]{

                code = errorDic["code"] as? Int
                msg = errorDic["msg"] as? String
                error = NSError(domain: "Network", code: code!, userInfo: [NSLocalizedDescriptionKey:msg ?? ""])
            }
        }
        return error
    }

    final fileprivate func parseServerError() -> Observable {

        return self.map{ (response) in

            let name = type(of: response)
            print(name)

            guard let dict = response as? [String:Any] else {

                throw RxSwiftMoyaError.ParseJSONError
            }
            if let error = self.parseError(response: dict){
                throw error
            }
            return self as! Element
        }
    }
}


enum RxSwiftMoyaError:String {

    case ParseJSONError
    case OtherError
}
extension RxSwiftMoyaError:Swift.Error{}

3、ObjectMapper的Model处理

//
//  RecommendList.swift
//  YYQCartoon
//
//  Created by kcl on 2018/3/21.
//  Copyright © 2018年 KCL. All rights reserved.
//

import Foundation
import ObjectMapper

struct RecommendListModel {

    var code:NSNumber?
    var data:RecommendList?
}

struct RecommendList {

    var stateCode:NSNumber?
    var message:String?
    var returnData:RecommendData?
}

struct RecommendData {

    var galleryItems:[GalleryItem]?
    var textItems:[TextItem]?
    var comicLists:[ComicList]?
    var editTime:String?
}

struct GalleryItem {

    var linkType:NSNumber?
    var cover:String?
    var id:NSNumber?
    var title:String?
    var content:String?
//    var ext:
}
struct TextItem {

}
struct ComicList {

    var comicType:NSNumber?
    var comics:[Comic]?
    var canedit:NSNumber?
    var sortId:String?
    var titleIconUrl:String?
    var newTitleIconUrl:String?
    var description:String?
    var itemTitle:String?
    var argName:String?
    var argValue:String?
    var argType:String?
}

struct Comic {
    //这里面有两块数据源
    var comicId:NSNumber?
    var wideCover:String?
    var tags:Array<Any>?
    var subTitle:String?
    var description:String?
    var cornerInfo:String?
    var short_description:String?
    var author_name:String?
    var is_vip:NSNumber?

    var cover:String?
    var name:String?
    var argName:String?
    var argValue:NSNumber?
    var itemTitle:String?

}
extension Comic: Mappable {

    init?(map: Map) {

    }
    mutating func mapping(map: Map) {


        comicId <- map["comicId"]
        wideCover <- map["wideCover"]
        tags <- map["tags"]
        subTitle <- map["subTitle"]
        description <- map["description"]
        cornerInfo <- map["cornerInfo"]
        short_description <- map["short_description"]
        author_name <- map["author_name"]
        is_vip <- map["is_vip"]

        cover <- map["cover"]
        name <- map["name"]
        argName <- map["argName"]
        argValue <- map["argValue"]
        itemTitle <- map["itemTitle"]
    }
}

extension GalleryItem: Mappable {

    init?(map: Map) {

    }
    mutating func mapping(map: Map) {
        linkType <- map["linkType"]
        cover <- map["cover"]
        id <- map["id"]
        title <- map["title"]
        content <- map["content"]
    }
}

extension RecommendList: Mappable {

    init?(map: Map) {

    }
    mutating func mapping(map: Map) {

        stateCode <- map["stateCode"]
        message <- map["message"]
        returnData <- map["returnData"]
    }
}


extension RecommendData: Mappable {

    init?(map: Map) {

    }
    mutating func mapping(map: Map) {
        galleryItems <- map["galleryItems"]
        textItems <- map["textItems"]
        comicLists <- map["comicLists"]
        editTime <- map["editTime"]
    }
}

extension RecommendListModel: Mappable {

    init?(map: Map) {

    }
    mutating func mapping(map: Map) {
        code <- map["code"]
        data <- map["data"]
    }
}



extension ComicList: Mappable {

    init?(map: Map) {

    }

    mutating func mapping(map: Map) {

        comicType <- map["comicType"]
        comics <- map["comics"]
        canedit <- map["canedit"]
        sortId <- map["sortId"]
        titleIconUrl <- map["titleIconUrl"]
        newTitleIconUrl <- map["newTitleIconUrl"]
        description <- map["description"]
        argName <- map["argName"]
        argValue <- map["argValue"]
        argType <- map["argType"]
    }
}

有兴趣学习的可以加群一起来探讨: 里面也有一些开发相关的PDF文档

群号:727323882

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容

  • iOS网络架构讨论梳理整理中。。。 其实如果没有APIManager这一层是没法使用delegate的,毕竟多个单...
    yhtang阅读 5,184评论 1 23
  • 用到的组件 1、通过CocoaPods安装 2、第三方类库安装 3、第三方服务 友盟社会化分享组件 友盟用户反馈 ...
    SunnyLeong阅读 14,612评论 1 180
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,030评论 25 707
  • 恍恍惚惚~ 周四,忙炸了的一天,但似乎我们的状态都不错,说实话中途因为太忙的关系导致悲痛… 但这次痛比之前的状态好...
    Ermao阅读 186评论 0 1
  • 其实还没画完哈哈orz 图层给我搞乱了 就这样了辣orz
    蜉凉阅读 121评论 1 0