iOS - 如何深拷贝Array内元素、自定义对象、及自定义对象的属性(上)

简介: 关于深拷贝、浅拷贝,请看上篇iOS - 深拷贝、浅拷贝探索验证
  1. 上篇我们知道:


  • copy: 对NSArray是浅拷贝,NSMutableArray是深拷贝


  • mutableCopy: 始终是深拷贝


  • 无论深浅拷贝,集合对象内元素都是浅拷贝


  1. 本篇主要探索


  • 实现NSArray内元素的拷贝


通过本文你将知道


  • 深拷贝NSArray


  • 深拷贝NSArray内元素Person


  • 深拷贝NSArray内元素Person的属性nickname


本文代码较多,建议一步步耐心跟着代码排查,会有不少收货


探索过程


1. 系统Api


我们发现系统系统了方法- (instancetype)initWithArray:(NSArray<ObjectType> *)array copyItems:(BOOL)flag

其中copyItems:(BOOL)flag表示是否拷贝元素,我们先代码测试一下

NSString *str1 = @"hello world";
    NSMutableString *str2 = [NSMutableString stringWithString:@"hello world"];
    NSArray *array1 = [NSArray arrayWithObjects: str1, str2, nil];
    NSMutableArray *array2 = [[NSMutableArray alloc] initWithArray:array1 copyItems:true];
    NSLog(@"\n array1 = %p class = %@ \n", array1, [array1 class]);
    NSLog(@"\n array2 = %p class = %@ \n", array2, [array2 class]);
    NSLog(@"\n\n======== 元素是String ======== ");
    NSLog(@"\n obj0 = %p class = %@ \n", array1[0], [array1[0] class]);
    NSLog(@"\n obj0 = %p class = %@ \n", array2[0], [array2[0] class]);
    NSLog(@"\n\n======== 元素是mutableString ========");
    NSLog(@"\n obj0 = %p class = %@ \n", array1[1], [array1[1] class]);
    NSLog(@"\n obj0 = %p class = %@ \n", array2[1], [array2[1] class]);


查看输出(为了下文对比,我对输出做了line标记)

2021-05-09 10:59:16.343163+0800 AlgorithmDemo[9904:51756] 
 array1 = 0x100607650 class = __NSArrayI  (line1)
 array2 = 0x1006074a0 class = __NSArrayM (line2)
======== 元素是String ======== 
 obj0 = 0x100008220 class = __NSCFConstantString (line3)
 obj0 = 0x100008220 class = __NSCFConstantString (line4)
======== 元素是mutableString ========
 obj0 = 0x100606330 class = __NSCFString (line5)
 obj0 = 0x100605430 class = __NSCFString (line6)


根据输出我们对得出系统initWithArray : copyItems :方法如下结论


  • 该方法对NSArray对象是深拷贝 (严格意义来说此处不能称为拷贝,是生成新对象) (line1 line2)


  • 当数组内元素为不可变对象时,该方法对元素执行浅拷贝(line3 line4)


  • 当数组内元素为可变对象时,该方法对元素执行深拷贝(line4 line5)


由此我们知道,当我们想深拷贝NSArray及其内部的元素时,用系统方法- (instancetype)initWithArray:(NSArray<ObjectType> *)array copyItems:(BOOL)flag是不行的


2. 归档


如果我们对Array执行归档再解档,结果会是什么样的呢,我们代码测试一下

