一、@dynamic与@synthesize的区别

@property有两个对应的词,一个是@synthesize,一个是@dynamic。如果@synthesize和@dynamic都没写,那么默认的就是@syntheszie var = _var;

@synthesize的语义是如果你没有手动实现setter方法和getter方法,那么编译器会自动为你加上这两个方法。

@dynamic告诉编译器,属性的setter与getter方法由用户自己实现,不自动生成。(当然对于readonly的属性只需提供getter即可)。假如一个属性被声明为@dynamic var,然后你没有提供@setter方法和@getter方法,编译的时候没问题,但是当程序运行到instance.var =someVar,由于缺setter方法会导致程序崩溃;或者当运行到 someVar = var时,由于缺getter方法同样会导致崩溃。编译时没问题,运行时才执行相应的方法,这就是所谓的动态绑定。

二、通过私有变量来实现@dynamic的存取方法

)Book.h

#import <Foundation/Foundation.h>

#import <CoreData/CoreData.h>

@interface Book :NSObject

{

 @private

    __strong NSString *_name;

    __strong NSString *_author;

}

@property(nonatomic, copy) NSString *name;

@property(nonatomic, copy) NSString *author;

@property(nonatomic, copy) NSString*version;

@end

)Book.m

#import "Book.h" 

@implementation Book

@dynamic name;

@dynamic author;

@synthesize version = _version;

- (id)init

{

    self = [super init];

    if(self)

    {

    }

    return self;

}

- (NSString *)name

{

    if(nil == _name)

    {

        _name = @"you forgot inputbook name";

    }

    return _name;

}

- (void)setName:(NSString *)name

{

    _name = name;

    NSLog(@"_name address:%p", _name);

}

- (NSString *)author

{

    if(nil == _author)

    {

        _author = @"you forgot inputbook author";

    }

    return _author;

}

- (void)setAuthor:(NSString *)author

{

    _author = author;

}

@end

从上面的代码可以看出,用@dynamic后,可以在存取方法中访问一个私有变量来赋值或取值。而@synthesize则直接用@synthesize var = _var;来让属性和私有变量直接等同起来。这就是二者在书写形式上的差别。

三、通过消息转发来实现@dynamic的存取方法

若对于一个属性使用了@dynamic var = _var,则编译器立马报错。这样你就无法像@synthesize那样在var的setter方法和getter方法中使用_var,当然你更不能编写如下的setter方法和getter方法

- (void)setVar:(id)newVar

{

    self.var =newVar;

}

- (void)var

{

    return self.var;

}

这两个方法都是自己调用自己,会导致无限循环直接导致程序崩溃。

这里提供一种利用消息转发机制来实现@dynamic的setter和getter方法。

先上代码:

)Book.h

#import <Foundation/Foundation.h>

@interface Book :NSObject

{

 @private

    NSMutableDictionary *_propertiesDict;

}

@property (nonatomic, copy) NSString *name;

@property (nonatomic, copy) NSString*author;

@property (nonatomic, copy) NSString*version;

@end

)Book.m

#import "Book.h"

@implementation Book

@dynamic    name; // 不能写成name = _name;否则编译器马上报错

@dynamic    author;

@synthesize version; 

- (id)init

{

    self = [super init];

    if(self)

    {

        _propertiesDict = [[NSMutableDictionary alloc] init];

    }

    return self;

}

- (NSMethodSignature *)methodSignatureForSelector:(SEL)selector

{

    NSString *sel = NSStringFromSelector(selector);

    if ([sel rangeOfString:@"set"].location == )

    {

        return [NSMethodSignature signatureWithObjCTypes:"v@:@"];

    }

    else

    {

        return [NSMethodSignature signatureWithObjCTypes:"@@:"];

    }

}

- (void)forwardInvocation:(NSInvocation *)invocation

{

    NSString *key = NSStringFromSelector([invocation selector]);

    if ([key rangeOfString:@"set"].location == )

    {

        key= [[key substringWithRange:NSMakeRange(, [key length]-)] lowercaseString];

        NSString *obj;

       [invocation getArgument:&obj atIndex:];

        [_propertiesDict setObject:obj forKey:key];

    }

    else

    {

        NSString *obj = [_propertiesDict objectForKey:key];

       [invocation setReturnValue:&obj];

    }

}

