ios - 带循环的 NSOperationQueue 和带委托(delegate)的类

标签 ios xcode multithreading nsoperation nsoperationqueue

我是 NSOperationQueue 的新手,我正在尝试创建一个可以 ping 我网络中所有主机的队列。 首先,我使用的是 Apple 的类 SimplePing.h,它一次只对一台主机执行 ping 操作。 这个类有一些委托(delegate)来通知主类 ping 是否成功。 现在,在我的例子中,我想对从 192.168.1.1192.168.1.254 的所有主机执行 ping 操作,所以这是我的代码:

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.

    // Create a new NSOperationQueue instance.
    operationQueue = [NSOperationQueue new];



    for (int i=1; i<254; i++) {

        NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithTarget:self
                                                                                selector:@selector(pingHost:)
                                                                                  object:[NSString stringWithFormat:@"192.168.1.%d",i]];
        // Add the operation to the queue and let it to be executed.
        [operationQueue addOperation:operation];


    }


}

-(void)pingHost:(NSString*)ip{

    ping = [SimplePing simplePingWithHostName:ip];  
    self.ping.delegate=self;
    [ping start];


};
- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

// When the pinger starts, send the ping immediately
- (void)simplePing:(SimplePing *)pinger didStartWithAddress:(NSData *)address {


    [pinger sendPingWithData:nil];
}

- (void)simplePing:(SimplePing *)pinger didSendPacket:(NSData *)packet{


    [self performSelector:@selector(didNotReceivedAnswer) withObject:nil afterDelay:2];

    NSLog(@"didsendpacket");

}
- (void)simplePing:(SimplePing *)pinger didFailWithError:(NSError *)error {
    NSLog(@"didFail");

    [self didNotReceivedAnswer];

}

- (void)simplePing:(SimplePing *)pinger didFailToSendPacket:(NSData *)packet error:(NSError *)error {
    NSLog(@"didfailtosendpacket");

    [self didNotReceivedAnswer];

}

- (void)simplePing:(SimplePing *)pinger didReceivePingResponsePacket:(NSData *)packet {

    NSLog(@"didreceivesresponse");

}

- (void)simplePing:(SimplePing *)pinger didReceiveUnexpectedPacket:(NSData *)packet{

    [self didNotReceivedAnswer];
    NSLog(@"didreceiveunexpected");

}

//Helper for delegate
-(void)didNotReceivedAnswer{
}

然后我得到这个错误:

Assertion failed: (self->_host == NULL), function -[SimplePing start], file /Users/Mike/Documents/Xcode/Unused/PingTest/PingTest/SimplePing.m, line 574. Assertion failed: (self->_host == NULL), function -[SimplePing start], file /Users/Mike/Documents/Xcode/Unused/PingTest/PingTest/SimplePing.m, line 574. Assertion failed: (self->_host == NULL), function -[SimplePing start], file /Users/Mike/Documents/Xcode/Unused/PingTest/PingTest/SimplePing.m, line 574. Assertion failed: (self->_host == NULL), function -[SimplePing start], file /Users/Mike/Documents/Xcode/Unused/PingTest/PingTest/SimplePing.m, line 574. Assertion failed: (self->_host == NULL), function -[SimplePing start], file /Users/Mike/Documents/Xcode/Unused/PingTest/PingTest/SimplePing.m, line 574.

知道这里出了什么问题吗?

最佳答案

您的问题是您必须保留对每个 ping 对象的引用。截至目前,您已将 ping 声明为类变量,因此每次调用 pingHost 时,它都会重新初始化,而前一个会丢失。这就是为什么您得到 (self->_host == NULL) 断言失败的原因。

您将需要使用一个NSOperationQueue 和一个子类NSOperation 来处理这个问题。拥有 NSOperation 将允许该 NSOperation 的每个实例都保留其唯一的 ping 对象。

我很快就完成了这个,所以你可以解决它。理想情况下,您希望为 PingOperation 类创建一个委托(delegate),这样您就可以从中获得对主线程的回调。

PingOperation.h

#import <Foundation/Foundation.h>

@interface PingOpertion : NSOperation

-(id)initWithHostName:(NSString*)hostName;

@end

PingOperation.m

#import "PingOpertion.h"
#include "SimplePing.h"

#include <sys/socket.h>
#include <netdb.h>

static NSString * DisplayAddressForAddress(NSData * address)
// Returns a dotted decimal string for the specified address (a (struct sockaddr)
// within the address NSData).
{
    int         err;
    NSString *  result;
    char        hostStr[NI_MAXHOST];

    result = nil;

    if (address != nil) {
        err = getnameinfo([address bytes], (socklen_t) [address length], hostStr, sizeof(hostStr), NULL, 0, NI_NUMERICHOST);
        if (err == 0) {
            result = [NSString stringWithCString:hostStr encoding:NSASCIIStringEncoding];
            assert(result != nil);
        }
    }

    return result;
}

@interface PingOpertion () <SimplePingDelegate>

@property NSString *hostName;

@property (nonatomic, strong, readwrite) SimplePing *   pinger;
@property (nonatomic, strong, readwrite) NSTimer *      sendTimer;
@end


@implementation PingOpertion

@synthesize pinger    = _pinger;
@synthesize sendTimer = _sendTimer;


-(id)initWithHostName:(NSString*)hostName {
    if (self = [super init]) {
        self.hostName = hostName;
    }
    return self;
}

