协议分非正式协议和正式协议

2024-03-28 22:18
文章标签 协议 正式 非正式

本文主要是介绍协议分非正式协议和正式协议,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

先说说非正式协议

非正式协议在Obj-c中关键字虽然是interface,但是这个和C#中的借口并不完全相同。

回忆之前学过的内容,我们定义一个类Sample时,会生成一个Sample.h,代码如下:

#import <foundation foundation.h>

@interface Sample:NSObject {

}

-(void) HelloWorld;

@end

它表明Sample类中,约定了应该有一个名为HelloWorld的方法(注:这里说的时应该,而不是必须)

它只是一种君子协议。

如果我们在Sample.m中,并不遵守这个约定(即不实现这个方法),编译器会给出警告。但是还时会编译成功

xieyi1

上图中的提示:Incomplete implementation of class “Sample”. 意为:Sample类并未完全实现interface中约定的方法。
这就是obj-c中的协议跟c#中的接口不一样的地方:在c#中接口是强制必须实现的,否则编译这一关就过不了,而obj-c虽然在编译时会警告,但是最终能编译通过。

正式协议(protocal)
其实就是非正式协议(interface)换了一种写法而已,看上去更正规一些,语义上更强烈一些:要求采用该协议的类,”必须”实现协议中约定的方法。但是比较娱乐的是,即使是号称正式协议,编译器在编译时,遇到不守规矩的情况,仍然只是给出警告。(当然正式协议也有它存在的意义,后面会提到)
这里我们定义一个IQuery的协议

IQuery.h

@protocol IQuery

-(void) Query:(NSString*) sql;

@end

除了把关键字@interface换成了@protocal,其它的基本上没变化。下面定义一个类DBQuery,并采用这个正式协议
DBQuery.h

#import <Foundation/Foundation.h>

#import “IQuery.h”

@interface DBQuery : NSObject<IQuery> {

}

@end

注意这里的DBQuery:NSObject<IQuery>,它表明DBQuery继承自NSObject,同时要实现接口IQuery。
DBQuery.m

#import “DBQuery.h”
@implementation DBQuery
-(void) Query:(NSString *)sql
{
NSLog(@”Query is called. sql:%@”,sql);
}

@end

当然,如果在DBQuery.m中不实现方法Query,也能编译通过,只是会收到一个警告。
也许到目前为止,你会觉得protocal跟interface比起来,都是类似的概念,protocal设计纯属多余。其实不然,protocal存在的一个重要意义在于:
正式协议(protocal)可以将业务中的方法定义剥离出来,形成一个单独的文件,这跟传统OO中的提取接口是不谋而合的。如果遇到二个系统需要交换数据,可以制定一套双方都遵守的protocal,然后这二个系统中都把这个协议文件添加到项目中,实现它即可。这一功能,非正式协议(@interface)就做不到。(不信大家可以把NSObject<IQuery>中的IQuery改成其它类的interface 定义名称试试,编译根本通不过)
此外,obj-C 2.0中对正式协议还做了一些扩展,允许把正式协议中的方法标识为“必须实现(@requied)”和“可选实现(@optional)”二类,如果协议中的方法被标识为@optional,即使采用该协议的类不实现这些方法,编译器也不会给出警告。这赋予了正式协议更多的灵活性。示例如下:

@protocol IQuery
@required
-(void) Query:(NSString*) sql;
@optional
-(void) HelloWorld;
@end

有了@optional关键字以后,其实“非正式协议”在语义上完全可以被“正式协议”所取代,事实上Cocoa中的非正式协议都在逐渐被标有@optional方法的正式协议所代替。
如果你在XCode的代码中,选中NSObject,右击–>Jump to Definition,会发现NSObject其实就是一个interface或protocal

xieyi2

选择protocal NSObject 继续,会看到NSObject.h文件中关于protocal NSObject的定义

xieyi3

同样的,你还可以看到interface NSObject的定义

xieyi4

从这里可以看到,非正式协议的interface NSObject其实最终采用的还是正式协议protocal NSObject.

