NSStream

流是位数据通过通信路径的连续传送序列。它是单向的,从一个应用程序的角度,流可以是输入流(读操作流)或者输出流(写操作流),除了基于文件的流之外,其余的都是non-seekable的。一旦流数据被提供或者被使用,数据就不能够从流中获取到。

Cocoa包括三种与流有关的类:NSStream,NSInputStream,NSOutputStream. NSStream是抽象类,它定义了流对象的基本接口和属性。NSInputStream和NSOutputStream是NSStream的子类,它们实现了输入流和输出流的基本操作。你可以为存储在内存中,向文件或者C buffer写的流数据创建NSOutputStream对象;可以为从NSData对象和文件中读取的流数据创建NSInputStream对象;也可以在网络套接字的两端创建NSInputStream和NSOutputStream对象,通过流对象,你可以不用一次性将所有的流数据加载到内存中。下图是就输入流和输出流对象的源和目的地为依据对输入流和输出流的分类:

NSStream及其子类进行的是比较底层的开发,对于某些特殊的需求如果有顶层的Cocoa API更加适合的话(比如NSURL,NSFileHandle),那么就用顶层的API进行编程。

流对象有许多属性,大多数属性都和网络安全及其配置有关,也就是SSL和SOCKS代理信息。另外有两个重要的属性,一个是NSStreamDataWrittenToMemoryStreamKey,对于一个输出流它可以用来获取到写入内存中的数据。另一个是NSStreamFileCurrentOffsetKey,对于一个基于文件的流,可以用它操作读或者写的位置。

每个流对象都有一个与其相关联的delegate,如果其delegate没有显示的设置,那么这个流对象自身成为其delegate(对于自定义子类的话这是一个很有用的约定)。流对象调用它唯一的delegate方法stream:handleEvent:来处理所有与stream-related事件。对于传入参数中的events事件,它指示了什么时候输入流中有数据可供读入,什么时候输出流中有空间可供数据写入。对于这两个事件中的NSStreamEventHasBytesAvailable事件,delegate向该stream发送read:maxLength:消息从流中读取数据,对于NSStreamHasSpaceAvailable事件,delegate向该stream发送write:maxlength:向流中写入数据。

NSStream是建立在Core Foundation的CFStream层之上的。这层紧密的关系意味着NSStream的具体子类-NSInputStream和NSOutputStream与Core Foundation中的CFReadStream和CFWriteStream是一一对应的。尽管Cocoa和Core Foundation的stream APIs有很大的相似性,但是它们的实现却不尽相同,Cocoa stream类使用delegate模式来实现异步操作(比如将其布置在run loop之上),而Core Foundation使用客户端的回调。Core Foundation的stream类型设置的是client(在Core Foundation中叫做context),NSStream中设置的delegate,这是两个不同的概念,不应该把设置delegate和设置context搞混淆。

相比CFStream而言,NSStream有更强的可扩展性,你可以生成NSStream,NSInputStream,NSOutputStream的子类来自定义其属性和方法。For example, you could create an input stream that maintains statistics on the bytes it reads; or you could make a NSStream subclass whose instances can seek through their stream, putting back bytes that have been read. NSStream has its own set of required overrides, as do NSInputStream and NSOutputStream.

NSInputStream

原文:Reading From Input Streams

ios cocoa 编程,从NSInputStream中读入数据包括几个步骤:

1.从数据源创建和初始化一个NSInputStream实例

2.将输入流对象配置到一个run loop,open the stream

3. 通过流对象的delegate函数处理事件

4. 当所有数据读完,进行流对象的内存处理

一,使用流对象的准备工作

在使用NSInputStream对象之前你必须有流的数据源,数据源的类型可以是文件,NSData对象,或者一个网络套接字。

