无法在使用CFStreamCreatePairWithSocket成功创建的CFWriteStream上写入

时间:2012-03-05 17:50:23

标签: ios sockets stream ios-simulator

由于iOS开发框架中的CFWrite / ReadStream,我尝试建立双向异步通信时遇到问题。

我开始创建套接字连接,一切似乎都是正确的:

CFSocketContext socketCtx = {0, self, NULL, NULL, NULL};
socket = CFSocketCreate(NULL, PF_INET, SOCK_STREAM, IPPROTO_TCP,kCFSocketConnectCallBack, (CFSocketCallBack)onConnectOrDisconnect, &socketCtx);

CFSocketEnableCallBacks(socket, kCFSocketConnectCallBack);
struct sockaddr_in addr;
memset(&addr, 0, sizeof(addr));
addr.sin_len = sizeof(addr);
addr.sin_family = AF_INET;
addr.sin_port = htons(appDelegate.portSocket);
addr.sin_addr.s_addr = inet_addr([adress cStringUsingEncoding:NSASCIIStringEncoding]);
CFDataRef connectAddr = CFDataCreate(NULL, (unsigned char *)&addr, sizeof(addr));   
CFRunLoopSourceRef sourceRef = CFSocketCreateRunLoopSource(kCFAllocatorDefault, socket, 0);
CFRunLoopAddSource(CFRunLoopGetCurrent(), sourceRef, kCFRunLoopCommonModes);
CFRelease(sourceRef);
CFSocketConnectToAddress(socket, connectAddr, -1);

然后,在我的连接回调中,我创建了两个流,一个用于读取,另一个用于写入:

void onConnectOrDisconnect(CFSocketRef socket, CFSocketCallBackType type, CFDataRef address,
                           const void *data, void *info)
{
    TCPClient * clientTCP = (TCPClient *)info;
    if(data==NULL)
    {
        [clientTCP ConnectionSuccess];
    }else {
        SInt32 *error = (SInt32*)data;
        NSLog(@"Connection error : %d", *error);
        StreamErrorOccured(clientTCP);
    }
}
- (void) ConnectionSuccess
{
    CFStreamCreatePairWithSocket(kCFAllocatorDefault, CFSocketGetNative(socket), &readStream, &writeStream);
    CFReadStreamSetProperty(readStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
    CFWriteStreamSetProperty(writeStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
    CFStreamClientContext dataStreamContext = {0, self, NULL, NULL, NULL};  
    CFOptionFlags events = kCFStreamEventHasBytesAvailable | kCFStreamEventOpenCompleted;
    if (CFReadStreamSetClient(readStream, events, readStreamEventCallBack , &dataStreamContext)) 
    {
        CFReadStreamScheduleWithRunLoop(readStream, CFRunLoopGetCurrent(), kCFRunLoopCommonModes);
    }
    if (CFWriteStreamSetClient(writeStream, kCFStreamEventOpenCompleted | kCFStreamEventCanAcceptBytes | kCFStreamEventErrorOccurred | kCFStreamEventEndEncountered,
                               writeStreamEventCallBack, &dataStreamContext))
    {
        CFWriteStreamScheduleWithRunLoop(writeStream, CFRunLoopGetCurrent(), kCFRunLoopCommonModes);
    }else {
        NSLog(@"can't create write stream client");
    }

    CFReadStreamOpen(readStream);
    CFWriteStreamOpen(writeStream);

    NSLog(@"Connection Success");
}

请注意,我在kCFStreamEventCanAcceptBytes事件和kCFStreamEventOpenCompleted事件上注册了一个回调。在这个回调中,我可以看到kCFStreamEventOpenCompleted发生了,但kCFStreamEventCanAcceptBytes永远不会发生!

当我尝试在我的流上写时,我检查流的可用性,这要归功于:CFWriteStreamCanAcceptBytes(writeStream)。此函数始终返回false。

如果我尝试在不调用CFWriteStreamCanAcceptBytes的情况下在流上写入,则写操作将永远循环。 我的问题是:  1.为什么我没有收到任何kCFStreamEventCanAcceptBytes事件?  2.为什么我不能写在我的流上?

我看到我的问题完全相同:SIMILAR QUESTION但是没有人回复他。

也许我会更幸运!

干杯,

1 个答案:

答案 0 :(得分:0)

我不知道答案,但我也遇到了这个问题。 解决方案是在writeStreamEventCallBack中接收事件'kCFStreamEventCanAcceptBytes',并在此回调函数上写入数据。 实际上问题是,如果你没有在'kCFStreamEventCanAcceptBytes'事件上发送任何数据,你下次不能发送数据。

Apple开发文档的

A page说:

当您使用CFStream API进行网络连接时,套接字上的读写操作可能会被阻止。为了防止阻塞:

  1. 调用CFReadStreamSetClient和CFWriteStreamSetClient进行注册以接收与流相关的事件通知。

  2. 调用CFReadStreamScheduleWithRunLoop和CFWriteStreamScheduleWithRunLoop在运行循环上安排流以接收与流相关的事件通知。

  3. 调用CFReadStreamOpen和CFWriteStreamOpen打开每个流。

  4. 在收到kCFStreamEventHasBytesAvailable通知后才能读取。仅在收到kCFStreamEventCanAcceptBytes通知后写入。