http://www.cnblogs.com/pengyingh/articles/2364600.html
isMemberOfClass的使用举例isKindOfClass的应用举例
sMemberOfClass方法是来确定对象是否是某一个类的实例。在下一个任务中,我们使用这个方法来验证一个特定的对象是否是一个特定的类实例
#import <Foundation/Foundation.h>
#import <stdio.h>
@interface Class1 : NSObject
{
}
-(void)print;
@end
@implementation Class1
-(void)print
{
printf("This is Class1 .\n");
}
@end
@interface Class2 : NSObject
{
}
-(void)prinf;
@end
@implementation Class2
-(void)prinf
{
printf("This is Class2.\n");
}
@end
int main (int argc, const char * argv[]) {
// insert code here...
Class1 *c1=[Class1 new];
Class2 *c2=[Class2 new];
if ([c1 isMemberOfClass:[Class1 class]]==YES) {
printf("c1 is Member of Class1 !\n\n");
}
if ([c2 isMemberOfClass:[Class2 class]]==YES) {
printf("c2 is Member of Class2 !\n\n");
}
return 0;
}
输出结果:
c1 is Member of Class1 !
c2 is Member of Class2 !
isKindOfClass
我们也可以使用isKindOfClass来检查一个对象是否是一个类的实例。isMemberOfClass和isKindOfClass之间区别是:我们可以使用isKindOfClass来确定一个对象是否是一个类的实例,或者是派生自该类的子类的实例。
例如:我们已经从NSObject派生了自己的类,isMemberOfClass不能检测任何的类都是基于NSObject类这一事实,而isKindOfClass可以。
应用举例:
import <Foundation/Foundation.h>
#import <stdio.h>
@interface Class1 : NSObject
{
}
-(void)print;
@end
@implementation Class1
-(void)print
{
printf("This is Class 1.\n");
}
@end
int main (int argc, const char * argv[]) {
// insert code here...
Class1 *c1=[Class1 new];
if ([c1 isKindOfClass: [NSObject class]]==YES) {
printf("c1 is a kind of NSObject . \n");
}
return 0;
}
输出结果;
c1 is a kind of NSObject .
http://www.linuxidc.com/Linux/2012-07/64453.htm
对象在运行时获取其类型的能力称为内省。内省可以有多种方法实现。
判断对象类型
-(BOOL) isKindOfClass: classObj判断是否是这个类或者这个类的子类的实例
-(BOOL) isMemberOfClass: classObj 判断是否是这个类的实例
我们试试这两个方法的使用。
1、新建Person类继承NSObject,新建Teacher类继承Person
1.1、新建Person类
- #import <Foundation/Foundation.h>
- @interface Person : NSObject
- {
- NSString *name;
- }
- -(void)setName:(NSString*)n;
- @end
- #import "Person.h"
- @implementation Person
- -(void)setName:(NSString *)n
- {
- name = n;
- }
- @end
1.2新建Teacher类
- #import "Person.h"
- @interface Teacher : Person
- -(void)teach;
- @end
- #import "Teacher.h"
- @implementation Teacher
- -(void)teach
- {
- NSLog(@"我教数学");
- }
- @end
1.3 我们先实验下isMemberOfClass方法。
- NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
- Person *person = [[Person alloc] init];
- Teacher *teacher = [[Teacher alloc] init];
- //YES
- if ([teacher isMemberOfClass:[Teacher class]]) {
- NSLog(@"teacher Teacher类的成员");
- }
- //NO
- if ([teacher isMemberOfClass:[Person class]]) {
- NSLog(@"teacher Person类的成员");
- }
- //NO
- if ([teacher isMemberOfClass:[NSObject class]]) {
- NSLog(@"teacher NSObject类的成员");
- }
- [person release];
- [teacher release];
- [pool release];
打印结果:
2012-07-04 14:23:07.965 ObjectiveCTest[2460:f803] teacher Teacher类的成员
只有第一个判断打印出来,isMemberOfClass判断是否是属于这类的实例,是否跟父类有关系他不管。
1.4 isKindOfClass方法
- NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
- Person *person = [[Person alloc] init];
- Teacher *teacher = [[Teacher alloc] init];
- //YES
- if ([teacher isKindOfClass:[Teacher class]]) {
- NSLog(@"teacher 是 Teacher类或Teacher的子类");
- }
- //YES
- if ([teacher isKindOfClass:[Person class]]) {
- NSLog(@"teacher 是 Person类或Person的子类");
- }
- //YES
- if ([teacher isKindOfClass:[NSObject class]]) {
- NSLog(@"teacher 是 NSObject类或NSObject的子类");
- }
- [person release];
- [teacher release];
- [pool release];
2012-07-04 14:34:17.315 ObjectiveCTest[2595:f803] teacher 是 Teacher类或Teacher的子类
2012-07-04 14:34:17.316 ObjectiveCTest[2595:f803] teacher 是 Person类或Person的子类
2012-07-04 14:34:17.316 ObjectiveCTest[2595:f803] teacher 是 NSObject类或NSObject的子类
三个结果都打印出来了。2、
-(BOOL) respondsToSelector: selector判读实例是否有这样方法+(BOOL) instancesRespondToSelector: 判断类是否有这个方法。此方法是类方法,不能用在类的实例或对象
2.1 respondsToSelector的使用
这里不写对象的创建和释放了,参考上面的代码
- // YES
- if ( [teacher respondsToSelector: @selector( setName: )] == YES ) {
- NSLog(@"teacher responds to setSize: method" );
- }
- // NO
- if ( [teacher respondsToSelector: @selector( abcde )] == YES ) {
- NSLog(@"teacher responds to nonExistant method" );
- }
- // YES
- if ( [Teacher respondsToSelector: @selector( alloc )] == YES ) {
- NSLog(@"teacher class responds to alloc method\n" );
- }
2012-07-04 14:39:49.853 ObjectiveCTest[2723:f803] teacher responds to setSize: method
2012-07-04 14:39:49.854 ObjectiveCTest[2723:f803] teacher class responds to alloc method
中间的那个判断我随便写了个selector,当然没有了。respondsToSelector 检查类方法 alloc返回YES
2.2 instancesRespondToSelector
- // NO
- if ( [Person instancesRespondToSelector: @selector(teach)] == YES ) {
- NSLog(@"Person instance responds to teach method" );
- }
- // YES
- if ( [Teacher instancesRespondToSelector: @selector(teach)] == YES ) {
- NSLog(@"Teacher instance responds to teach method");
- }
- // YES
- if ( [Teacher instancesRespondToSelector: @selector(setName:)] == YES ) {
- NSLog(@"Teacher instance responds to setName: method" );
- }
- 2012-07-04 14:52:29.378 ObjectiveCTest[2961:f803] Teacher instance responds to teach method
- 2012-07-04 14:52:29.379 ObjectiveCTest[2961:f803] Teacher instance responds to setName: method
3、Objective-C的id类型
C++ 使用的是强类型:对象必须符合其类型,否则不能通过编译。在 Objective-C 中,i d类型类似于(void*) ,可以指向任何类的实例 。而不需要强制转换。
下面看看使用,
先把Teacher类中的 teach方法修改一下,改成
-(void)teach
{
NSLog(@"%@ 教数学" ,name);
}
然后实现并调用- NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
- Person *person = [[Person alloc] init];
- Teacher *teacher = [[Teacher alloc] init];
- id p = person;
- id t = teacher;
- [t setName:@"张三老师"];
- [t teach];
- [person release];
- [teacher release];
- [pool release];
- 2012-07-04 14:57:55.905 ObjectiveCTest[3085:f803] 张三老师 教数学
#import "NSBundle+Extension.h"
@implementation NSBundle (Extension)
+ (UIView *)viewFromNibName:(NSString *)nibName className:(NSString *)className
{
NSArray *array = [[NSBundle mainBundle] loadNibNamed:nibName owner:self options:nil];
for (id nib in array) {
if ([nib isKindOfClass:NSClassFromString(className)]) {
return (UIView *)nib;
}
}
return nil;
}
@end
使用NSClassFromString
http://www.cocoachina.com/b/?p=219
NSClassFromString是一个很有用的东西,尤其在进行iPhone toolchain的开发上。
正常来说,
id myObj = [[NSClassFromString(@"MySpecialClass") alloc] init];
和
id myObj = [[MySpecialClass alloc] init];
是一样的。但是,如果你的程序中并不存在MySpecialClass这个类,下面的写法会出错,而上面的写法只是返回一个空对象而已。
因此,在某些情况下,可以使用NSClassFromString来进行你不确定的类的初始化。
比如在iPhone中,NSTask可能就会出现这种情况,所以在你需要使用NSTask时,最好使用:
[[NSClassFromString(@"NSTask") .....]]
而不要直接使用[NSTask ...]这种写法。
NSClassFromString的好处是:
1 弱化连接,因此并不会把没有的Framework也link到程序中。
2 不需要使用import,因为类是动态加载的,只要存在就可以加载。因此如果你的toolchain中没有某个类的头文件定义,而你确信这个类是可以用的,那么也可以用这种方法。