———————————————————————————————————————————

 类别(Category)的声明和实现



实质:类别又叫类目,它其实是对类的一个拓展!但是他不同于继承后的拓展!



(1)在.h文件中,声明类别:



@interface ClassName (CategoryName)   //注意:这里没有冒号,在类名后面直接紧跟小括号,小括号里面是你创建的类别名称

    NewMethod;  //在类别中添加方法

  //★另外不允许在类别中添加变量

@end



说明:



声明类别的格式

①新添加的方法必须写在 @interface 和 @end 之间

②ClassName   现有类的类名(要为哪个类扩展方法)

③CategoryName   待声明的类别名称(也就是你创建的类别名称)

④NewMethod   新添加的方法



(2)在.m文件中,实现类别:



@implementation ClassName (CategoryName)

NewMethod

{

    ……

}

@end



说明:



实现类别的格式

①新方法的实现必须写在 @implementation 和 @end 之间

②ClassName   现有类的类名(要为哪个类扩展方法)

③CategoryName   待声明的类别名称(也就是你创建的类别名称)

④NewMethod   新添加的方法





代码:



代码比较简单,在这里我就不敲了。共有一个Cat类和一个Cat类的拓展类别。总共四个文件。

其中Cat类只是单纯创建了一个类,里面没有做任何的声明和实现。而Cat的拓展类别是通过自动创建出来的,具体拓展类别的操作下面以图片形式呈现:

创建类别时,选择New File,然后选择Objective-C File。

我们会看到多出来了这两个文件。 Cat+doDate.h 和 Cat+doDate.m  ,这也是类别文件的标准书写格式。

这样一个类别就创建成功了。





———————————————————————————————————————————

类别(Category)的使用注意事项



(1)分类只能增加方法,不能增加成员变量、@property(增加@property可能编译不报错,但是运行会出现问题)。

★我们要明白类别是为某个类增加功能(方法),而不是成员变量。



(2)分类可以访问原来类中的成员变量



(3)如果拓展类别中和原类中出现同名的方法,那么优先调用拓展类别中的方法



(4)如果有多个类别中出现同名的方法,那么按照最后编译的那个类别优先调用。

(那么怎么算最后编译的类别呢?我们看下图)

———————————————————————————————————————————

类别(Category)——非正式协议



★要注意,“非正式协议”和我们平时说的协议一点边儿关系都没有。非正式协议是类别(分类)的一种!!!

★非正式协议就是给NSObject类创建的类别(分类),非正式协议一般不需要进行实现,一般在子类中进行方法的重写!

(另外注意一点:非正式协议是NSObject类(还包括他的子类)的类别)



非正式协议优点就是如果你想让某个对象创建出来就有一个方法,那么就可以利用非正式协议写一个。但是这也是他的缺点,如果你不想让其中一个对象有这个方法 ,那么就不能写非正式协议了。所以说,非正式协议的优缺点十分明显,“要么都有,要么都没有”。





非正式协议的应用——利用NSString非正式协议去统计阿拉伯数字的个数:



代码:



//因为我们是统计字符串中阿拉伯数字的个数,所以我们应该为NSString类型增加类别,让每一个字符串都能在创建完成后得到统计其阿拉伯数字个数的方法。

#import <Foundation/Foundation.h>



@interface NSString (numberCountWithNSString)

//我们先写一个类方法去统计字符串中阿拉伯数字的个数

+(int)countWithString:(NSString *)str;



//对象方法去完成上述操作

-(int)countString;

@end



@implementation NSString (numberCountWithNSString)



//我们先写一个类方法去统计字符串中阿拉伯数字的个数

+(int)countWithString:(NSString *)str

{

    unsigned long len=str.length;//整型有无符号(unsigned)和有符号(signed)

    int count=0;//记录阿拉伯数字个数的标记

    for(int i=0;i<len;i++)

    {

        //用一个无符号字符型的变量c去接收传进来的字符串str的每一个字符

        unsigned char c=[str characterAtIndex:i];//characterAtIndex:X 这个方法就是取字符串的第X位字符,将X写为i,然后再套上for循环,自然就是取字符串的每一位的字符了

        if (c<='9'&&c>='0') {

            count++;//符合要求,count+1

        }

    }

    return count;

}



//我们再写一个对象方法去完成上述操作

//不同点在于我们可以直接用字符串去调用该方法,然后用self去调用length和characterAtIndex方法

//-(int)countString;

//{

//    unsigned long len=self.length;//整型有无符号(unsigned)和有符号(signed)