也就是说,在obj-c的OO世界中,身为万物之祖的NSObject其实也就一个”正式协议”,所以从NSObject派生出的所有类,都只是在遵守一个或多个协议而已。

另一个话题”泛型”

在obj-c中,一切皆为指针。前面的学习中,我们已经接触到了一种特殊的类型id,它可以认为是一种特殊的指针:可以指向任何类型的对象。id 再加上正式协议,能够达到形似c#中泛型的效果(注:只是形似,并非神似)

#import <Foundation/Foundation.h>
#import “IQuery.h”

@interface DBQuery : NSObject<IQuery> {

}

-(void) test:(id<IQuery>) obj;
@end

注意这里的 -(void) test:(id<IQuery>) obj; 这表明test方法接受一个任意类型的对象做为参数,但是该参数对象必须实现接口IQuery(也可以说成该参数对象必须采用正式协议IQuery),是不是跟c#中的
void test(List<IQuery> obj) 长得很象?

自己的理解:

其实协议就相当于Java 中的interface 或者C++ 中的virtual class

举个例子说说:

首先我们声明一个协议

@protocol MyProtocol
- (void)myProtocolMethod;
@end

@interface TestA
{
TestB<myProtocol>* pB; // 说明类B应该支持<遵守>myProtocol这个协议
}
@interface TestB: NSObject<myProtocol>{

}

现在我们看如果我们在TestA里面昨晚了一件事情,然后要通知TestB。那就要通过myProtocol来实现,那通过pB我们就可以给myProtocol发通知,那TestB也就收到消息了

下面再附上一个用协议实现回调函数的例子,(网上转来的,但可以帮助我们理解)

在编写android开发时,最常用的就是回调函数。自己编写回调函数,实现动态加载数据,加载完数据之后就利用回调函数通知给前台页面,显示相应数据的界面。在iphone中利用协议可以很容易的实现回调函数,后台加载数据,然后显示在前台页面。

实现一个显示文字为测试的视图,然后经过3秒钟测试文字变为回调函数文字。相应的截图如下:

实现的代码如下:

#import <UIKit/UIKit.h>
@protocol NoteDelegate //声明协议
//回调函数
-(void)messageCallBack:(NSString *)string;
@end

调用协议:

#import <Foundation/Foundation.h>
#import “NoteDelegate.h”
@interface ManagerMessage : NSObject {
id<NoteDelegate> *noteDelegate;
}
@property (nonatomic,retain) id<NoteDelegate> *noteDelegate; //@property预编译命令的作用是自动声明属性的 setter 和getter方法
-(void)startThread;
@end

#import “ManagerMessage.h”
@implementation ManagerMessage
@synthesize noteDelegate;
//开始一个线程
-(void)startThread
{
//启动线程
[NSTimer scheduledTimerWithTimeInterval:3
target:self
selector:@selector(targetMethod:)
userInfo:nil
repeats:NO];
}
-(void)targetMethod:(NSString *)string
{
if (self.noteDelegate!=nil) {
//完成线程 调用回调函数
[self.noteDelegate messageCallBack:@"回调函数"]; //在这通过协议进行回调
}
}
@end

前台页面实现:

#import “IphoneDeleteViewController.h”
#import “ManagerMessage.h”
@implementation IphoneDeleteViewController
@synthesize textView;

//回调函数
-(void)messageCallBack:(NSString *)string
{
self.textView.text=string; //前台收到回调信息 给了textView
}

- (void)viewDidLoad {
[super viewDidLoad];
self.textView.text=@”测试”;
ManagerMessage *message=[[ManagerMessage alloc] init];
//通知调用协议
message.noteDelegate=self;
[message startThread];
[message release];
}

- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
}

- (void)viewDidUnload {
self.textView=nil;
}

- (void)dealloc {
[self.textView release];
[super dealloc];
}

@end


转载地址:http://www.iloss.me/?p=545


这篇关于协议分非正式协议和正式协议的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Nginx中配置HTTP/2协议的详细指南

《Nginx中配置HTTP/2协议的详细指南》HTTP/2是HTTP协议的下一代版本,旨在提高性能、减少延迟并优化现代网络环境中的通信效率,本文将为大家介绍Nginx配置HTTP/2协议想详细步骤,需... 目录一、HTTP/2 协议概述1.HTTP/22. HTTP/2 的核心特性3. HTTP/2 的优

