iOS如何将不可变变量改变成可变变量, deep copy objects into mutable

来源:互联网 发布:ubuntu 开机任务栏没了 编辑:程序博客网 时间:2024/05/29 16:22

最近在做一个金融方面的项目,遇到了这个问题。
大概就是我会从后台拿到一个object(它代表一个公司),当然了,它是一个通过json转换过来的不可变(inmutable)字典。

在创建了与之对应的model class之后呢操作变得很方便,可是问题来了,后台总是不断的增加这个公司的信息,而我又不能时刻的更新对应的model,所以就想到在model里添加一个类似trunk的东西用来保存整个字典,而后只要修改这个trunk后再提交到后台就可以了。但是问题是这个字典里面一层套一层,每一层都是不可变的字典或者数组,问题提出来了。

如何将一个object变成mutable object?

收集了一下,大概有如下方法:

1. 将object归档(archive),再解压缩(unarchive)
这样一来,原本的object,包括object里面的object都会变成mutable,也是我正在使用的最为简便的方法。

1
2
NSData* archivedData = [NSKeyedArchiverarchivedDataWithRootObject:model];
idmutableModel = [NSKeyedUnarchiverunarchiveObjectWithData:archivedData];

2. 使用CFPropertyListCreateDeepCopy
这个也是个简便的方法,但是有局限性。只适用于合法的property,比如array, number, date, data, string, and dictionary。另外,跟所有的CF函数一样,我们要负责释放CF函数产生的变量,不然会导致内存泄露。

1
2
3
4
5
6
7
8
9
idoriginalDictionary = something;
NSMutableDictionary*newDictionary = nil;
 
// 注意flag的设置 kCFPropertyListMutableContainers
newDictionary = (NSMutableDictionary*)CFPropertyListCreateDeepCopy(kCFAllocatorDefault, (CFPropertyListRef)originalDictionary, kCFPropertyListMutableContainers);
 
// 使用这个newDictionary
// 最后千万别忘了释放它
[newDictionary release];

3. 递归mutableCopy
笨点的方法但是保证质量适于用任何object甚至是自定义的object

这个方法的英文原文参考: 参考1, 参考2, 参考3

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
- (NSMutableDictionary*) mutableDeepCopy
{
    NSMutableDictionary* ret = [[NSMutableDictionaryalloc] initWithCapacity: [selfcount]];
    NSArray*keys = [selfallKeys];
    for(idkey in keys)
    {
        idoneValue = [selfvalueForKey:key]; // should return the array
        idoneCopy = nil;
 
        if([oneValue respondsToSelector: @selector(mutableDeepCopy)])
            oneCopy = [oneValue mutableDeepCopy];
 
        if([oneValue respondsToSelector:@selector(mutableCopy)])
            oneCopy = [oneValue mutableCopy];
 
        if(oneCopy == nil)// not sure if this is needed
            oneCopy = [oneValue copy];
 
        [ret setValue:oneCopy forKey:key];
 
        //[oneCopy release];
    }
    returnret;
}

4. 如果你的app要求iOS5.0以上的系统,你当然可以考虑不在使用第三方json库,改用苹果自家的NSJSONSerialization
值得注意的是,这个方法似乎并不支持递归的深度转变,我没试过。如果各位有人试过请给我留个言,在这先谢谢了。
点击这里查看NSJSONSerialization的官方文档.

1
2
3
// 注意flag的使用
NSData*data = [[NSFileManagerdefaultManager] contentsAtPath:jSONFile];
NSMutableDictionary*mydict = [NSJSONSerializationJSONObjectWithData:data options:NSJSONReadingMutableContainers|NSJSONReadingMutableLeaveserror:&error];

ok,方法大概就这么多,希望在苹果没有给出官方的函数之前大家都能开心的选择自己喜欢的方法使用。


转载:http://blog.langyufei.com/?p=495
0 0
原创粉丝点击