- (void)main {
    // a lengthy operation
    @autoreleasepool {
        assert(self.pinger == nil);

        self.pinger = [SimplePing simplePingWithHostName:self.hostName];
        assert(self.pinger != nil);

        self.pinger.delegate = self;
        [self.pinger start];

        do {
            [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
        } while (self.pinger != nil);
    }
}


- (NSString *)shortErrorFromError:(NSError *)error
// Given an NSError, returns a short error string that we can print, handling
// some special cases along the way.
{
    NSString *      result;
    NSNumber *      failureNum;
    int             failure;
    const char *    failureStr;

    assert(error != nil);

    result = nil;

    // Handle DNS errors as a special case.

    if ( [[error domain] isEqual:(NSString *)kCFErrorDomainCFNetwork] && ([error code] == kCFHostErrorUnknown) ) {
        failureNum = [[error userInfo] objectForKey:(id)kCFGetAddrInfoFailureKey];
        if ( [failureNum isKindOfClass:[NSNumber class]] ) {
            failure = [failureNum intValue];
            if (failure != 0) {
                failureStr = gai_strerror(failure);
                if (failureStr != NULL) {
                    result = [NSString stringWithUTF8String:failureStr];
                    assert(result != nil);
                }
            }
        }
    }

    // Otherwise try various properties of the error object.

    if (result == nil) {
        result = [error localizedFailureReason];
    }
    if (result == nil) {
        result = [error localizedDescription];
    }
    if (result == nil) {
        result = [error description];
    }
    assert(result != nil);
    return result;
}

- (void)runWithHostName:(NSString *)hostName
// The Objective-C 'main' for this program.  It creates a SimplePing object
// and runs the runloop sending pings and printing the results.
{
    assert(self.pinger == nil);

    self.pinger = [SimplePing simplePingWithHostName:hostName];
    assert(self.pinger != nil);

    self.pinger.delegate = self;
    [self.pinger start];

    do {
        [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
    } while (self.pinger != nil);
}

- (void)sendPing
// Called to send a ping, both directly (as soon as the SimplePing object starts up)
// and via a timer (to continue sending pings periodically).
{
    assert(self.pinger != nil);
    [self.pinger sendPingWithData:nil];
}

- (void)simplePing:(SimplePing *)pinger didStartWithAddress:(NSData *)address
// A SimplePing delegate callback method.  We respond to the startup by sending a
// ping immediately and starting a timer to continue sending them every second.
{
#pragma unused(pinger)
    assert(pinger == self.pinger);
    assert(address != nil);

    NSLog(@"pinging %@", DisplayAddressForAddress(address));

    // Send the first ping straight away.

    [self sendPing];

    // And start a timer to send the subsequent pings.

    assert(self.sendTimer == nil);
    self.sendTimer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(sendPing) userInfo:nil repeats:YES];
}

- (void)simplePing:(SimplePing *)pinger didFailWithError:(NSError *)error
// A SimplePing delegate callback method.  We shut down our timer and the
// SimplePing object itself, which causes the runloop code to exit.
{
#pragma unused(pinger)
    assert(pinger == self.pinger);
#pragma unused(error)
    NSLog(@"failed: %@", [self shortErrorFromError:error]);

    [self.sendTimer invalidate];
    self.sendTimer = nil;

    // No need to call -stop.  The pinger will stop itself in this case.
    // We do however want to nil out pinger so that the runloop stops.

    self.pinger = nil;
}

- (void)simplePing:(SimplePing *)pinger didSendPacket:(NSData *)packet
// A SimplePing delegate callback method.  We just log the send.
{
#pragma unused(pinger)
    assert(pinger == self.pinger);
#pragma unused(packet)
    NSLog(@"#%u sent", (unsigned int) OSSwapBigToHostInt16(((const ICMPHeader *) [packet bytes])->sequenceNumber) );
}

- (void)simplePing:(SimplePing *)pinger didFailToSendPacket:(NSData *)packet error:(NSError *)error
// A SimplePing delegate callback method.  We just log the failure.
{
#pragma unused(pinger)
    assert(pinger == self.pinger);
#pragma unused(packet)
#pragma unused(error)
    NSLog(@"#%u send failed: %@", (unsigned int) OSSwapBigToHostInt16(((const ICMPHeader *) [packet bytes])->sequenceNumber), [self shortErrorFromError:error]);
}

- (void)simplePing:(SimplePing *)pinger didReceivePingResponsePacket:(NSData *)packet
// A SimplePing delegate callback method.  We just log the reception of a ping response.
{
#pragma unused(pinger)
    assert(pinger == self.pinger);
#pragma unused(packet)
    NSLog(@"#%u received", (unsigned int) OSSwapBigToHostInt16([SimplePing icmpInPacket:packet]->sequenceNumber) );
}

- (void)simplePing:(SimplePing *)pinger didReceiveUnexpectedPacket:(NSData *)packet
// A SimplePing delegate callback method.  We just log the receive.
{
    const ICMPHeader *  icmpPtr;

#pragma unused(pinger)
    assert(pinger == self.pinger);
#pragma unused(packet)

    icmpPtr = [SimplePing icmpInPacket:packet];
    if (icmpPtr != NULL) {
        NSLog(@"#%u unexpected ICMP type=%u, code=%u, identifier=%u", (unsigned int) OSSwapBigToHostInt16(icmpPtr->sequenceNumber), (unsigned int) icmpPtr->type, (unsigned int) icmpPtr->code, (unsigned int) OSSwapBigToHostInt16(icmpPtr->identifier) );
    } else {
        NSLog(@"unexpected packet size=%zu", (size_t) [packet length]);
    }
}

@end

现在当你想开始 ping 一些人时,你只需通过 NSOperationQueue 来管理它。

例子

self.pingQueue = [[NSOperationQueue alloc] init];


    NSArray *host = [NSArray arrayWithObjects:@"http://www.google.com", @"http://www.stackoverflow.com", @"http://www.woot.com", nil];

    for (int i = 0; i < host.count; i++) {
        PingOpertion *pingOperation = [[PingOpertion alloc] initWithHostName:host[i]];
        [self.pingQueue addOperation:pingOperation];
    }

关于ios - 带循环的 NSOperationQueue 和带委托(delegate)的类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31254825/

相关文章:

ios - 模态视图水平翻转过渡期间的背景

Java SWT 和无效的线程访问

android - 如何在 Android NDK 中创建线程?

ios - 照片框架 : Connection to assetsd was interrupted or assetsd died

ios - 选择段时如何从 TableView 加载选定数据?

c++ - XCode:链接器命令失败,退出代码为 1(使用 -v 查看调用)。如何?

iphone - 构建类似 iPhone 手机的应用程序

javascript - 如何在 iOS 应用程序中通过 websockets 实现和通信?

.net - IfxConnection 和线程可以相处吗?

ios - 我不能在两个不同的文件中使用#import "Header.h"