1

我正在尝试使用 CFStream 进行简单的 HTTP 通信,但是使用下面的代码,即使我提供了无效的 URL (wwwww.yahoo.com),我仍然可以成功连接。

有人可以指出我这段代码有什么问题吗?我预计对 CFReadStreamOpen() 的调用会失败,但即使 URL 错误,它也会成功。

我还没有找到一个简单的 CFNetwork 示例,它可以在 ARC 上运行 HTTP(不使用 NSURL)。如果有人知道这样的示例代码,请告诉我。

- (void) test
{

    CFStringRef m_host = CFStringCreateWithCString(NULL, "wwwww.yahoo.com", kCFStringEncodingASCII);

    int m_port = 80;
    CFHostRef host;

    NSLog(@"Attempting to (re)connect to %@:%d", m_host, m_port);

    {

        CFReadStreamRef     m_in = NULL;
        CFWriteStreamRef    m_out = NULL;

        host = CFHostCreateWithName(kCFAllocatorDefault, (CFStringRef)m_host);
        if (!host)
        {
            NSLog(@"Error resolving host %@", m_host);
        }

        CFStreamCreatePairWithSocketToCFHost(kCFAllocatorDefault, host , m_port, &m_in, &m_out);
        CFRelease(host);

        CFStreamClientContext context = {0, nil,nil,nil,nil}; 


        if (CFReadStreamSetClient(m_in, kCFStreamEventHasBytesAvailable | kCFStreamEventErrorOccurred | kCFStreamEventEndEncountered, networkReadEvent, &context))
        {
            CFReadStreamScheduleWithRunLoop(m_in, CFRunLoopGetCurrent(),kCFRunLoopCommonModes);
        }

        if (CFWriteStreamSetClient(m_out, kCFStreamEventErrorOccurred | kCFStreamEventEndEncountered, networkWriteEvent, &context))
        {
            CFWriteStreamScheduleWithRunLoop(m_out, CFRunLoopGe   tCurrent(),kCFRunLoopCommonModes);
        }        

        BOOL success = CFReadStreamOpen(m_in);
        CFErrorRef error = CFReadStreamCopyError(m_in);
        if (!success || (error && CFErrorGetCode(error) != 0))
        {
            NSLog(@"Connect error %s : %ld", CFErrorGetDomain(error), CFErrorGetCode(error));
            [NSThread sleepForTimeInterval:5.0];
        }

        success = CFWriteStreamOpen(m_out);
        error = CFReadStreamCopyError(m_in);
        if (!success || (error && CFErrorGetCode(error) != 0))
        {
            NSLog(@"Connect error %s : %d", CFErrorGetDomain(error), CFErrorGetCode(error));
            [NSThread sleepForTimeInterval:5.0];
        }
        else 
        {
            NSLog(@"Connected");
        }

    }
}
4

1 回答 1

0

对于像您这样的异步代码,您需要设置委托并在那里捕获错误。查看http://iphonedevsdk.com/forum/iphone-sdk-development/17542-cfstream-catch-error-in-connection.html

根据 Apple Developer 网站,“如果流可以在后台打开而不会阻塞,则此函数始终返回 true。”。这意味着打开流的请求是在不同的线程上发送的,因此当您调用它并在后台线程上启动连接过程时,该函数会返回 true。然后,各个流上的所有读/写调用都会阻塞,直到后台线程完成打开流。如果您在“CFReadStreamOpen(m_in);”上设置断点,您将能够看到这一点 看看那一行执行后会发生什么。希望这能回答你的问题。

于 2014-10-24T05:48:13.603 回答