OC 中的分类(Category)详解

2024-08-28 16:38
文章标签 详解 分类 oc category

本文主要是介绍OC 中的分类(Category)详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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;
}
编译顺序与运行结果如下:



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













 

 

 

 

这篇关于OC 中的分类(Category)详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1115307

相关文章

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

基于人工智能的图像分类系统

目录 引言项目背景环境准备 硬件要求软件安装与配置系统设计 系统架构关键技术代码示例 数据预处理模型训练模型预测应用场景结论 1. 引言 图像分类是计算机视觉中的一个重要任务,目标是自动识别图像中的对象类别。通过卷积神经网络(CNN)等深度学习技术,我们可以构建高效的图像分类系统,广泛应用于自动驾驶、医疗影像诊断、监控分析等领域。本文将介绍如何构建一个基于人工智能的图像分类系统,包括环境

认识、理解、分类——acm之搜索

普通搜索方法有两种:1、广度优先搜索;2、深度优先搜索; 更多搜索方法: 3、双向广度优先搜索; 4、启发式搜索(包括A*算法等); 搜索通常会用到的知识点:状态压缩(位压缩,利用hash思想压缩)。

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

K8S(Kubernetes)开源的容器编排平台安装步骤详解

K8S(Kubernetes)是一个开源的容器编排平台,用于自动化部署、扩展和管理容器化应用程序。以下是K8S容器编排平台的安装步骤、使用方式及特点的概述: 安装步骤: 安装Docker:K8S需要基于Docker来运行容器化应用程序。首先要在所有节点上安装Docker引擎。 安装Kubernetes Master:在集群中选择一台主机作为Master节点,安装K8S的控制平面组件,如AP

嵌入式Openharmony系统构建与启动详解

大家好,今天主要给大家分享一下,如何构建Openharmony子系统以及系统的启动过程分解。 第一:OpenHarmony系统构建      首先熟悉一下,构建系统是一种自动化处理工具的集合,通过将源代码文件进行一系列处理,最终生成和用户可以使用的目标文件。这里的目标文件包括静态链接库文件、动态链接库文件、可执行文件、脚本文件、配置文件等。      我们在编写hellowor

LabVIEW FIFO详解

在LabVIEW的FPGA开发中,FIFO(先入先出队列)是常用的数据传输机制。通过配置FIFO的属性,工程师可以在FPGA和主机之间,或不同FPGA VIs之间进行高效的数据传输。根据具体需求,FIFO有多种类型与实现方式,包括目标范围内FIFO(Target-Scoped)、DMA FIFO以及点对点流(Peer-to-Peer)。 FIFO类型 **目标范围FIFO(Target-Sc

019、JOptionPane类的常用静态方法详解

目录 JOptionPane类的常用静态方法详解 1. showInputDialog()方法 1.1基本用法 1.2带有默认值的输入框 1.3带有选项的输入对话框 1.4自定义图标的输入对话框 2. showConfirmDialog()方法 2.1基本用法 2.2自定义按钮和图标 2.3带有自定义组件的确认对话框 3. showMessageDialog()方法 3.1

脏页的标记方式详解

脏页的标记方式 一、引言 在数据库系统中,脏页是指那些被修改过但还未写入磁盘的数据页。为了有效地管理这些脏页并确保数据的一致性,数据库需要对脏页进行标记。了解脏页的标记方式对于理解数据库的内部工作机制和优化性能至关重要。 二、脏页产生的过程 当数据库中的数据被修改时,这些修改首先会在内存中的缓冲池(Buffer Pool)中进行。例如,执行一条 UPDATE 语句修改了某一行数据,对应的缓