我一直在尝试测试我创建的一些使用NSNetServer
类执行网络操作的类。我有一些问题确保调用委托方法。
我尝试了很多方法,包括:
在其他操作发生时,使用[NSThread sleepForTimeInterval:5.0f];
和[[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:5.0f]];
暂停。 NSRunLoop
方法在第一次调用时工作(如下面的示例代码所示),但在第二次调用时崩溃。我知道"正确"做事的方式,但我不知道"纠正"方式是。
使用NSCondition
和NSConditionLock
类,它们似乎锁定代码并且永远不会调用回调。
对回调方法中更改的变量使用while
循环,与上面相同。
以下是带有一些额外注释的代码,为简单起见,删除了一些测试:
- (void)testCheckCredentials
{
[self.server start];
// Create a client
self.nsb = [[NSNetServiceBrowser alloc] init];
// Set the delegate to self
self.nsb.delegate = self;
// Search for the server
[self.nsb searchForServicesOfType:self.protocol inDomain:@""];
// Wait for the service to be found and resolved
[[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:self.timeout]];
XCTAssertTrue(self.serviceWasFound, @"Service was not found");
// Open the connection to the server
XCTAssertTrue([self.serverConnection open], @"Connection to server failed to open");
// Wait for the client to connect
/* This is where it crashes */
[[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:self.timeout]];
XCTAssertTrue(self.clientDidConnect, @"Client did not connect");
/* Further, more class-specific tests */
}
- (void)netServiceBrowser:(NSNetServiceBrowser *)aNetServiceBrowser didFindService:(NSNetService *)service moreComing:(BOOL)moreComing
{
NSLog(@"Found a service: %@ (%@)", service.name, service.domain);
if ([self.serverName isEqualToString:service.name]) {
self.serviceWasFound = YES;
}
}
- (void)clientDidConnect:(RCFClientConnection *)client
{
XCTAssertNotNil(client, @"Connected client is nil");
self.clientConnection = client;
self.clientDidConnect = YES;
}
我还尝试在lock
对象上执行NSCondition
:
[self.nsb searchForServicesOfType:self.protocol inDomain:@""];
// Wait for the service to be found and resolved
[self.lock lockWhenCondition:1];
XCTAssertTrue(self.serviceWasFound, @"Service was not found");
和
self.serviceWasFound = YES;
[self.lock unlockWithCondition:1]
使用lock方法时,永远不会调用netServiceBrowser:didFindService:moreComing:
方法,使用时也是如此:
while (!self.serviceWasFound) {};
我还在学习Objective-C,但我完全坚持这个问题。
答案 0 :(得分:14)
为了处理调用异步执行方法和函数的测试组件,XCTest在Xcode 6中得到了增强,包括使用新API和类XCTestExpectation对象处理块的能力。这些对象响应新的XCTest方法,这些方法允许测试方法等待,直到异步调用返回或达到超时。
以下是上述摘录的Apple文档链接。 Writing Tests of Asynchronous Operations
@interface sampleAPITests : XCTestCase<APIRequestClassDelegate>{
APIRequestClass *apiRequester;
XCTestExpectation *serverRespondExpectation;
}
@end
//implementation test class
- (void)testAPIConnectivity {
// This is an example of a functional test case.
serverRespondExpectation = [self expectationWithDescription:@"server responded"];
[apiRequester sendAPIRequestForMethod:nil withParams:nil];//send request to server to get tap info
apiRequester.delegate = self;
[self waitForExpectationsWithTimeout:1 handler:^(NSError *error) {
if (error) {
NSLog(@"Server Timeout Error: %@", error);
}
nslog(@"execute here after delegate called or timeout");
}];
XCTAssert(YES, @"Pass");
}
//Delegate implementation
- (void) request:(APIRequest *)request didReceiveResponse:(NSDictionary *)jsonResponse success:(BOOL)success{
[serverRespondExpectation fulfill];
XCTAssertNotNil(jsonResponse,@"json object returned from server is nil");
}
答案 1 :(得分:1)
许多测试库支持测试异步和线程操作。
它们的工作方式基本相同:
以下是一些提供这些功能的库:
据我所知,所有这些库都使用运行循环方法,因为其他库可能会导致死锁。为了使这种方法可靠地工作:
修改强>
从Xcode6开始,有一个新的断言XCTestExpectation
用于异步测试。
可以使用CocoaPods安装上述每个库。