//    int count=0;//记录阿拉伯数字个数的标记

//    for(int i=0;i<len;i++)

//    {

//        //用一个无符号字符型的变量c去接收传进来的字符串str的每一个字符

//        unsigned char c=[self characterAtIndex:i];//characterAtIndex:X 这个方法就是取字符串的第X位字符,将X写为i,然后再套上for循环,自然就是取字符串的每一位的字符了

//        if (c<='9'&&c>='0') {

//            count++;//符合要求,count+1

//        }

//    }

//    return count;

//}



//当然,我们还可以将上面的对象方法简化:直接用对象方法去调用我们写好的类方法,看起来又简便了

-(int)countString;

{

    return [NSString countWithString:self];;

}



@end



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

    @autoreleasepool {

        int n=[NSString countWithString:@"12321dfs321fsdf"];

        NSLog(@"n=%d",n);

        

        int n2=[@"s2123ads123asdsf" countString];

        NSLog(@"n2=%d",n2);

    }

    return 0;

}





———————————————————————————————————————————

类别(Category)——类的延展(匿名类别)



延展类别又称为扩展(Extendsion),Extendsion是Category的一个特例,其名字为匿名(为空),并且新添加的方法一定要予以实现。(一般的Category没有这个限制)



如:@interface Student ()

       @end



上面就是写的 Student 的一个延展。





注意:



①类的延展是绝对私有的,



②类的延展可以不通过创建文件来创建延展,可以直接在类本身的.m文件里写延展的@interface

和@implementation。



③注意类的延展的声明和实现(类的延展的@interface和@implementation)都要写在本类的.m文件里,因为如果把延展的@interface写在.h文件里,那么里面的方法都是public的。



④我们也可以直接省略延展的声明@interface,而直接在本类的.m文件里写延展的实现即可。这里要注意,延展的实现(也就是延展的@implementation)要写在原类的@implementation里面,也就是不能在一个.m文件里写两个@implementation。





代码理解:



*************************************************Dog.h文件**************************************************





#import <Foundation/Foundation.h>



@interface Dog : NSObject

-(void)run;

+(void)bark;

@end



*************************************************Dog.h文件**************************************************



*************************************************Dog.m文件**************************************************



#import "Dog.h"



//延展的声明是可以不写的,我们可以直接在Dog类的@implementation中写延展的实现

//@interface Dog ()

//-(void)eat;

//+(void)eateat;

//@end



@implementation Dog

//原Dog类中我们定义的方法实现

-(void)run

{

    NSLog(@"Dog run!");

    [self eat];

}



+(void)bark

{

    NSLog(@"Dog bark!");

    [self eateat];

}



-(void)love

{

    NSLog(@"ainiaini!");

}



//Dog类的延展中我们声明的方法实现

-(void)eat

{

    NSLog(@"Dog eat!");

    [self lalala];

}



+(void)eateat

{

    NSLog(@"eateat!");

}



-(void)lalala

{

    NSLog(@"lalala!");

}

@end



*************************************************Dog.m文件**************************************************



*************************************************main.m文件**************************************************



#import <Foundation/Foundation.h>

#import "Dog.h"

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

    @autoreleasepool {

        Dog *dog=[[Dog alloc]init];

        [dog run];

        [Dog bark];

        

//        [dog eateat];  这样的调用是错误的,延展中定义的方法是绝对私有的,不可以在外面调用,只能通过在.m中我们写的方法实现中用self调用(不管是在 原方法实现中用self调用 还是 在延展的方法实现中用self调用 都可以)

//        延展中定义的方法是绝对私有的,但是他和一类方法不同,就是和在原类中只有实现没有声明的方法不同。后者可以称之为相对私有的方法,而前者是绝对私有的。

    }

    return 0;

}

*************************************************main.m文件**************************************************

———————————————————————————————————————————

版权声明:本文为博主原创文章,未经博主允许不得转载。

