0

在我的一个视图控制器中,我根据从单独的NSObject类接收到的“GET”数据设置标签。显然,设置标签所需的时间比获取数据所需的时间要少得多,因此标签始终设置为 nil。如何确保在数据获取完成之前不会设置标签。

NSObject这是在类中执行“获取”的方法myClass

- (void) doGetURL:(NSString *) urlstring
callBackTarget:(id) target
callBackMethod:(NSString *) method
 failedMethod:(NSString *) method_failed
 {

NSMutableURLRequest *request = [[NSMutableURLRequest alloc] init];
[request setURL:[NSURL URLWithString:urlstring]];
NSLog(@"-- get URL with cookie : [%@] and hash:(%@)", [self cookie], [self modhash]);
if (cookie && [cookie length] > 0)
{
    NSDictionary *properties = [NSDictionary dictionaryWithObjectsAndKeys:
                                cookieDomain, NSHTTPCookieDomain,
                                @"/", NSHTTPCookiePath, 
                                @"reddit_session", NSHTTPCookieName,
                                cookie, NSHTTPCookieValue,
                                //                                      [cookie stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding], NSHTTPCookieValue,
                                nil];
    NSHTTPCookie *http_cookie = [NSHTTPCookie cookieWithProperties:properties];
    NSArray* cookies = [NSArray arrayWithObjects: http_cookie, nil];
    NSDictionary * headers = [NSHTTPCookie requestHeaderFieldsWithCookies:cookies];
    [request setAllHTTPHeaderFields:headers];
}
NSURLConnection * connection = [NSURLConnection connectionWithRequest:request delegate:self];

NSString *connectionKey = [NSString stringWithFormat: @"%ld", ((intptr_t) connection)];
NSMutableDictionary *dl = [[NSMutableDictionary alloc] init];

[dl setValue:connectionKey forKey:@"connectionKey"];
if (target && method)
{
    [dl setValue:target forKey:@"afterCompleteTarget"];
    [dl setValue:method forKey:@"afterCompleteAction"];
}
[dl setValue:method_failed forKey:@"failedNotifyAction"];
[connections setValue:dl forKey:connectionKey];
}

这是在另一个方法中调用的myClass

- (void)getUserInfo:(NSString*)user
{
NSString *getString = [NSString stringWithFormat:@"%@/user/%@/about.json",server,user];

[self doGetURL:getString callBackTarget:self callBackMethod:@"userInfoResponse:" failedMethod:@"connectionFailedDialog:"];
}

回调方法:

- (void)userInfoResponse:(id)sender
{    
NSLog(@"userInfoResponse in()");
NSData * data = (NSData *) sender;
NSError *error;

NSDictionary *json = [NSJSONSerialization
                      JSONObjectWithData:data

                      options:kNilOptions
                      error:&error];

NSDictionary *response = [json objectForKey:@"data"];

//futureLabelStr is a property of myClass
futureLabelStr = [response objectForKey:@"name"];;

}

然后在 View Controller 中设置标签:

- (void)viewDidLoad
{
[myClass getUserInfo:@"some_user"];

myLabel.txt = myClass.futureLabelStr;

}

请让我知道我需要添加更多或任何我试图尽我所能组织它的东西,但我可能错过了一些东西。

4

2 回答 2

3

您不想“停止”您的 viewController 的 viewDidLoad,您想在信息更改时通知它。

您可以通过在 myClass 完成并被调用时发送通知来做到这-userInfoResponse:一点(查看 NSNotificationCenter),或者在 myClass 中实现委托模式。您可以将 viewController 设置为 myClass 的委托,并在 myClass 完成对 viewController 的获取时调用委托方法,该方法本身会更新标签。

或者,查看您的代码,您可以将您的 viewController 设置为回调方法的接收者,而对代码的更改最少,即使这不是最好的方法,因为它违反了 MVC 模式:

[self doGetURL:getString callBackTarget:viewController callBackMethod:@"userInfoResponse:" failedMethod:@"connectionFailedDialog:"];

您当然需要在 myClass 中引用 viewController 并且 viewController 需要实现此方法(这是违反 MVC 模式的)。

于 2013-06-16T23:29:44.703 回答
0

在新线程上发送数据调用并正常完成 viewDidLoad。然后使用 NSNotification 中心从获取这个(应该是模型)的人到 viewController 说“嘿,我为你拿到了那个标签,来拿它并刷新”

然后 VC 将使用模型中的数据设置标签。查看此链接以使用 NSNotificationCenter stackoverflow.com/questions/2191594/send-and-receive-messages-through-nsnotificationcenter-in-objective-c。

对于大中央调度上的多线程阅读。

于 2013-06-17T02:12:43.147 回答