NSInputStream的初始化函数和工厂方法可以从NSData和文件创建和初始化一个NSInputStream的实例。下面的例子是从文件创建一个NSInputStream的实例:

  1. - (void)setUpStreamForFile:(NSString *)path {
  2. // iStream is NSInputStream instance variable
  3. iStream = [[NSInputStream alloc] initWithFileAtPath:path];
  4. [iStream setDelegate:self];
  5. [iStream scheduleInRunLoop:[NSRunLoop currentRunLoop]
  6. forMode:NSDefaultRunLoopMode];
  7. [iStream open];
  8. }

上面的例子显示,当你创建对象之后你应该设置其delegate。当把NSInputStream对象配置到一个run loop,并且有与流相关的事件(例如流中有可读数据)发生时,该对象会收到stream:handleEvent:消息。

在你open stream之前,给流对象发送一个scheduleInRunLoop:forMode:消息,来将该对象配置到一个run loop接收stream events。这样,当流中没有数据可读时可以避免delegate阻塞。如果流是发生在另一个线程,你需要确认该流对象是配置在那个线程的run loop中。你不应该尝试从一个除了包含该流对象的run loop的线程的其他线程中对流进行操作。最后,对NSInputStream对象发送open消息开始对输入数据的流操作。

二,处理Stream Events

当你对一个流对象发送open消息之后,你可以查找到它的当前状态。通过下面的消息可以知道流对象中是否有数据可读,以及任何错误的属性:

  • streamStatus

  • hasBytesAvailable

  • streamError

返回的状态是一个NSStreamStatus常量,它可以指示流对象是处于opening,reading,或者at the end of the stream等等。返回的错误是一个NSError对象,它封装了可能发生的所有错误信息。

