OC 中的分类(Category)详解

本文详细介绍了如何利用Objective-C的分类特性为NSString类添加一个方法,用于计算字符串中阿拉伯数字的个数。通过创建分类并实现相应方法,可以在不修改原始类代码的情况下增加功能。同时,文章还展示了如何设置文件编译顺序以正确调用分类中的方法,并通过代码示例验证了实现效果。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1、Category 分类、类别

2、分类:可以给某一个类扩充一些方法(不修改原来类的代码)

3、分类也分为声明(放在.h文件)和实现(放在.m文件):

分类的声明格式如下

@interface 类名(分类名称)

//要给类扩充的方法的声明

@end

分类的实现

@implementation 类名(分类名称)

//对应的方法实现

@end

4、分类名一般以模块进行命名,在使用的时候用类的哪部分方法就导入相应的分类所在的头文件即可。分类相当于把一个完整的类拆分成多个部分。

5、分类的作用:在不改变原来类内容的基础上,可以增添一些方法。

6、分类的注意事项:

(1)分类只能增加方法,但是不能增加成员变量。

(2)分类方法实现中可以访问原来类中生命的成员变量。

(3)分类中不要增添和原先类中相同的方法,否则会把原来类中的同名方法给覆盖掉,将永远不能再使用。(分类的优先级是最高的,其次是原来类,然后是父类,当用原来类的对象调用方法时,会先从分类中进行查找,分类中找不到时再从原来类中查找,原来类中查找不到时再从原来类直接继承的父类依次进行查找。)

优先级:类分类(最后参与编译的分类优先)——>类——>父类

7、当同一个类的多个分类中有相同的方法时,就要根据文件的编译顺序来进行判断先执行哪个分类中的同名方法。即最后编译的分类会覆盖掉先编译的分类,所以编译器只会执行最后编译的分类中的同名方法。查看文件编译顺序的方法如下步骤所示:

点击项目——>Build Phrase——>Compile Sources 点击下拉框即可得知所有.m的编译顺序。如果想执行哪个分类的同名方法,就把这个分类的.m文件拉到最下面即可。详细步骤如下所示:


8、.h文件只是用来拷贝的,不会被编译,只有.m文件才会被编译。只有.m文件才被称为源代码文件。

9、 给系统自带的类写分类:例如

(1)给NSString增加一个类方法:计算某字符串中阿拉伯数字的个数

(2)给NSString增加一个对象方法:计算当前字符串中阿拉伯数字的个数。

注意:(1)和(2)的功能相同,只不过(1)是由类名调用,计算参数NSString对象中的阿拉伯数字的个数。可以分别为两个方法进行实现。但最常用的解决方法是先写好自身的实例方法,然后为类方法提供一个自身类型NSString *的对象参数,然后在类方法中用这个参数调用自身定义的对象方法即可。

10、类方法中虽然不能直接用self调用自身的实例方法,但是可以换一种方式不使用self调用,就是为自己的类方法添加自身类类型的对象参数,然后直接在类方法中用这个对象参数调用自身类定义的非成员方法即可。类库中大多都是这种方法进行实现。

代码验证(项目一)为系统类添加分类:

为NSString类添加分类以计算阿拉伯数字的个数

编辑NSString+NN.h代码如下:

//
//  NSString+NN.h
//  为NSString添加分类
//
//  Created by apple on 15/8/18.
//  Copyright (c) 2015年 liu. All rights reserved.
//

#import <Foundation/Foundation.h>

@interface NSString (NN)
+(int)numberCountOfString:(NSString *)str; //给自身添加类方法要为类方法添加自身类型的对象参数
-(int)numberCount;//给自身类添加对象方法不需要添加自身类类型的对象参数
@end
编辑实现文件 NSString+NN.m如下

//
//  NSString+NN.m
//  为NSString添加分类
//
//  Created by apple on 15/8/18.
//  Copyright (c) 2015年 liu. All rights reserved.
//

#import "NSString+NN.h"

@implementation NSString (NN)
+(int)numberCountOfString:(NSString *)str
{
//    int count=0;
//    for (int i=0; i<str.length; i++) {
//        unichar c=[str characterAtIndex:i];
//        if (c>='0'&&c<='9') {
//            count++;
//        }
//    }
    return [str numberCount]; //完全可以用已经提供的自身类类型的对象参数来调用对象方法进行实现。
}
-(int)numberCount
{
    int count=0;
    for (int i=0; i<self.length; i++) {
        unichar c=[self characterAtIndex:i];
        if (c>='0'&&c<='9') {
            count ++;
        }
    }
    return count;
}
@end
在main.m中调用如下:

