繁体   English   中英

NSOperationQueue和ASIHTTPRequest

[英]NSOperationQueue and ASIHTTPRequest

我正在为围绕ASIHTTPRequest编写的包装类编写测试用例 由于我无法确定的原因,我的测试用例在ASIHTTPRequest完成之前以失败告终。

这是程序流程的工作方式。

  1. 从我的测试案例开始。
  2. 初始化我的http引擎对象,指示它创建一个新列表
  3. 创建新的ASIHTTPRequest对象并进行设置。
  4. 将请求添加到操作队列。
  5. 等待直到队列为空
  6. 检查是否调用了我的委托方法,如果没有,则通过测试。

现在,大多数情况下一切正常,测试通过了,但是有时失败了,因为在操作队列将控制权返回给我的等待方法之后,我的委托方法被称为。

测试用例

// Set my flags to 'NO'
- (void)setUp {
    requestDidFinish = NO;
    requestDidFail = NO;
}

- (void)testCreateList {
    NSString *testList = @"{\"title\": \"This is a list\"}";

    JKEngine *engine = [[JKEngine alloc] initWithDelegate:self];
    NSString *requestIdentifier = [engine createList:jsonString];

    [self waitUntilEngineDone:engine];
    NSString *responseString = responseString_;
    [engine release];

    GHAssertNotNil(requestIdentifier, nil);
    GHAssertTrue(requestDidFinish, nil);
    GHAssertTrue([responseString hasPrefix:@"{\"CreateOrEditListResult\""], nil);

}

// Puts the test into a holding pattern until the http request is done
- (void)waitUntilEngineDone:(JKEngine *)engine {
    [engine waitUntilFinishedRunning]; 
}

// The delegate method called on successful completion
- (void)requestFinished:(NSString *)requestIdentifier withResponse:(NSString *)response {
    NSLog(@"request did finish");
    requestDidFinish = YES;
    responseIdentifier_ = [requestIdentifier retain];
    responseString_ = [response retain];
}

引擎代码

- (NSString *)createList:(NSString *)list {
    ASIHTTPRequest *request = [[ASIHTTPRequest alloc] initWithURL:[NSURL URLWithString:url]];
    [request addRequestHeader:@"Content-Type" value:kContentType];
    [request setRequestMethod:kPOST];
    request.delegate = self;

    [request appendPostData:[list dataUsingEncoding:NSUTF8StringEncoding]];

    NSString *requestIdentifier = [NSString stringWithNewUUID];

    [operationQueue_ addOperation:request];
    [operationDictionary_ setObject:request forKey:requestIdentifier];

    return requestIdentifier;
}

// This is the ASIHTTPRequest delegate method that's called on success
//   but it sometimes isn't called until AFTER the operationQueue finishes running
- (void)requestFinished:(ASIHTTPRequest *)request {
    DLog([request responseString]);

    BOOL canNotifiyDelegate = [self.delegate respondsToSelector:@selector(requestFinished:withResponse:)];
    if (canNotifiyDelegate) {
        NSArray *keyArray = [operationDictionary_ allKeysForObject:request];
        NSString *requestIdentifier = [keyArray objectAtIndex:0];
        [operationDictionary_ removeObjectForKey:requestIdentifier];

        if ([keyArray count] != 1) {
            ALog(@"It looks like a request was added to the operation dictionary multiple times. There's a bug somewhere.", nil);
        }

        [self.delegate requestFinished:requestIdentifier withResponse:[request responseString]];
    }
}

- (void)waitUntilFinishedRunning {
    [operationQueue_ waitUntilAllOperationsAreFinished];
}

这就是ASIHTTPRequest的工作方式。 委托方法是在主线程上调用的,对委托的调用不会阻塞请求线程,因此,完全有可能在队列完成后调用您的委托。

ASIHTTPRequest在主线程上调用委托方法,默认情况下, GH-Unit在后台线程上运行其测试。 我对到底发生了什么还不太了解,但是强制我的网络测试在主线程上运行可以解决此问题。

我在网络测试课程中实现了以下方法。

- (BOOL)shouldRunOnMainThread { 
    return YES;
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM