未调用CFSocket接受回调.我错过了什么?

pb5*_*593 5 sockets objective-c callback ios cfsocket

我目前正在Objective-C中编写一个简单的TCP服务器类,它应该能够绑定并侦听特定端口.特别是我有一个-(void) start创建CFSockets 的函数,将它绑定到指定的端口并将其添加到运行循环中.它看起来像这样:

- (void) start {
    //This function is called after the initialiser

    if([self isStarted]) //if already started, return without doing anything
        return;


    //create a socket
    CFSocketContext socketCtxt = {0, (__bridge void *)self, NULL, NULL, NULL};
    socket = CFSocketCreate(kCFAllocatorDefault, PF_INET, SOCK_STREAM, IPPROTO_TCP, kCFSocketAcceptCallBack, (CFSocketCallBack)&SocketAcceptCallback, &socketCtxt);


    NSAssert(socket, @"Error: Cannot create a socket...");


    int yes = 1;
    setsockopt(CFSocketGetNative(socket), SOL_SOCKET, SO_REUSEADDR, (void *)&yes, sizeof(yes));


//bind it to the desired port
    struct sockaddr_in serverAddress;
    socklen_t nameLen = 0;
    nameLen = sizeof(serverAddress);
    memset(&serverAddress, 0, sizeof(serverAddress));

    serverAddress.sin_len = nameLen;
    serverAddress.sin_family = AF_INET;
    serverAddress.sin_port = htons(self.port);
    serverAddress.sin_addr.s_addr = htonl(INADDR_ANY);

    NSData* address4 = [NSData dataWithBytes:&serverAddress length:nameLen];


    if(kCFSocketSuccess != CFSocketSetAddress(socket, (CFDataRef)address4)) {
        if (socket) CFRelease(socket);
        socket = NULL;
        NSAssert(NO, @"Unable to bind the created socket to the desired port");
    }


    //add to the current run loop
    CFRunLoopSourceRef rls = CFSocketCreateRunLoopSource(kCFAllocatorDefault, socket, 0);


    assert(rls!=0);

    CFRunLoopAddSource(CFRunLoopGetCurrent(), rls, kCFRunLoopCommonModes);


    //done!

}
Run Code Online (Sandbox Code Playgroud)

我还定义了这样的回调函数SocketAcceptCallback:

static void SocketAcceptCallback(CFSocketRef sock, CFSocketCallBackType type, CFDataRef address, const void *data, void *info) {
    //this method is called after a connection is accepted
    //here we bind the received connection to a pair of
    //NSStreams
    if(type == kCFSocketAcceptCallBack){
        CFSocketNativeHandle nativeSocketHandle = *(CFSocketNativeHandle *)data;

        struct sockaddr_in peerAddress;
        socklen_t peerLen = sizeof(peerAddress);
        NSString * peer = nil;

        if (getpeername(nativeSocketHandle, (struct sockaddr *)&peerAddress, (socklen_t *)&peerLen) == 0) {
            peer = [NSString stringWithUTF8String:inet_ntoa(peerAddress.sin_addr)];
        } else {
            peer = @"Generic Peer";
        }


        NSLog(@"%@ has connected", peer);


        //Find corresponding SocketServer object -- it is passed to the callback in the INFO pointer
        SocketServer *obj = (__bridge SocketServer*)info;

        CFReadStreamRef readStream = NULL;
        CFWriteStreamRef writeStream = NULL;
        CFStreamCreatePairWithSocket(NULL, nativeSocketHandle, &readStream, &writeStream);

         if (readStream && writeStream) {
             CFReadStreamSetProperty(readStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);
             CFWriteStreamSetProperty(writeStream, kCFStreamPropertyShouldCloseNativeSocket, kCFBooleanTrue);


             //bridge to NSStreams
             obj.inputStream = (NSInputStream *)readStream;
             obj.outputStream = (NSOutputStream *)writeStream;

             //set delegates
             [obj.inputStream setDelegate:obj];
             [obj.outputStream setDelegate:obj];

             //schedule the runloops
             [obj.inputStream scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
             [obj.outputStream scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];

             //open connections
             [obj.inputStream open];
             [obj.outputStream open];


         } else {
             // on any failure, need to destroy the CFSocketNativeHandle
             // since we are not going to use it any more
             close(nativeSocketHandle);
         }

        /*
         if (readStream) CFRelease(readStream);
         if (writeStream) CFRelease(writeStream); */
    }
}
Run Code Online (Sandbox Code Playgroud)

在回调中,我将连接桥接到一对NSStreams,我在我的类中使用它作为成员字段,我用它来读写.

现在问题是:我试图运行这个服务器并通过telnet连接到它.它连接,但上面的回调没有被调用(我已通过在其中添加诊断NSLog来检查).我怀疑我没有正确地将套接字添加到运行循环中,但我自己也找不到任何错误.关于它可能是什么的任何想法?