iOS序列化和反序列化

今天和大家一起来学习一下iOS序列化和反序列化,有疏忽的地方,还望各位不吝赐教。


一、序列化和反序列化

序列化: 将数据结构或对象转换成二进制串的过程。
反序列化:将在序列化过程中所生成的二进制串转换成数据结构或者对象的过程。

在iOS中简单粗暴的解释

  • JSON -> OC 反序列化
  • OC -> JSON 序列化

二、JSON简介

1、什么是JSON

  • json是一种轻量级的数据格式
  • 服务器返回给客户端的数据,一般都是json或者xml的数据
  • 80%都是json

2、JSON和OC的对应关系

  • 标准的JSON的 key 必须使用双引号 iOS必须是双引号
  • json中的{} 对应OC中的NSDictionary
  • json中的[] 对应OC中的NSArray
  • json中的双引号 对应OC中的NSString
  • json中的数字 对应OC中的NSNumber

3、JSON的解析方案

  • 第三方框架:MJExtension(不需要继承,代码没有侵入性)
    在选用框架时注意的问题:(面试福利)
    a、侵入性
    b、易用性
    c、拓展性
  • 苹果原生:NSJSONSerialization(性能最好的)

三、JSON的解析——NSJSONSerialization

1、JSON -> OC 反序列化

 /*
  * 第一个参数:json的二进制数据
  * 第二个参数:
  * NSJSONReadingMutableContainers = (1UL << 0),得到OC对象是可变的
  * NSJSONReadingMutableLeaves = (1UL << 1), 字典和数组中的字符串都是可变的,iOS7以后出现很多问题,一般不会用到
  * NSJSONReadingAllowFragments = (1UL << 2) 既不是字典也不是数组,则必须使用该枚举值 如果返回数据为@“\"hahahahaha\"”
  * 第三个参数:错误信息
  */
    NSDictionary *dic = [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil];

