iOS 自动实现对象序列化方法

iOS 中对象序列化,需要遵行NSCoding协议,然后对对象的每个属性进行归档和接档赋值,响应的操作比较繁琐。本文主要介绍 利用 runtime遍历属性 大大简化代码量

具体实现代码如下:
1.先建立NSobject的分类, 定义可能用到的相关类型

static NSString *intType     = @"i"; // int_32t(枚举int型)
static NSString *longTpye    = @"l"; //long类型
static NSString *longlongType= @"q"; // longlong类型
static NSString *BoolType    = @"B"; //bool类型
static NSString *floatType   = @"f"; // float
static NSString *doubleType  = @"d"; // double
static NSString *boolType    = @"c";
static NSString *stringType  = @"NSString"; // NSString 类型
static NSString *numberType  = @"NSNumber"; // NSNumber 类型
static NSString *arrayType   = @"arrayType";//array类型
static NSString *imageType   = @"UIImage"; // UIImage 类型

然后在归档方法中便利自身的属性名称,并且取出自身属性对应的值,进行存储到本地。此时遍历类属性本身,用到了Ivar指针(定义对象的实例变量,包括类型和名字),具体代码如下

//归档
- (void)encodeWithCoder:(NSCoder *)aCoder
{
    unsigned int count; // 属性个数
    Ivar *varArray = class_copyIvarList([self class], &count);

    for (int i = 0; i < count; i++)
    {
        Ivar var = varArray[i];
        const char *cName = ivar_getName(var); // 属性名c字符串
        NSString *proName = [[NSString stringWithUTF8String:cName] substringFromIndex:1]; //OC字符串,并且去掉下划线 _
        const char *cType = ivar_getTypeEncoding(var); // 获取变量类型,c字符串

        id value = [self valueForKey:proName];
        NSString *proType = [NSString stringWithUTF8String:cType]; // oc 字符串

        if ([proType containsString:@"NSString"]) {
            proType = stringType;
        }
        if ([proType containsString:@"NSNumber"]) {
            proType = numberType;
        }
        if ([proType containsString:@"NSArray"]) {
            proType = arrayType;
        }
        if ([proType containsString:@"UIImage"]) {
            proType = imageType;
        }

        // (5). 根据类型进行编码
        if ([proType isEqualToString:intType] || [proType isEqualToString:boolType] || [proType isEqualToString:BoolType]) {
            [aCoder encodeInt32:[value intValue] forKey:proName];
        }
        else if ([proType isEqualToString:longTpye]) {
            [aCoder encodeInt64:[value longValue] forKey:proName];
        }
        else if ([proType isEqualToString:floatType]) {
            [aCoder encodeFloat:[value floatValue] forKey:proName];
        }
        else if ([proType isEqualToString:longlongType] || [proType isEqualToString:doubleType]) {
            [aCoder encodeDouble:[value doubleValue] forKey:proName];
        }
        else if ([proType isEqualToString:stringType]) { // string 类型
            [aCoder encodeObject:value forKey:proName];
        }
        else if ([proType isEqualToString:numberType]) {
            [aCoder encodeObject:value forKey:proName];
        }
        else if ([proType isEqualToString:arrayType]) {
            [aCoder encodeObject:value forKey:proName];
        }
        else if ([proType isEqualToString:imageType]) { // image 类型
            [aCoder encodeDataObject:UIImagePNGRepresentation(value)];
        }
    }
    free(varArray);
}

其次进行解档, 原理和归档差不多, 直接上代码