重要的是,一旦 open 流对象,流对象会一直向其delegate发送stream:handleEvent: 消息直到到达了流对象的末尾。这些消息的参数中包含一个指示流事件类型的NSStreamEvent常量。对NSInputStream对象而言,最常用的事件类型是NSStreamEventOpenCompleted,NSStreamEventHasBytesAvailable,NSStreamEventEndEncountered。我们尤其感兴趣的应该是NSStreamEventHasBytesAvailable事件。下面的例子就是一个处理NSStreamEventHasBytesAvailable事件的好的方法:

  1. - (void)stream:(NSStream *)stream handleEvent:(NSStreamEvent)eventCode {
  2. switch(eventCode) {
  3. case NSStreamEventHasBytesAvailable:
  4. {
  5. if(!_data) {
  6. _data = [[NSMutableData data] retain];
  7. }
  8. uint8_t buf[1024];
  9. unsigned int len = 0;
  10. len = [(NSInputStream *)stream read:buf maxLength:1024];
  11. if(len) {
  12. [_data appendBytes:(const void *)buf length:len];
  13. // bytesRead is an instance variable of type NSNumber.
  14. [bytesRead setIntValue:[bytesRead intValue]+len];
  15. } else {
  16. NSLog(@"no buffer!");
  17. }
  18. break;
  19. }
  20. // continued
  21. }

stream:handleEvent: 函数使用switch语句来判别NSStreamEvent常量,当这个常量是MSStreamEventHasBytesAvailable的时候,delegate函数会lazy create 一个NSMutableData对象_data来接收读取的数据。然后声明一个大小为1024的uint8_t类型数组buf,调用read:maxLength:函数从stream中读取指定大小的数据到buf中,如果读取成功,delegate将会将读取到的数据添加到NSMutableData对象_data中,并且更新总的读取到的数据bytesRead.

至于一次从stream中读取多大的数据,一般来说,使用一些常用的数据大小规格,比如说512Bytes,1kB,4kB(一个页面大小)。

三,处理stream object

当NSInputStream对象到达steam的末尾的时候,它会向stream:handleEvent:函数发送一个NSStreamEventEndEncountered事件类型常量,delegate函数应该做出与准备使用流对象相反的操作,也就是说,需要关闭流对象,从run loop中移除,最终释放流对象。如下面的代码所示:

    1. - (void)stream:(NSStream *)stream handleEvent:(NSStreamEvent)eventCode
    2. {
    3. switch(eventCode) {
    4. case NSStreamEventEndEncountered:
    5. {
    6. [stream close];
    7. [stream removeFromRunLoop:[NSRunLoop currentRunLoop]
    8. forMode:NSDefaultRunLoopMode];
    9. [stream release];
    10. stream = nil; // stream is ivar, so reinit it
    11. break;
    12. }
    13. // continued ...
    14. }

}

NSOutputStream

译自:Writing To Output Streams

使用NSOutputStream实例需要以下几个步骤:

1,使用存储写入数据的存储库创建和初始化一个NSOutputSteam实例,并且设置它的delegate。

2,将这个流对象布置在一个runloop上并且open the stream。

3,处理流对象向其delegate发送的事件消息。

4,如果流对象向内存中写入了数据,那么可以通过使用NSStreamDataWrittenToMemoryStreamKey属性获取数据。

5,当没有数据可供写入时,清理流对象。

一,使用流对象的准备工作

使用NSOutputStream对象之前你必须指定数据写入的流的目标位置,输出流对象的目标位置可以是file,C buffer, application memory,network socket。

NSOutputStream的初始化方法和工厂方法可以使用a file,a buffer, memory来创建和初始化实例,下面的代码初始化了一个NSOutputStream实例,用来向 application memory 写入数据。

  1. - (void)createOutputStream
  2. {
  3. NSLog(@"Creating and opening NSOutputStream...");
  4. // oStream is an instance variable
  5. oStream = [[NSOutputStream alloc] initToMemory];
  6. [oStream setDelegate:self];
  7. [oStream scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
  8. [oStream open];
  9. }

上面的代码显示,在你初始化一个NSOutputStream对象之后应该设置它的delegate(通常是self),当流对象有 有空间可供数据写入 之类的与流有关的事件消息发送时,delegate会收到从NSOutputStream对象发送来的消息。

当你在open the stream对象之前,向流对象发送scheduleInRunLoop:forMode:消息使其在一个runloop上可以接收到stream events,这样,当流对象不能接收更多数据的时候,可以使delegate避免阻塞。当streaming发生在另外一个线程时,你必须将流对象布置在那个线程的run loop上,You should never attempt to access a scheduled stream from a thread different than the one owning the stream’s run loop. 最后 open the stream 开始数据向 NSOutputStream对象传送。

二,处理 Stream Events

当你向流对象发送open消息之后,你可以通过以下消息获取到流对象的状态,比如说当前是否有空间可供数据写入以及其他错误信息的属性。

  • streamStatus
  • hasSpaceAvailable
  • streamError

返回的状态是NSStreamStatus常量,它指示流当前的状态是opening,writing,at the end of the stream等等,返回的错误是NSError对象,它封装的是所有错误的信息。

重要的是,一旦open the stream,只要delegate持续想流对象写入数据,流对象就是一直向其delegate发送stream:handleEvent:消息,直到到达了流的末尾。这些消息中包含一个NSStreamEvent常量参数来指示事件的类型。对于一个NSOutputStream对象,最常见的事件类型是NSStreamEventOpenCompleted,NSStreamEventHasSpaceAvailable,NSStreamEventEndEncountered,delegate通常对NSStreamEventHasSpaceAvaliable事件最感兴趣。下面的代码就是处理NSStreamEventHasSpaceAvaliable事件的一种方法:

  1. - (void)stream:(NSStream *)stream handleEvent:(NSStreamEvent)eventCode
  2. {
  3. switch(eventCode)
  4. {
  5. case NSStreamEventHasSpaceAvailable:
  6. {
  7. uint8_t *readBytes = (uint8_t *)[_data mutableBytes];
  8. readBytes += byteIndex; // instance variable to move pointer
  9. int data_len = [_data length];
  10. unsigned int len = ((data_len - byteIndex >= 1024) ? 1024 : (data_len-byteIndex));
  11. uint8_t buf[len];
  12. (void)memcpy(buf, readBytes, len);
  13. len = [stream write:(const uint8_t *)buf maxLength:len];
  14. byteIndex += len;
  15. break;
  16. }
  17. // continued ...
  18. }
  19. }

在stream:handleEvent:的实现中使用switch语句来判别NSStreamEvent常量,当这个常量是NSStreamEventHasSpacesAvailable的时候,delegate从NSMutableData对象_data中获取数据,并且将其指针转化为适合当前操作的类型u_int8.下一步计算即将进行写操作的字节数(是1024还是所有剩余的字节数),声明一段相应大小的buffer,向该buffer写入相应大小的数据,然后delegate调用流对象write:maxLength:方法将buffer中的数据置入output stream中,最后更新byteIndex用于下一次的读取操作。

如果delegate收到NSStreamEventHasSpacesAvailable事件消息但是没有向stream里写入任何数据,它不会从runloop再接收到space-available的事件消息直到NSOutputStream对象接收到数据,这样由于space-available事件该run loop会重新启动。如果这种情况很有可能在你的程序设计中出现,在收到NSStreamEventHasSpaceAvailable消息并且没有向该stream中写入数据时可以在delegate中设置一个标志位flag,之后,当存在更多的数据需要写入时,先检查该标志位,如果该标志位被设置,那么直接向output-stream实例写入数据。

对于一次向output-stream实例中写入多少数据没有严格的限定,一般情况下使用一些合理值,如512Bytes,1kB,4kB(一个页面大小)。

当在向stream中写数据时NSOutputStream对象发生错误,它会停止streaming并且使用NSStreamEventErrorOccurred消息通知其delegate。

三,清理 Stream Object

当一个NSOutputStream对象结束向一个output stream写入数据,它通过stream:handleEvent:消息向delegate发送NSStreamEventEndEncountered事件消息,这个时候delegate应该清理 stream object,先关闭该stream object,从run loop中移除,释放该stream object。此外,如果NSOutputStream对象的目的存储库是application memory(也就是,你通过initToMemory方法或者其工厂方法outputStreamToMemory创建的该对象),现在就可以从内存中获取数据了。下面的代码实现的清理 stream object的工作:

  1. - (void)stream:(NSStream *)stream handleEvent:(NSStreamEvent)eventCode
  2. {
  3. switch(eventCode)
  4. {
  5. case NSStreamEventEndEncountered:
  6. {
  7. NSData *newData = [oStream propertyForKey:NSStreamDataWrittenToMemoryStreamKey];
  8. if (!newData) {
  9. NSLog(@"No data written to memory!");
  10. }
  11. else {
  12. [self processData:newData];
  13. }
  14. [stream close];
  15. [stream removeFromRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
  16. [stream release];
  17. oStream = nil; // oStream is instance variable
  18. break;
  19. }
  20. // continued ...
  21. }
  22. }

通过向NSOutputStream对象发送propertyForKey:消息获取从流向内存中写入的数据,设定key的值为NSStreamDataWrittenToMemoryStreamKey,该stream object将数据返回到一个NSData对象中。

NSStream的更多相关文章

  1. iOS - OC NSStream 文件流

    前言 @interface NSStream : NSObject @interface NSOutputStream : NSStream 1.文件流的使用 NSString *filePath = ...

  2. NSStream实现发送和接受数据

    一.基本概念在iOS中以NSStream(流)来发送和接收数据,可以设置流的代理,对流状态的变化做出相应.1连接建立2接收到数据3连接关闭NSStream:数据流的父类,用于定义抽象特性,例如:打开. ...

  3. iOS -- 原生NSStream实现socket

    - (void)startSocket:(NSString *)address andPort:(int)port { CFReadStreamRef readRef; CFWriteStreamRe ...

  4. NSStream 流式思想

    流式思想的本质是将数据或信号看作流.流的管理者NSStream看作管道. 内容包含两方面: 1.流的建立:源.目的地: 2.流的管理:状态事件与数据事件. 本质上是建立联系.处理数据.处理状态.

  5. NSStream文件流

    1.文件流的使用 NSString *filePath = @"/Users/JHQ0228/Desktop/test.txt"; NSData *data = [@"h ...

  6. 【原】AFNetworking源码阅读(五)

    [原]AFNetworking源码阅读(五) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 上一篇中提及到了Multipart Request的构建方法- [AFHTTP ...

  7. RunLoop 总结:RunLoop的应用场景(一)

    参考资料 好的书籍都是值得反复看的,那好的文章,好的资料也值得我们反复看.我们在不同的阶段来相同的文章或资料或书籍都能有不同的收获,那它就是好文章,好书籍,好资料.关于iOS 中的RunLoop资料非 ...

  8. iOS并发编程笔记【转】

    线程 使用Instruments的CPU strategy view查看代码如何在多核CPU中执行.创建线程可以使用POSIX 线程API,或者NSThread(封装POSIX 线程API).下面是并 ...

  9. iOS-技巧性总结

    1.AFN与ASI对比 -- AFN1. 基于 NSURLConnection & NSURLSession 进行的封装2. 使用简单3. 提供了自动的序列化 & 反序列化支持! AF ...

随机推荐

  1. 【MySql】5.6.14版本的安装和测试

    当前状态:apache2.4.6和php5.5.6已经安装成功: mysql的安装和测试: 一.安装mysql5.6.14,参考http://wenku.baidu.com/link?url=_0jk ...

  2. 计算系数 (codevs 1137) 题解

    [问题描述] 给定一个多项式(ax + by)^k,给定a.b.k.n.m,请求出多项式展开后x^n y^m项的系数. [样例输入] 1 1 3 1 2 [样例输出] 3 [解题思路] 本题为NOIP ...

  3. SQL Server 基础:Case两种用法

    测试数据 1).等值判断->相当于switch case select S#,C#,C#=( case C# when 1 then '语文' when 2 then '数学' when 3 t ...

  4. openerp 常见问题 OpenERP为什么选择了时区后时间还是不对?(转载)

    OpenERP为什么选择了时区后时间还是不对? 原文地址:http://cn.openerp.cn/%E4%B8%BA%E4%BB%80%E4%B9%88%E9%80%89%E6%8B%A9%E4%B ...

  5. Python学习教程(learning Python)--3.3.4 Python的if-elif-else语句

    Python的if-elif-else语句用于多种条件判断后选择某个语句块执行.该语句可以利用一系列条件表达式进行检查,并在某个表达式为真的情况下执行相应的代码.需要注意的是,虽然if/elif/el ...

  6. Web Design:给实验室UI们的一堂课(下)

    [讲稿]From top to down,自顶向下哈,首部栏.导航栏之后一般是页面的主模块,也就是Body部分,这一块儿才是你网站的核心内容,文章.新闻.动态.数据.图表.相册等都是在这儿体现出来.在 ...

  7. 【微信平台&后台管理】第一个外包项目:XX科技城微信平台项目总结

    苍天有眼啊,学了半年的网站开发终于派上用处,终于能赚钱了啊. 这个项目是和学长一起做的,项目的甲方是大庆某房地产土豪,项目要求就是搭建一整套的微信平台和微信平台管理系统,具体要求就是:回复关键字能拿到 ...

  8. 安装使用rspec

    一,安装ruby. 二,运行命令,安装rspec的gem包: gem install rspec 会看到如下的结果: Fetching: rspec-core-2.14.7.gem (100%) Fe ...

  9. .NET开源工作流RoadFlow-流程设计-流程步骤设置-基本设置

    流程属性设置完成后点击确定之后,即可进行流程步骤设置了. 点击工具栏上的步骤按钮,即可添加一个新步骤. 在新步骤图形上双击即可弹出该步骤相应属性设置框. 步骤ID:系统自动为该步骤生成的唯一ID. 步 ...

  10. Mybatis 的日志管理

    Mybatis通过日志工厂提供日志信息,Mybatis内置的日志模版是log4j,commons.log,jdk log也可以通过slf4j简单日志模版结合log4j使用日志信息输出.具体选择哪个日志 ...