zoukankan      html  css  js  c++  java
  • Objective-C中的copy协议

    NSObject对象是否可以copy自己

    NSObject类没有实现NSCopying或者NSMutableCopying协议,但是却有copy以及mutableCopy实例方法。然而,如果用NSObject对象直接调用其copy或者mutableCopy方法,会报错。也就是说,NSObject对象无法copy自己。

    既然NSObject类没有实现copy协议,那为什么还要有对应的copy和mutableCopy方法呢?

    因为NSObject类的copy和mutableCopy方法里面分别都会调用copyWithZone以及mutableCopyWithZone,那么所有继承自NSObject类并且实现NSCopying或者NSMutableCopying协议的类的拷贝行为,都可以简化为调用copy或者mutableCopy方法。

    实现copy协议时,何时需要调用[super copyWithZone:zone]

    1 类直接继承自NSObject,无需调用[super copyWithZone:zone]

    @interface X : NSObject <NSCopying>
    
    @property (assign, nonatomic) NSInteger i;
    
    @end
    
    @implementation X
    
    - (id)copyWithZone:(NSZone *)zone {
        X *x = [[[self class] alloc] init]; //没有调用[super copyWithZone:zone]
        x.i = self.i;
        
        return x;
    }

    类X直接继承自NSObject,由于NSObject没有实现copy协议,因此如果调用[super copyWithZone:zone]编译器会报错:

    No visible @interface for 'NSObject' declares the selector 'copyWithZone:'

    2 父类实现了copy协议,子类也实现了copy协议,子类需要调用[super copyWithZone:zone]

    @interface Y : X <NSCopying>
    
    @property (assign, nonatomic) NSInteger j;
    
    
    @end
    
    @implementation Y
    
    
    - (id)copyWithZone:(NSZone *)zone {
        Y *y = [super copyWithZone:zone]; //子类调用[super copyWithZone:zone]
        y.j = self.j;
    
        return y;
    }
    
    
    @end

    子类通过调用[super copyWithZone:zone]方法来分配内存,并且拷贝父类中的实例变量,子类自己的实例变量需要自己拷贝

    3 父类没有实现copy协议,子类实现了copy协议,子类无需调用[super copyWithZone:zone]

    如果上面的例子中类X没有实现copy协议,而子类Y实现了copy协议,这种情况类似与情形1,子类Y无需调用[super copyWithZone:zone]

    - (id)copyWithZone:(NSZone *)zone {
        Y *y = [[[self class] alloc] init]; //子类无需调用[super copyWithZone:zone]
        y.i = self.i;
        y.j = self.j;
    
        return y;
    }

    但是这种时候子类需要自己copy父类中的实例变量。

    为何在copyWithZone方法中要调用[[[self class] alloc] init]来分配内存

    在上面的情形2中,子类Y继承子父类X,如果父类X的copyWithZone方法使用:

    [[X alloc] init];

    来分配内存,那么在子类Y copy的时候就会出现内存分配的问题。

    因为在子类Y的copyWithZone的方法里面调用会调用父类[super copyWithZone:zone]方法,而父类中[[X alloc] init]创建的对象分配的内存大小和类X的实例一样大,而这里是要拷贝Y,类Y继承了X的实例变量,同时也有自己的实例变量,所以类Y的实例所占用的内存明显比类X的实例大,因此,这里会导致内存分配错误。而父类X中如果使用[[[self class] alloc] init]来创建对象分配内存,在运行时self class就是类Y,这样保证了内存分配正确。

    propery中的copy属性

    这里只需记住,property的copy只会产生copyWithZone的调用,而不会产生mutableCopyWithZone的调用。

  • 相关阅读:
    POJ2155 Matrix 【二维线段树】
    BZOJ4785 [Zjoi2017]树状数组 【二维线段树 + 标记永久化】
    B1027 打印沙漏
    Tomcat无法成功启动——双击startup.bat闪退
    MySQL在cmd命令行查看端口号
    1009 说反话(类似回文字符串)
    除基取余法,
    日期差值
    怎么把VS里的scanf_s换成scanf
    联想小新潮怎么修改fn热键以及怎么进入bios状态
  • 原文地址:https://www.cnblogs.com/chaoguo1234/p/6088593.html
Copyright © 2011-2022 走看看