//
//  main.m
//  为NSString添加分类
//
//  Created by apple on 15/8/18.
//  Copyright (c) 2015年 liu. All rights reserved.
//

#import <Foundation/Foundation.h>
#import "NSString+NN.h"
int main(int argc, const char * argv[])
{

    @autoreleasepool {
        
        int count1=[NSString numberCountOfString:@"122dsd144sdsdsd"];
        int count2=[@"sdsa787sdsa78777dsdsd" numberCount];
        NSLog(@"类方法%d  对象方法%d",count1,count2);
    }
    return 0;
}
运行结果如下:


-----------------------------------------------------------------------------------------------------------------------------------------------

代码验证(项目二):验证当前类、多个分类、父类的优先级大小

新建Person.h编辑如下:

//
//  Person.h
//  cate
//
//  Created by apple on 15/8/18.
//  Copyright (c) 2015年 liu. All rights reserved.
//

#import <Foundation/Foundation.h>

@interface Person : NSObject
-(void)Test;
-(void) Test1;
-(void) Test2;
@end
编辑Person.m文件如下:

//
//  Person.m
//  cate
//
//  Created by apple on 15/8/18.
//  Copyright (c) 2015年 liu. All rights reserved.
//

#import "Person.h"

@implementation Person
-(void)Test{
    NSLog(@"父类Person——Test方法被调用");
}
-(void) Test1{
 NSLog(@"父类Person——Test1方法被调用");
}
-(void) Test2{
   NSLog(@"父类Person——Test2方法被调用");
}
@end
新建Student.h编辑如下:

//
//  Student.h
//  cate
//
//  Created by apple on 15/8/18.
//  Copyright (c) 2015年 liu. All rights reserved.
//

#import "Person.h"

@interface Student : Person
-(void)Test;
-(void) Test1;
@end
编辑Student.m如下:

//
//  Student.m
//  cate
//
//  Created by apple on 15/8/18.
//  Copyright (c) 2015年 liu. All rights reserved.
//

#import "Student.h"

@implementation Student
-(void)Test{
    NSLog(@"子类Student——Test方法被调用");
}
-(void)Test1{
    NSLog(@"子类Student——Test1方法被调用");
}
@end
新建分类Student+A_stu.h,编辑如下:

//
//  Student+A_stu.h
//  cate
//
//  Created by apple on 15/8/18.
//  Copyright (c) 2015年 liu. All rights reserved.
//

#import "Student.h"

@interface Student (A_stu)
-(void)Test;
@end
编辑 Student+A_stu.m如下:

//
//  Student+A_stu.m
//  cate
//
//  Created by apple on 15/8/18.
//  Copyright (c) 2015年 liu. All rights reserved.
//

#import "Student+A_stu.h"

@implementation Student (A_stu)
-(void)Test{
    NSLog(@"子类Student的分类A_stu——Test方法被调用");
}
@end
新建分类Student+B_stu.h编辑如下:

//
//  Student+B_stu.h
//  cate
//
//  Created by apple on 15/8/18.
//  Copyright (c) 2015年 liu. All rights reserved.
//

#import "Student.h"

@interface Student (B_stu)
-(void)Test;
@end
编辑 Student+B_stu.m如下:

//
//  Student+B_stu.m
//  cate
//
//  Created by apple on 15/8/18.
//  Copyright (c) 2015年 liu. All rights reserved.
//

#import "Student+B_stu.h"

@implementation Student (B_stu)
-(void)Test{
    NSLog(@"子类Student的分类B_stu——Test方法被调用");
}

@end
在main.m中调用如下:

//
//  main.m
//  cate
//
//  Created by apple on 15/8/18.
//  Copyright (c) 2015年 liu. All rights reserved.
//

#import <Foundation/Foundation.h>
#import "Student.h"
#import "Student+A_stu.h"
#import "Student+B_stu.h"
int main(int argc, const char * argv[])
{

    @autoreleasepool {
        Student *s=[[Student alloc] init];
        [s Test]; //子类Student的分类B_stu——Test方法被调用
        [s Test1];//子类Student——Test1方法被调用
        [s Test2];//父类Person——Test2方法被调用
        [s release];
        /*可以看出先从最后编译的分类文件中寻找方法,,,当分类找不到时再从当前类查找,当前类
         没有时,才从父类进行查找。所以优先级由大到小关系是:分类(最后编译的优先级最大)——>
         当前类——>父类*/
        }
    return 0;
}
编译顺序与运行结果如下:



改变编译顺序,再次运行,结果如下:













评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值