- (instancetype)initWithCoder:(NSCoder *)aDecoder
{
    self = [self init];
    if (self) {

        unsigned int count;
        Ivar *varArray = class_copyIvarList([self class], &count);

        for (int i = 0; i < count; i++) {
            Ivar var = varArray[i];
            const char *cName = ivar_getName(var); // 属性名c字符串
            NSString *proName = [[NSString stringWithUTF8String:cName] substringFromIndex:1]; //OC字符串,并且去掉下划线 _
            const char *cType = ivar_getTypeEncoding(var); // 获取变量类型,c字符串
            NSString *proType = [NSString stringWithUTF8String:cType]; // oc 字符串

            if ([proType containsString:@"NSString"]) {
                proType = stringType;
            }
            if ([proType containsString:@"NSNumber"]) {
                proType = numberType;
            }
            if ([proType containsString:@"NSArray"]) {
                proType = arrayType;
            }
            if ([proType containsString:@"UIImage"]) {
                proType = imageType;
            }

            if ([proType isEqualToString:intType] || [proType isEqualToString:boolType] || [proType isEqualToString:BoolType]) {
                int32_t number = [aDecoder decodeInt32ForKey:proName];
                [self setValue:@(number) forKey:proName];
            }
            else if ([proType isEqualToString:longTpye]) {
                int64_t number = [aDecoder decodeInt64ForKey:proName];
                [self setValue:@(number) forKey:proName];
            }
            else if ([proType isEqualToString:floatType]) {
                float number = [aDecoder decodeFloatForKey:proName];
                [self setValue:@(number) forKey:proName];
            }
            else if ([proType isEqualToString:longlongType] || [proType isEqualToString:doubleType]) {
                double number = [aDecoder decodeFloatForKey:proName];
                [self setValue:@(number) forKey:proName];
            }
            else if ([proType isEqualToString:stringType]) { // string 类型
                NSString *string = [aDecoder decodeObjectForKey:proName];
                [self setValue:string forKey:proName];
            }
            else if ([proType isEqualToString:numberType]) {
                NSString *number = [aDecoder decodeObjectForKey:proName];
                [self setValue:number forKey:proName];
            }
            else if ([proType isEqualToString:arrayType]) {
                NSArray *array = [aDecoder decodeObjectForKey:proName];
                [self setValue:array forKey:proName];
            }
            else if ([proType isEqualToString:imageType]) { // image 类型
                UIImage *image = [UIImage imageWithData:[aDecoder decodeDataObject]];
                [self setValue:image forKey:proName];
            }
        }
    }
    return self;
}

最后也就是 存储方法 、 清除存储的本地缓存 和 获取本地存储数据的方法

//存储路径
- (NSString *)filePathWithUniqueFlagString:(NSString *)uniqueFlag
{
    NSString *docPath = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];
    NSString *detailPath = [NSString stringWithFormat:@"%@_%@",uniqueFlag,[NSString stringWithUTF8String:object_getClassName(self)]];
    NSString *path = [docPath stringByAppendingPathComponent:detailPath];
    return path;
}

//保存对象数据到本地
- (void)saveDataToLocalWithUniqueFlagKey:(NSString *)uniqueFlagKey
{
    [NSKeyedArchiver archiveRootObject:self toFile:[self filePathWithUniqueFlagString:uniqueFlagKey]];
}

//清空本地存储的对象数据
- (id)getDataFromLocalWithUniqueFlagKey:(NSString *)uniqueFlagKey
{
    return [NSKeyedUnarchiver unarchiveObjectWithFile:[self filePathWithUniqueFlagString:uniqueFlagKey]];
}

//从本地获取对象数据
- (BOOL)removeDataFromLocalWithUniqueFlagKey:(NSString *)uniqueFlagKey
{
    NSError *error = nil;
    [[NSFileManager defaultManager] removeItemAtPath:[self filePathWithUniqueFlagString:uniqueFlagKey] error:&error];
    if (!error) {
        return YES;
    }
    else {
        return NO;
    }
}

完整项目下载地址如下:https://github.com/maxzhang123/MXCoding.git

转载请注明 出处,谢谢

时间: 2024-08-26 10:16:51

iOS 自动实现对象序列化方法的相关文章

python对象序列化或持久化的方法

http://blog.csdn.net/chen_lovelotus/article/details/7233293 一.Python对象持久化方法 目前为止,据我所知,在python中对象持久化有以下几种方法: 1. 使用(dbhash/bsddb, dbm, gdbm, dumbdbm 等)以及它们的"管理器"( anydbm ).只提供了 Python 字 符串的永久性储存. 提供一个类似字典和文件的对象,可以完成字符串的永久性存储. 2. 使用marshal和pickle来序

iOS修炼之道:objective-c 类、对象和方法 1