2、OC -> JSON 序列化

    /* 注意:并不是所有的OC对象都可以转化为json 例如字符串
     * 顶层必须是NSArray或者NSDictionnary
     * 所有的元素必须是 NSString, NSNumber, NSArray, NSDictionary, or NSNull
     * 字典中所有的key必须是NSStrings类型的
     * NSNumbers 不能是无穷大
     */
     // 1、提供转化的OC字典
       NSDictionary *dic = @{
                          @"status":@200,
                          @"content":@"hello",
                          @"error":@"enen"
                          };
    // 2、判断OC对象是否可以转换为json对象
    BOOL isVaild = [NSJSONSerialization isValidJSONObject:dic];
    if (isVaild) {
        /*
         * 第一个参数:需要序列化的数据
         * 第二个参数:如果添加就输出排版美观效果 如果不添加按照默认输出
         *        NSJSONWritingPrettyPrinted = (1UL << 0)
         */
        NSData *data = [NSJSONSerialization dataWithJSONObject:dic options:NSJSONWritingPrettyPrinted error:nil];
        NSLog(@"%@",[[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]);
    }else{
    
        NSLog(@"该对象无法转换");
    }

3、Plist文件转换为JSON文件

     NSArray *arrayM = [NSArray arrayWithContentsOfFile:@"plist文件目录"];
    NSLog(@"%@",arrayM);
    
    // 这里要先进行序列化
    NSData *data = [NSJSONSerialization dataWithJSONObject:arrayM options:NSJSONWritingPrettyPrinted error:nil];
    // 说明:不能直接这样保存,会导致输出不是合法的json数据,相当于一个xml文件
    //   [arrayM writeToFile:@"/Users/Urag/Desktop/apps.json" atomically:YES];
    [data writeToFile:@"JSON文件目录" atomically:YES];

四、JSON的解析——MJExtension

这里只是说一下关于使用MJExtension遇到关键字冲突时的解决方式。

// 在模型实现中进行关键字的转换 但是侵入性太强 可以到控制器中进行设置
+ (NSDictionary *)mj_replacedKeyFromPropertyName{

    return @{
             @"ID":@"id",
             };
}
// 控制器中进行设置
[model  mj_setupReplacedKeyFromPropertyName:^NSDictionary *{
        return @{
                 @"ID":@"id",
                 };
    }];

五、XML简介

因为XML现在用的很少,所以上面在iOS中序列化和反序列化说法有点以偏概全了,在这里作补充。。。

一个常见的XML文档一般由以下几部分组成
文档声明
元素(Element)
属性(Attribute)

1、文档声明

  • <?xml version="1.0" encoding="UTF-8"?>
  • 在XML文档的最前面,必须编写一个文档声明,用来声明XML文档的类型
  • 最简单的声明
  • <?xml version="1.0"?>
  • 用encoding属性说明文档的字符编码
  • <?xml version="1.0" encoding="UTF-8"?>

2、元素(Element)

  • <error>用户名不存在</error>
  • 一个元素包括了开始标签和结束标签
  • 拥有内容的元素:<video>Fate</video>
  • 没有内容的元素:<video/>(简写之后)
  • 可以嵌套 但是不能交差嵌套 只能由一个根元素

元素的注意点:

  • xml中的所有的空格和换行,都会当做具体内容处理

3、属性

  • 一个元素可以有多个属性
  • <video name="Fate 第一部" length = "30"/>
  • video元素拥有name和length两个属性
  • 属性必须用“” 或者‘’扩住
  • 实际上,属性表示的信息也可以使用子元素来表示,如
    <video>
    <name>Fate 第一部</name>
    <length> 30 </length>
    </video>

六、XML的解析方案

1、解析方式:

  • DOM 一次性将整个XML文档加载进入内存 比较适合解析小文件
  • SAX 从根元素开始。按照顺序一个元素一个元素往下解析,比较是和解析大文件

2、iOS解析

苹果原生

  • NSXMLParser SAX解析方式,使用简单

第三方框架

  • libxml2 : 纯C语言,默认包含子安iOS SDK中,同时支持DOM和SAX的方式解析
  • GDataxml:DOM的方式解析,基于libxml2
    XML解析方式的建议
  • 小文件: NSXMLParser libxml2
  • 大文件:NSXMLParser libxml2 GDataxml

七、XML解析尝试

1、SAX 解析方式

    // 1、创建XML解析器:SAX 
    NSXMLParser *parser = [[NSXMLParser alloc] initWithData:data];
    
    // 2、设置代理 <NSXMLParserDelegate>
    parser.delegate = self;
    
    // 3、开始解析阻塞的方法
    [parser parse];
// 代理方法的介绍
// 1、开始解析时调用
- (void)parserDidStartDocument:(NSXMLParser *)parser{
    
}
// 2、开始解析某个元素
- (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary<NSString *,NSString *> *)attributeDict{

    NSLog(@"开始解析某个元素 %@ ===== %@",elementName,attributeDict);
    
    // 使用XML的时候记得过滤根元素
    // 字典转模型在这里实现就可以了
    
}
// 3、某个元素解析完毕
- (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName{

    NSLog(@"某个元素解析完毕%@",elementName);
}


// 4、结束解析时调用
- (void)parserDidEndDocument:(NSXMLParser *)parser{

}

2、DOM 解析方式

   /*
    *  使用GDataXML来进行XML的解析 先要进行以下配置 而且记得在编译时要将其改为MRC环境 -fno-objc-arc
    *  libxml includes require that the target Header Search Paths contain
    * /usr/include/libxml2
    * and Other Linker Flags contain
    * -lxml2
    */ 
     // 1、加载XML文档
    GDataXMLDocument *doc = [[GDataXMLDocument alloc] initWithData:data options:kNilOptions error:nil];

    // 2、拿到根元素,得到所有子孙元素
    NSArray * elements = [doc.rootElement elementsForName:@"root"];
   
    // 3、遍历操作
    for (GDataXMLElement *ele in elements) {
        // 元素内部所有的属性-->模型
        GHModel * model = [[GHModel alloc] init];
        model.name = [ele attributeForName:@"name"].stringValue;
        // 将模型存入数组,作为数据源使用
        [self.data addObject:model];
    } 

八、控制台打印JSON数据的中文输出

 /*
  *  控制中文输出核心是重写系统的description方法
  */ 
@implementation NSDictionary (log)
// 重写系统的方法控制输出
- (NSString *)descriptionWithLocale:(id)locale indent:(NSUInteger)level{

    NSMutableString *str = [NSMutableString string];
    
    //{}
    [str appendString:@"{\n"];
    
    // 拼接key-value
    [self enumerateKeysAndObjectsUsingBlock:^(id  _Nonnull key, id  _Nonnull obj, BOOL * _Nonnull stop) {
        [str appendFormat:@"\t%@",key];
        [str appendString:@" : "];
        [str appendFormat:@"%@,\n",obj];
    }];
    
    [str appendString:@"}"];
    
    // 删除逗号 从后往前搜索,第一个符号的位置
    NSRange range = [str rangeOfString:@"," options:NSBackwardsSearch];
    if (range.location != NSNotFound) {
        
        [str deleteCharactersInRange:range];
        
    }
    
    return str;
}
@end

@implementation NSArray(log)
// 重写系统的方法控制输出
- (NSString *)descriptionWithLocale:(id)locale indent:(NSUInteger)level{
    
    NSMutableString *str = [NSMutableString string];
    
    //{}
    [str appendString:@"[\n"];
    
    // 拼接obj
    [self enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        [str appendFormat:@"\t%@,\n",obj];
    }];
    
    
    [str appendString:@"]"];
    
    // 删除逗号 从后往前搜索,第一个符号的位置 如果找到了就删除
    NSRange range = [str rangeOfString:@"," options:NSBackwardsSearch];
    if (range.location != NSNotFound) {
        
        [str deleteCharactersInRange:range];
        
    }
    
    return str;
}

写在最后的话:关于iOS序列化和反序列化相关知识今天就分享到这里,关于iOS序列化和反序列化方面的问题欢迎大家和我交流,共同进步,谢谢各位。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

友情链接更多精彩内容