@end

)main.m

#import <Foundation/Foundation.h>

#import "Book.h"

int main(int argc, const char * argv[])

{

    @autoreleasepool

    {

        Book *book = [[Book alloc] init];

        book.name = @"c++ primer";

        book.author = @"Stanley B.Lippman";

        book.version = @"5.0";

        NSLog(@"%@", book.name);

        NSLog(@"%@", book.author);

        NSLog(@"%@", book.version);

    }

    return ;

}

程序分析:

)在给程序添加消息转发功能以前,必须覆盖两个方法,即methodSignatureForSelector:和forwardInvocation:。methodSignatureForSelector:的作用在于为另一个类实现的消息创建一个有效的方法签名。forwardInvocation:将选择器转发给一个真正实现了该消息的对象。

)Objective-C中的方法默认被隐藏了两个参数:self和_cmd。self指向对象本身,_cmd指向方法本身。举两个例子来说明:

    例一:- (NSString *)name

这个方法实际上有两个参数:self和_cmd。

    例二:- (void)setValue:(int)val

这个方法实际上有三个参数:self, _cmd和val。

    被指定为动态实现的方法的参数类型有如下的要求:

    A.第一个参数类型必须是id(就是self的类型)

    B.第二个参数类型必须是SEL(就是_cmd的类型)

    C.从第三个参数起,可以按照原方法的参数类型定义。举两个例子来说明:

    例一:setHeight:(CGFloat)height中的参数height是浮点型的,所以第三个参数类型就是f。

    例二:再比如setName:(NSString *)name中的参数name是字符串类型的,所以第三个参数类型就是@

)在main.m中有一句代码是book.name = @"c++ primer";程序运行到这里时,会去Book.m中寻找setName:这个赋值方法。但是Book.m里并没有这个方法,于是程序进入methodSignatureForSelector:中进行消息转发。执行完之后,以"v@:@"作为方法签名类型返回。

    这里v@:@是什么东西呢?实际上,这里的第一个字符v代表函数的返回类型是void,后面三个字符参考上面2)中的解释就可以知道,分别是self, _cmd, name这三个参数的类型id, SEL, NSString。

    接着程序进入forwardInvocation方法。得到的key为方法名称setName:,然后利用[invocationgetArgument:&objatIndex:];获取到参数值,这里是“c++ primer”。这里的index为什么要取2呢?如前面分析,第0个参数是self,第1个参数是_cmd,第2个参数才是方法后面带的那个参数。

    最后利用一个可变字典来赋值。这样就完成了整个setter过程。

)在main.m中有一句代码是 NSLog(@"%@", book.name);,程序运行到这里时,会去Book.m中寻找name这个取值方法 。但是Book.m里并没有这个取值方法,于是程序进入methodSignatureForSelector:中进行消息转发。执行完之后,以"@@:"作为方法签名类型返回。这里第一字符@代表函数返回类型NSString,第二个字符@代表self的类型id,第三个字符:代表_cmd的类型SEL。

    接着程序进入forwardInvocation方法。得到的key为方法名称name。最后根据这个key从字典里获取相应的值,这样就完成了整个getter过程。

)注意,调试代码的过程,我们发现只有name和author的赋值和取值进入methodSignatureForSelector:和forwardInvocation:这两个方法。还有一个属性version的赋值和取值,并没有进入methodSignatureForSelector:和forwardInvocation:这两个方法。这是因为,version属性被标识为@synthesize,编译器自动会加上setVersion和version两个方法,所以就不用消息转发了。

四、@dynamic在NSManagedObject的子类中的使用

    @dynamic最常用的使用是在NSManagedObject中,此时不需要显示编程setter和getter方法。原因是:@dynamic告诉编译器不做处理,使编译通过,其getter和setter方法会在运行时动态创建,由Core Data框架为此类属性生成存取方法。