Objective-C 【Category-非正式协议-延展】的更多相关文章

  1. Objective-C( Category 分类,非正式协议,分类延展)

    分类: 主要是给原来类增加一些方法;不可以增加成员变量 Person (Person+eat) 非正式协议: 非正式协议就是类别,即凡是NSObject或其子类Foundation框架中的类增加的类别 ...

  2. Object-C非正式协议与正式协议的区别

    Object-C非正式协议与正式协议的区别 这两个概念困扰我很久了,一直都很像搞清楚到非正式协议和正式协议有什么区别和联系,下面结合网上的资料和自己的看法谈谈这个问题. 一.非正式协议 显然这个名词是 ...

  3. Objective-C非正式协议与正式协议

    这两个概念困扰我很久了,一直都很像搞清楚到非正式协议和正式协议有什么区别和联系,下面结合网上的资料和自己的看法谈谈这个问题. 一.非正式协议 显然这个名词是相对于正式协议而言的.在解释非正式协议之前, ...

  4. 从C#到Objective-C,循序渐进学习苹果开发(3)--分类(category)和协议Protocal的理解

    本随笔系列主要介绍从一个Windows平台从事C#开发到Mac平台苹果开发的一系列感想和体验历程,本系列文章是在起步阶段逐步积累的,希望带给大家更好,更真实的转换历程体验.本文继续上一篇随笔<从 ...

  5. objective-C中的"非正式协议"和“正式协议”

    objective-C中的接口与泛型 先承认我是标题党,因为在obj-c的世界中,官方根本没有"接口"与"泛型"这样的说法. 不过在obj-c中有二个与之接近的 ...

  6. OC基础--分类(category) 和 协议(protocol)

    OC 中的category分类文件相当于 C#中的部分类:OC 中的protocol协议文件(本质是头文件)相当于 C#中的接口.今天就简单说明一下OC中的这两个文件. 由于视频中的Xcode版本低, ...

  7. Objective-C中的类目(Category),延展(Extension)

    类目和延展的作用都是为了扩展一个类. Objective-C中的类目(Category) 一.类目的定义和作用 类目也叫分类,英文Category,在没有原类.m文件的基础上,给该类添加方法. 比如, ...

  8. category与协议的不同

    1,category是针对类进行扩展,而且该类必须有里面的所有成员 协议不同可以选择性实现 2,category是针对一个具体的类实现,其他类没有 协议允许任何类使用,只要实现以后只能通过重写修改 3 ...

  9. objective c, category 和 protocol 中添加property

    property的本质是实例变量 + getter 和 setter 方法 category和protocol可以添加方法 category 和 protocol中可以添加@property 关键字 ...

随机推荐

  1. Azure编程笔记(1):序列化复杂类型的TableEntity字段

    内容提要 在使用MicrosoftAzure的CloudTable存储数据时,我们先要把数据定义成TableEntity的子类.假设TableEntity中包括复杂类型(比方容器类型如List等.或者 ...

  2. 【转】Delphi多线程学习(9):多线程数据库查询(ADO)

    原文:http://www.cnblogs.com/djcsch2001/articles/2382559.html ADO多线程数据库查询通常会出现3个问题: 1.CoInitialize 没有调用 ...

  3. ProgressCircular

    https://github.com/eltld/ProgressCircular

  4. java追加文件的几种方式

    import java.io.BufferedWriter; import java.io.FileOutputStream; import java.io.FileWriter; import ja ...

  5. 商户怎样选择商业wifi进行移动营销

    互联网移动营销的大潮席卷而来,带给很多其它的商户营销理念上的升级和更新.商业wifi营销成为很多其它的商户选择.以往,人们在咖啡店不过聊聊天喝喝咖啡,如今,人们能够更悠闲的歇息娱乐享受带有wifi的咖 ...

  6. android 下修改 hosts文件 及 out of memory的解决

    因为android模拟器host文件无法修改,导致无法通过域名使用http方法调用内网服务,因此从网上大量转载的一种方法,这种方法: 1. 通过emulator -avd avdName -parti ...

  7. HTML5本地存储 localStorage

    HTML5的本地存储是大势所趋,如果仅存储在内存中,则是sessionStorage,他们的语法都是一样,仅仅是一个存储在本地文件系统中,另一个存储在内存中(随着浏览器的关闭而消失),其语句如下: l ...

  8. 解决fonts.gstatic.com无法访问

    最近很多google的服务又在大陆地区受限了,原因不做过多讨论.屏蔽这些服务不仅仅意味着gmail,谷歌学术等方便的工具使用受到限制,更意味着很多寄托于google的web服务无法使用.wordpre ...

  9. java 数组初始化

    一维数组 1)   int[] a;   //声明,没有初始化 2)   int[] a=new int[5];   //初始化为默认值,int型为0 3)   int[] a={1,2,3,4,5} ...

  10. c++ 设计模式8 (Factory Method 工厂方法)

    5. “对象创建”类模式 通过“对象创建”类模式绕开new,来避免对象创建(new)过程中所导致的紧耦合(依赖具体类),从而支持对象创建的稳定.它是接口抽象之后的第一步工作. 5.1 工厂方法 动机: ...