【iOS】在Swift中使用JSONModel


前言html

首先全部的Model仍是使用oc來寫——看到這一句是否是想關網頁了- - #,在swift裏面直接寫一直報錯因此就將就用oc來寫了,這裏主要是分享一下搭配Alamofire使用的經驗。git


聲明
歡迎轉載,但請保留文章原始出處:) 
博客園:http://www.cnblogs.com
農民伯伯: http://over140.cnblogs.com


正文github

這裏不討論JSONModel和Alamofire這兩個項目,直接上代碼swift

BaseModel.h數組

#import  " JSONModel.h "

@interface BaseModel : JSONModel

-(instancetype)initWithDictionary:(NSDictionary*)dict;

@end

 

BaseModel.m app

#import  " BaseModel.h "

@implementation BaseModel

// Make all model properties optional (avoid if possible)
+(BOOL)propertyIsOptional:(NSString*)propertyName
{
     return YES;
}

-(instancetype)initWithDictionary:(NSDictionary*)dict {
     return (self = [[super init] initWithDictionary:dict error:nil]);
}

@end

全部的Model都要繼承BaseModel,其餘寫法都同樣atom

 

BaseAPI.swiftspa

     internal func requestModel<T: BaseModel>(method: Method, _ URLString: URLStringConvertible, parameters: [String: AnyObject]? = nil, success: (T) -> Void, failure: (NSError?) -> Void) {
        mHttpManager.request(method, URLString , parameters: parameters, encoding: ParameterEncoding.JSON)
            .responseJSON { (request, response, data, error)  in
                 if error == nil {
                     if let dict = data  as? NSDictionary {
                         if let model = T(dictionary: dict  as [NSObject : AnyObject]) {
                            success(model)
                             return
                        }
                    }
                }
                failure(error)
        }
    }
    
     internal func requestArray<T: BaseModel>(method: Method, _ URLString: URLStringConvertible, parameters: [String: AnyObject]? = nil, success: (Array<T>) -> Void, failure: (NSError?) -> Void) {
        mHttpManager.request(method, URLString , parameters: parameters, encoding: ParameterEncoding.JSON)
            .responseJSON { (request, response, data, error)  in
                 if error == nil {
                     if let array = data  as? NSArray {
                         if let result = T.arrayOfModelsFromDictionaries(array  as [AnyObject]).copy()  as? Array<T>{
                            success(result)
                             return
                        }
                    }
                }
                failure(error)
        }
    }

  代碼說明code

一、mHttpManager這個是Alamofire的Manager對象orm

二、注意服務端的返回的數據格式,這裏支持Model和Array<Model>

三、注意在Swift裏面NSDictionary轉Model,用T(dictionary: dict as [NSObject : AnyObject]),這個T就是具體的泛型類型

四、注意在Swift裏面NSArray轉Model數組,用T.arrayOfModelsFromDictionaries(array as [AnyObject]).copy() as? Array<T>,注意不要用BaseModel. arrayOfModelsFromDictionaries(編譯不會報錯可是類型轉不出來)

五、具體用法:

             public func casts(success: (Array<CustomModel>) -> Void, failure: (NSError?) -> Void) {
                requestArray(Method.GET, URL_CASTS, parameters: nil, success: success, failure: failure)
            }
            
             public func like(id: String, success: (CustomModel) -> Void, failure: (NSError?) -> Void) {
                requestModel(Method.PATCH, String(format: URL_CASTS_LIKE, id), parameters: nil, success: success, failure: failure)
            }

很是輕鬆和簡單, 能夠少寫不少重複代碼。

 

後期維護

2015-05-20 Alamofire兼容iOS7有點問題,設置head無論用,請參考我另一篇文章:【iOS】Alamofire庫在iOS7下設置Head無效的問題

 

2016-04-21 

錯誤:fatal error: NSArray element failed to match the Swift Array Element type,參考帖子這裏

Swift 中使用不支持嵌套 JSONModel 數組

@property (strong, nonatomic) NSArray<App *> *apps;
@property (strong, nonatomic) NSArray<User *> *users;

解決辦法: 

@property (strong, nonatomic) NSArray<NSDictionary *> *apps;
@property (strong, nonatomic) NSArray<NSDictionary *> *users;

     而後在 Swift 這邊轉一下

                   self.users = users.map{ User(dictionary: $ 0 ) }
                   self.apps = apps.map{ App(dictionary: $ 0 ) }

 

結束

之後還會分享更多swift的經驗,歡迎交流! 

相關文章
相關標籤/搜索