Objective-C中的@dynamic的更多相关文章

  1. 理解Objective C 中id

    什么是id,与void *的区别 id在Objective C中是一个类型,一个complier所认可的Objective C类型,跟void *是不一样的,比如一个 id userName, 和vo ...

  2. 浅谈Objective—C中的面向对象特性

    Objective-C世界中的面向对象程序设计 面向对象称程序设计可能是现在最常用的程序设计模式.如何开发实际的程序是存在两个派系的-- 面向对象语言--在过去的几十年中,很多的面向对象语言被发明出来 ...

  3. Objective-C中的@dynamic(转)

    转自 http://blog.csdn.net/haishu_zheng/article/details/12873151 Objective-C中的@dynamic 一.@dynamic与@synt ...

  4. 在Eclipse中创建Dynamic Web Project具有和MyEclipse中Web Project一样的目录结构

    1.在Eclipse中新建Dynamic Web Project 1.1.修改default output folder build\classes修改为:WebRoot\WEB-INF\classe ...

  5. objective C中的字符串NSStirng常用操作

    objective C中的字符串操作 在OC中创建字符串时,一般不使用C的方法,因为C将字符串作为字符数组,所以在操作时会有很多不方便的地方,在Cocoa中NSString集成的一些方法,可以很方便的 ...

  6. Objective-C中的@dynamic与@synthesize的区别

    Objective-C中的@dynamic 转自:http://blog.csdn.net/haishu_zheng/article/details/12873151 一.@dynamic与@synt ...

  7. Objective-C中的@dynamic 、@synthesize

    Objective-C中的@dynamic 一.@dynamic与@synthesize的区别 @property有两个对应的词,一个是@synthesize,一个是@dynamic.如果@synth ...

  8. Objective C中的ARC的修饰符的使用---- 学习笔记九

    #import <Foundation/Foundation.h> @interface Test : NSObject /** * 默认的就是__strong,这里只是做示范,实际使用时 ...

  9. Objective - C中属性和点语法的使用

    一.属性        属性是Objective—C 2.0定义的语法,为实例变量提供了setter.getter方法的默认实现能在一定程度上简化程序代码,并且增强实例变量的访问安全性         ...

  10. 关于MVC中使用dynamic

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAA2kAAAB6CAIAAACqQIxZAAAgAElEQVR4nO2dT2wcx53v6zgXAgsYvA

随机推荐

  1. Java_SSH项目主要步骤记录

    建立Spring-Struts-Hibernate的步骤整理 1. 建立web project 2. 建立hernate, action, service包 3. 右击项目,add myeclipse ...

  2. CC++初学者编程教程(8) VS2013配置编程助手与QT

    1. 2. 配置编程助手 3. 4. 5. 6. 7. 8. 9. 10. 11. 12. 13. 14. 15. 16. 17. 18. 19. 20. 21. 22. 23. 24. 25. 26 ...

  3. web本地存储-WebSQL

    Web SQL数据库API实际上未包含在HTML 5规范之中,它是一个独立的规范,它引入了一套使用SQL操作客户端数据库的API.W3C 官方在 2011 年 11 月声明已经不再维护 Web SQL ...

  4. java 数字前自动补零实现

    /** * 里数字转字符串前面自动补0的实现. * */ public class TestStringFormat { public static void main(String[] args) ...

  5. Android上运行本地c

    在android代码中编译一个可执行文件,主要功能是对文件的读写, 简单贴出来: #include <errno.h> #include <stdio.h> #include  ...

  6. 网易云课堂_程序设计入门-C语言_第七周:指针与字符串_2GPS数据处理

    2 GPS数据处理(6分) 题目内容: NMEA-0183协议是为了在不同的GPS(全球定位系统)导航设备中建立统一的BTCM(海事无线电技术委员会)标准,由美国国家海洋电子协会(NMEA-The N ...

  7. hdu 5578 Friendship of Frog(multiset的应用)

    Problem Description N frogs . Two frogs are friends if they come from the same country. The closest ...

  8. Android应用中使用百度地图API定位自己的位置(二)

    官方文档:http://developer.baidu.com/map/sdkandev-6.htm#.E7.AE.80.E4.BB.8B3 百度地图SDK为开发人员们提供了例如以下类型的地图覆盖物: ...

  9. extern外部方法使用C#简单样例

    外部方法使用C#简单样例 1.添加引用using System.Runtime.InteropServices; 2.声明和实现的连接[DllImport("kernel32", ...

  10. Clojure 学习入门(19)—— 数组

    1.创建数组 1.1 从集合创建数组 into-array into-array (into-array aseq) (into-array type aseq) 演示样例: user=> (i ...