关于WebSocket协议状态码解析

《关于WebSocket协议状态码解析》:本文主要介绍关于WebSocket协议状态码的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录WebSocket协议状态码解析1. 引言2. WebSocket协议状态码概述3. WebSocket协议状态码详解3

Qt 中集成mqtt协议的使用方法

《Qt中集成mqtt协议的使用方法》文章介绍了如何在工程中引入qmqtt库,并通过声明一个单例类来暴露订阅到的主题数据,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录一,引入qmqtt 库二,使用一,引入qmqtt 库我是将整个头文件/源文件都添加到了工程中进行编译,这样 跨平台

TP-Link PDDNS服将于务6月30日正式停运:用户需转向第三方DDNS服务

《TP-LinkPDDNS服将于务6月30日正式停运:用户需转向第三方DDNS服务》近期,路由器制造巨头普联(TP-Link)在用户群体中引发了一系列重要变动,上个月,公司发出了一则通知,明确要求所... 路由器厂商普联(TP-Link)上个月发布公告要求所有用户必须完成实名认证后才能继续使用普联提供的 D

Java如何接收并解析HL7协议数据

《Java如何接收并解析HL7协议数据》文章主要介绍了HL7协议及其在医疗行业中的应用,详细描述了如何配置环境、接收和解析数据,以及与前端进行交互的实现方法,文章还分享了使用7Edit工具进行调试的经... 目录一、前言二、正文1、环境配置2、数据接收:HL7Monitor3、数据解析:HL7Busines

【Linux】应用层http协议

一、HTTP协议 1.1 简要介绍一下HTTP        我们在网络的应用层中可以自己定义协议,但是,已经有大佬定义了一些现成的,非常好用的应用层协议,供我们直接使用,HTTP(超文本传输协议)就是其中之一。        在互联网世界中,HTTP(超文本传输协议)是一个至关重要的协议,他定义了客户端(如浏览器)与服务器之间如何进行通信,以交换或者传输超文本(比如HTML文档)。

【Go】go连接clickhouse使用TCP协议

离开你是傻是对是错 是看破是软弱 这结果是爱是恨或者是什么 如果是种解脱 怎么会还有眷恋在我心窝 那么爱你为什么                      🎵 黄品源/莫文蔚《那么爱你为什么》 package mainimport ("context""fmt""log""time""github.com/ClickHouse/clickhouse-go/v2")func main(

2024.9.8 TCP/IP协议学习笔记

1.所谓的层就是数据交换的深度,电脑点对点就是单层,物理层,加上集线器还是物理层,加上交换机就变成链路层了,有地址表,路由器就到了第三层网络层,每个端口都有一个mac地址 2.A 给 C 发数据包,怎么知道是否要通过路由器转发呢?答案:子网 3.将源 IP 与目的 IP 分别同这个子网掩码进行与运算****,相等则是在一个子网,不相等就是在不同子网 4.A 如何知道,哪个设备是路由器?答案:在 A

Modbus-RTU协议

一、协议概述 Modbus-RTU(Remote Terminal Unit)是一种基于主从架构的通信协议,采用二进制数据表示,消息中的每个8位字节含有两个4位十六进制字符。它主要通过RS-485、RS-232、RS-422等物理接口实现数据的传输,传输距离远、抗干扰能力强、通信效率高。 二、报文结构 一个标准的Modbus-RTU报文通常包含以下部分: 地址域:单个字节,表示从站设备

微软正式推出 Spartan 斯巴达浏览器

作为用于替代 IE 浏览器的下一代继任者,微软的 Project Spartan 斯巴达浏览器可算是吊足了玩家们的胃口!如今,在最新的 Windows 10 Build 10049 版本起,它终于正式登场了。 斯巴达浏览器搭载了全新的渲染引擎、新的用户界面并集成了 Cortana 语音助手。功能上新增了稍后阅读列表、阅读视图、F12开发者工具、支持网页注释 (手写涂鸦),可以保存到 O