NSString *str1 = @"hello world";
    NSMutableString *str2 = [NSMutableString stringWithString:@"hello world"];
    NSArray *array1 = [NSArray arrayWithObjects: str1, str2, nil];
    NSMutableArray *array2 = [[NSMutableArray alloc] initWithArray:array1 copyItems:true];
    NSData *data = [NSKeyedArchiver archivedDataWithRootObject:array1];
    NSMutableArray *array3 = [NSKeyedUnarchiver unarchiveObjectWithData:data];
    NSLog(@"\n array1 = %p class = %@ \n", array1, [array1 class]);
    NSLog(@"\n array2 = %p class = %@ \n", array2, [array2 class]);
    NSLog(@"\n array3 = %p class = %@ \n", array3, [array3 class]);
    NSLog(@"\n\n======== 元素是String ======== ");
    NSLog(@"\n obj0 = %p class = %@ \n", array1[0], [array1[0] class]);
    NSLog(@"\n obj0 = %p class = %@ \n", array2[0], [array2[0] class]);
    NSLog(@"\n obj0 = %p class = %@ \n", array3[0], [array3[0] class]);
    NSLog(@"\n\n======== 元素是mutableString ========");
    NSLog(@"\n obj0 = %p class = %@ \n", array1[1], [array1[1] class]);
    NSLog(@"\n obj0 = %p class = %@ \n", array2[1], [array2[1] class]);
    NSLog(@"\n obj0 = %p class = %@ \n", array3[1], [array3[1] class]);


查看输出

2021-05-09 11:14:50.710409+0800 AlgorithmDemo[12261:67019] 
 array1 = 0x1006b9e40 class = __NSArrayI  (line1)
 array2 = 0x1006ba180 class = __NSArrayM  (line1)
 array3 = 0x1006b7410 class = __NSArrayI  (line1)
======== 元素是String ======== 
 obj0 = 0x100008220 class = __NSCFConstantString  (line1)
 obj0 = 0x100008220 class = __NSCFConstantString  (line1)
 obj0 = 0x1006b6d20 class = __NSCFString  (line1)
======== 元素是mutableString ========
 obj0 = 0x1006b9cb0 class = __NSCFString  (line1)
 obj0 = 0x1006ba0e0 class = __NSCFString  (line1)
 obj0 = 0x1006b70e0 class = __NSCFString  (line1)


根据输出,我们很明显得出如下结论


  • 对Array归档并解档后,会生成新的Array对象


  • Array对象内的不可变元素进行了深拷贝


  • Array对象内的可变元素进行了深拷贝


由此,我们得出对如下总结


  • 只深拷贝NSArray对象,我们用mutableCopy就可以


  • 只深拷贝NSMutableArray对象,我们用Copy或mutableCopy都可以


  • 深拷贝Array对象及内部元素,我们可以通过归档的方法解决


3. 重写copyWithZonemutableCopyWithZone的方式深拷贝自定义对象


我们写一个person类测试一下

//
//  Person.h
//  AlgorithmDemo
//
//  Created by Ternence on 2021/5/9.
//
#import <Foundation/Foundation.h>
@interface Person : NSObject
@property (nonatomic, copy) NSString *nickname;
@end


测试代码如下

Person *person = [[Person alloc] init];
    person.nickname = @"码代码的小马";
    NSMutableArray *array1 = [NSMutableArray arrayWithObjects:person, nil];
    NSArray *array2 = [array1 copy];
    NSMutableArray *array3 = [array1 mutableCopy];
    NSLog(@"\n array1 = %p class = %@", array1, [array1 class]);
    NSLog(@"\n array2 = %p class = %@", array2, [array2 class]);
    NSLog(@"\n array3 = %p class = %@", array3, [array3 class]);
    NSLog(@"\n\n======== 数组内元素 ======== ");
    NSLog(@"\n array1[0] = %p class = %@", array1[0], [array1[0] class]);
    NSLog(@"\n array2[0] = %p class = %@", array2[0], [array2[0] class]);
    NSLog(@"\n array3[0] = %p class = %@", array3[0], [array3[0] class]);


