0

我正在尝试从服务器中提取图像以获取滚动视图。用户放大或缩小视图后,应下载图像:

- (void)scrollViewDidEndZooming:(UIScrollView *)scrollView withView:(UIView *)view atScale:(float)scale {

Ymin=365000+375000*_scrollView.contentOffset.x/(scale*1024);
Ymax=365000+375000*(_scrollView.contentOffset.x/scale+1024/scale)/1024;
Xmin=6635000-260000*(_scrollView.contentOffset.y/scale+748/scale)/748;
Xmax=6635000-260000*_scrollView.contentOffset.y/(scale*748);

[self looYhendus];  //Creates NSURLConnection and downloads the image according to scale, Ymin, Ymax, Xmin and Xmax values

UIImage *saadudPilt=_kaardiPilt;
imageView=[[UIImageView alloc] initWithImage:saadudPilt];
imageView.frame=CGRectMake(_scrollView.contentOffset.x,_scrollView.contentOffset.y,1024,748);
[_scrollView addSubview:imageView];

}

在某些情况下(我不知道在什么条件下),它可以工作,但在某些情况下 NSURLConnection 委托方法不会被触发,并且设置为子视图的图像仍然是最初下载的图像(当应用程序发射)。然后,只有在我再次触摸屏幕(滚动视图滚动)后,NSLog 消息才显示图像已下载。这种行为的原因可能是什么?

编辑:添加了 NSURLConnection 委托方法。我尝试了其他一些方法,但它们最终都没有执行委托方法。这让我认为这不是关于 NSURConnection 而是关于 UIScrollView (显然,我可能错了)。

- (void)looYhendus
{   
yhendused=CFDictionaryCreateMutable(
                                    kCFAllocatorDefault, 
                                    0,
                                    &kCFTypeDictionaryKeyCallBacks,
                                    &kCFTypeDictionaryValueCallBacks);

NSString *aadress = [NSString stringWithFormat:@"http://xgis.maaamet.ee/wms-pub/alus?version=1.1.1&service=WMS&request=GetMap&layers=MA-ALUSKAART&styles=default&srs=EPSG:3301&BBOX=%d,%d,%d,%d&width=%d&height=%d&format=image/png",Ymin,Xmin,Ymax,Xmax,512,374];
NSURL *url = [[NSURL alloc] initWithString:aadress];
NSMutableURLRequest *theRequest = [NSMutableURLRequest requestWithURL:url];
NSURLConnection *theConnection = [[NSURLConnection alloc] initWithRequest:theRequest delegate:self];

if( theConnection )
{
    andmedServerist = [NSMutableData data];
    CFDictionaryAddValue(
                         yhendused,
                         (__bridge void *)theConnection,
                         (__bridge_retained CFMutableDictionaryRef)[NSMutableDictionary
                                                                    dictionaryWithObject:[NSMutableData data]
                                                                    forKey:@"receivedData"]);
}
CFRunLoopRun();
}

-(void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
[andmedServerist setLength: 0];
}

-(void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
NSMutableDictionary *connectionInfo =
(NSMutableDictionary*)objc_unretainedObject(CFDictionaryGetValue(yhendused, (__bridge void *)connection));
[[connectionInfo objectForKey:@"receivedData"] appendData:data];

}

-(void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Ühenduse viga" message:@"Kõige tõenäolisemalt on kaardiserveril probleeme või puudub seadmel internetiühendus" delegate:self cancelButtonTitle:@"Sulge" otherButtonTitles:nil];
[alert show];
CFRunLoopStop(CFRunLoopGetCurrent());
}


-(void)connectionDidFinishLoading:(NSURLConnection *)connection
{
NSMutableDictionary *connectionInfo =
(NSMutableDictionary*)objc_unretainedObject(CFDictionaryGetValue(yhendused, (__bridge void *)connection));
[connectionInfo objectForKey:@"receivedData"];
andmedServerist=[connectionInfo objectForKey:@"receivedData"];

_kaardiPilt = [UIImage imageWithData: andmedServerist];
CFDictionaryRemoveValue(yhendused, (__bridge void *)connection);

CFRunLoopStop(CFRunLoopGetCurrent());

}

编辑:添加了这个:

- (void)viewDidLoad
{
[super viewDidLoad];
Ymin=365000;
Ymax=740000;
Xmin=6375000;
Xmax=6635000;
[self looYhendus];
UIImage *saadudPilt=_kaardiPilt;
imageView=[[UIImageView alloc] initWithImage:saadudPilt];
imageView.frame=CGRectMake(0,0,1024,748);
[_scrollView addSubview:imageView];
[_scrollView setContentSize: CGSizeMake(1024, 748)];
_scrollView.minimumZoomScale = 1.0;
_scrollView.maximumZoomScale = 50.0;
_scrollView.delegate = self;
}
4

2 回答 2

1

我最近遇到了同样的问题。问题是我将连接放在异步线程中,而连接已经是异步的。

我在这里找到了解决方案:NSURLConnection delegate methods are not called

还有一些链接指向在该线程中遇到类似问题的其他人。

如果我是你,我会尝试简单地使用[theConnection start]并使用设置的超时初始化请求,这样你就不必担心在下载图像之前后台线程会关闭。

例如:

[request setTimeoutInterval:30];
于 2012-09-04T03:49:02.213 回答
1

为什么要在or方法中显式调用CFRunLoopRun();和停止它?( )connectionDidFinishLoading:didFailWithError:CFRunLoopStop(CFRunLoopGetCurrent());

假设您在主线程上执行此操作。您正在停止主线程的运行循环。可以有计时器,ScrollView 使用停止响应的事件,因为您停止了主线程的运行循环。

如果您NSURLConnection在主线程上调用,则不需要显式运行它(或停止它)。您可以安排在当前运行循环上运行,这是主线程运行循环。如果您在后台线程上执行此操作,那么您的代码似乎是有效的(尽管如果在单独的线程上调用它,您不应该显示UIAlertViewdidFailWithError:

更新的答案(基于@Shiim 的评论):

viewDidLoad您正在调用的方法中,该方法[self looYhendus];立即返回(因为您正在使用异步 URL 连接进行加载)。所以它按预期工作。移至下载完成后将下载的 imageView 数据添加到 scrollView 的 subView 的方法[scrollView addSubview:imageView]connectionDidFinishLoading:或者您可以考虑使用dispatch_queue's 创建一个线程并同步加载 URL 请求,然后使用dispatch_queue's 主队列将作为子视图添加到 ScrollView 的 imageView 的绘图调度到主线程上。

在您的情况下,我的建议是使用 dispatch_queue 重新设计方法。这将使您更好地理解解决问题(在这种情况下),并提高您的代码可读性。

于 2012-09-03T20:45:06.060 回答