1.类,对象和方法的概念 类是对象的独特表现,每个对象包含各自的私有信息,方法是改变和提供访问的这些数据的手段. 因此,objective-c 采用特定的语法对类和实例应用方法: [classOrInstance method]; 或者理解为消息的传递和接受 [receiver message]; 实例如下: 1 myCar = [[Car new] init]; 2 [myCar dive]; 详细描述关于一个处理分数Objective-c类 代码如下: 1 // 2 // main.m 3

iOS对象序列化

系统对象的归档我就不介绍了,这个不复杂,自己看一下就会了. 我在这里主要介绍自定义对象的归档. Sample.h文件 9 #import <Foundation/Foundation.h> 10 11 @interface Sample : NSObject<NSCoding> { 12 13 NSString* name; 14 int magicNumber; 15 float shoeSize; 16 NSMutableArray *subThingies; 17 } 18

iOS writeTofile 和对象的序列化

前言:做了一个图片浏览的小demo,支持随意添加.删除图片,图片放大.缩小,带矩形框的截图.随后几篇博客都会详细讲解在此过程中遇到的各种问题.这篇主要讲,在做添加.删除这个功能时,遇到的存文件的问题. 我们想实现在以后每次打开这个程序时,不用再一张张手动添加,而会保留用户上次已经选择的图片,那么就需要把这些图片存入内存,每次从内存中读取. 正文: 一.存文件有几种方式? 直接写入FILE.对象序列化 二.直接写入FILE 在我的程序里,我是把图片都存在一个NSMutableDictionary,

iOS中僵尸对象的实现方法

什么是僵尸对象?所谓僵尸,就是过度释放的对象.在ios开发中,僵尸对象对于开发人员调试程序来说很有用.我们通常将NSZombieEnabled环境变量设置为YES来打开僵尸对象,但这会导致所有的对象都不会被释放,程序长时间运行会占用大量内存. 那有么有什么其他方法来实现僵尸对象呢?下面小编就给大家介绍下,模仿XCode用代码实现僵尸对象的方法. 创建僵尸对象 在ios开发中,当一个普通对象的retainCount变成0的时候,会调用dealloc,代码要勾住dealloc后,就可进行一下操作:

(远程方法的调用)对象序列化、反射在网络编程的运用

对象的序列化与反序列化[p213页]当两个进程进行远程通信时,彼此可以发送各种类型的数据,如文本.图 片.语音和视频等.无论不纯二进制序列的形式在网络上传送.当两个 java进程进行远程通信时,一个进程把一个java对象发送到另一个进程 中.不过发送方需要把这个java对象转换为字节序列,才能在网络上传 送:接收方则需要把自己序列在恢复为java对象.把java对象转换为字 节序列的过程称为对象序列化:把字节序列恢复为java对象的过程称为 对象称为对象的反序列化.JDK类库中的序列化APIja

李洪强iOS开发之OC[018]对象和方法之间的关系

// //  main.m //  18 - 对象和方法之间的关系 // //  Created by vic fan on 16/7/14. //  Copyright © 2016年 李洪强. All rights reserved. // /** * OC有参有返回值的方法 有参有返回值方法的声明 - (int)sum:(int)x andY:(int)y; - (int)sum:(int)x andY:(int)y{ return x+y; 关系: 1 对象作为方法的参数; 显示人的信息

黑马程序员——25,打印流,合并流,对象序列化,管道流,RandomAccessFile

------<ahref="http://www.itheima.com" target="blank">Java培训.Android培训.iOS培训..Net培训</a>.期待与您交流! ------- /* IO流的打印流:专门用于打印的流 字节打印流PrintStream PrintStream的构造函数可以接收file对象,String型字符串路径,字节输出流 字符打印流PrintWriter PrintWriter的构造函数可以接收

JavaScript之对象序列化详解

一.什么是对象序列化? 对象序列化是指将对象的状态转换为字符串(来自我这菜鸟的理解,好像有些书上也是这么说的,浅显易懂!): 序列化(Serialization)是将对象的状态信息转换为可以存储或传输的形式的过程(来自“百度百科—序列化“,学术性强,略显高端): 二.为什么会有对象序列化? 世间万物,都有其存在的原因.为什么会有对象序列化呢?因为程序猿们需要它.既然是对象序列化,那我们就先从一个对象说起: var obj = {x:1, y:2}; 当这句代码运行时,对象obj的内容会存储在一块