相关文章
|
6月前
|
JavaScript 前端开发 API
【JavaScript】<JS内建对象>JavaScript内建对象-Array对象
【1月更文挑战第17天】【JavaScript】<JS内建对象>JavaScript内建对象-Array对象
|
6月前
|
iOS开发 UED
实现一个自定义的iOS动画效果
【4月更文挑战第9天】本文将详细介绍如何在iOS平台上实现一个自定义的动画效果。我们将通过使用Core Animation框架来实现这个动画效果,并展示如何在不同的场景中使用它。文章的目标是帮助读者理解如何使用Core Animation框架来创建自定义动画,并提供一个简单的示例代码。
52 1
|
18天前
|
Swift iOS开发 UED
如何使用Swift和UIKit在iOS应用中实现自定义按钮动画
本文通过一个具体案例,介绍如何使用Swift和UIKit在iOS应用中实现自定义按钮动画。当用户点击按钮时,按钮将从圆形变为椭圆形,颜色从蓝色渐变到绿色;释放按钮时,动画以相反方式恢复。通过UIView的动画方法和弹簧动画效果,实现平滑自然的过渡。
34 1
|
27天前
|
Swift iOS开发 UED
如何使用Swift和UIKit在iOS应用中实现自定义按钮动画
【10月更文挑战第18天】本文通过一个具体案例,介绍如何使用Swift和UIKit在iOS应用中实现自定义按钮动画。当用户按下按钮时,按钮将从圆形变为椭圆形并从蓝色渐变为绿色;释放按钮时,动画恢复原状。通过UIView的动画方法和弹簧动画效果,实现平滑自然的动画过渡。
47 5
|
1月前
|
Swift iOS开发 UED
实现一个自定义的iOS动画效果
本文介绍如何使用Swift和UIKit在iOS应用中实现一个自定义按钮动画,当按钮被点击时,其颜色从蓝色渐变为绿色,形状从圆形变为椭圆形,释放后恢复原状。通过UIView动画方法实现这一效果,代码示例展示了动画的平滑过渡和状态切换,有助于提升应用的视觉体验和用户交互。
50 1
|
1月前
|
存储 JavaScript 前端开发
JavaScript Array(数组) 对象
JavaScript Array(数组) 对象
27 3
|
2月前
|
iOS开发 UED 开发者
iOS 手势中cancelsTouchesInView delaysTouchesBegan delaysTouchesEnded 三种属性的使用
iOS 手势中cancelsTouchesInView delaysTouchesBegan delaysTouchesEnded 三种属性的使用
92 9
|
2月前
|
Swift iOS开发 UED
揭秘一款iOS应用中令人惊叹的自定义动画效果,带你领略编程艺术的魅力所在!
【9月更文挑战第5天】本文通过具体案例介绍如何在iOS应用中使用Swift与UIKit实现自定义按钮动画,当用户点击按钮时,按钮将从圆形变为椭圆形并从蓝色渐变到绿色,释放后恢复原状。文中详细展示了代码实现过程及动画平滑过渡的技巧,帮助读者提升应用的视觉体验与特色。
62 11
|
3月前
|
Swift iOS开发 UED
【绝妙创意】颠覆你的视觉体验!揭秘一款iOS应用中令人惊叹的自定义动画效果,带你领略编程艺术的魅力所在!
【8月更文挑战第13天】本文通过一个具体案例,介绍如何使用Swift与UIKit在iOS应用中创建独特的按钮动画效果。当按钮被按下时,其形状从圆形变化为椭圆形,颜色则从蓝色渐变为绿色;释放后,动画反向恢复原状。利用UIView动画方法及弹簧动画效果,实现了平滑自然的过渡。通过调整参数,开发者可以进一步优化动画体验,增强应用的互动性和视觉吸引力。
52 7
|
3月前
|
Swift iOS开发
iOS开发-属性的内存管理
【8月更文挑战第12天】在iOS开发中,属性的内存管理至关重要,直接影响应用性能与稳定性。主要策略包括:`strong`(强引用),不维持对象生命期,可用于解除循环引用;`assign`(赋值),适用于基本数据类型及非指针对象属性;`copy`,复制对象而非引用,确保对象不变性。iOS采用引用计数管理内存,ARC(自动引用计数)自动处理引用增减,简化开发。为避免循环引用,可利用弱引用或Swift中的`[weak self]`。最佳实践包括:选择恰当的内存管理策略、减少不必要的强引用、及时释放不再使用的对象、注意block内存管理,并使用Xcode工具进行内存分析。