画像のURLはあります(UIImagePickerControllerから取得しました)が、画像はメモリにありません(URLはアプリの前回の実行で保存されました)。 URLからUIImageを再度リロードできますか?
UIImageにはimageWithContentsOfFile:があることがわかりますが、URLがあります。 NSDataのdataWithContentsOfURL:を使用してURLを読み取ることはできますか?
@Danielの答えに基づいて、次のコードを試しましたが、うまくいきません...
NSLog(@"%s %@", __PRETTY_FUNCTION__, photoURL);
if (photoURL) {
NSURL* aURL = [NSURL URLWithString:photoURL];
NSData* data = [[NSData alloc] initWithContentsOfURL:aURL];
self.photoImage = [UIImage imageWithData:data];
[data release];
}
実行すると、コンソールに以下が表示されます。
-[PhotoBox willMoveToWindow:] file://localhost/Users/gary/Library/Application%20Support/iPhone%20Simulator/3.2/Media/DCIM/100Apple/IMG_0004.JPG
*** -[NSURL length]: unrecognized selector sent to instance 0x536fbe0
*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '*** -[NSURL length]: unrecognized selector sent to instance 0x536fbe0'
呼び出しスタックを見て、URLWithString:relativeToURL:、initWithString:relativeToURL:、_ CFStringIsLegalURLString、CFStringGetLength、forwarding_prep_、forwardingを呼び出します、次に-[NSObject doesNotRecognizeSelector]。
NSString(photoURLのアドレスが0x536fbe0)が長さに応答しない理由は何ですか? -[NSURL length]に応答しないと言うのはなぜですか? paramがNSURLではなくNSStringであることを知らないのですか?
OK、コードの唯一の問題は文字列からURLへの変換です。文字列をハードコーディングすると、他のすべてが正常に機能します。 NSStringには何か問題があり、それを理解できない場合は、別の質問として入力する必要があります。この行を挿入すると(上記のコンソールログからパスを貼り付けました)、正常に動作します。
photoURL = @"file://localhost/Users/gary/Library/Application%20Support/iPhone%20Simulator/3.2/Media/DCIM/100Apple/IMG_0004.JPG";
この方法で行うことができます(同期的に、しかしコンパクトに):
UIImage *image = [UIImage imageWithData:[NSData dataWithContentsOfURL:[NSURL URLWithString:MyURL]]];
はるかに優れたアプローチは、AppleのLazyTableImagesを使用して対話性を維持することです。
SDWebImage を試すことができます:
簡単な例:
[cell.imageView setImageWithURL:[NSURL URLWithString:@"http://www.domain.com/path/to/image.jpg"] placeholderImage:[UIImage imageNamed:@"placeholder.png"]];
DLImageLoader を取得し、次のコードを試してください
[DLImageLoader loadImageFromURL:imageURL
completed:^(NSError *error, NSData *imgData) {
imageView.image = [UIImage imageWithData:imgData];
[imageView setContentMode:UIViewContentModeCenter];
}];
DLImageLoaderを使用する別の典型的な実世界の例は、誰かを助けるかもしれません...
PFObject *aFacebookUser = [self.fbFriends objectAtIndex:thisRow];
NSString *facebookImageURL = [NSString stringWithFormat:
@"http://graph.facebook.com/%@/picture?type=large",
[aFacebookUser objectForKey:@"id"] ];
__weak UIImageView *loadMe = self.userSmallAvatarImage;
// ~~note~~ you my, but usually DO NOT, want a weak ref
[DLImageLoader loadImageFromURL:facebookImageURL
completed:^(NSError *error, NSData *imgData)
{
if ( loadMe == nil ) return;
if (error == nil)
{
UIImage *image = [UIImage imageWithData:imgData];
image = [image ourImageScaler];
loadMe.image = image;
}
else
{
// an error when loading the image from the net
}
}];
上記で言及したように、最近考慮すべきもう1つの優れたライブラリはHanekeです(残念ながらそれほど軽量ではありません)。
Swiftバージョン:
let url = NSURL.URLWithString("http://live-wallpaper.net/iphone/img/app/i/p/iphone-4s-wallpapers-mobile-backgrounds-dark_2466f886de3472ef1fa968033f1da3e1_raw_1087fae1932cec8837695934b7eb1250_raw.jpg");
var err: NSError?
var imageData :NSData = NSData.dataWithContentsOfURL(url,options: NSDataReadingOptions.DataReadingMappedIfSafe, error: &err)
var bgImage = UIImage(data:imageData)
あなたがreallyの場合、絶対にpositively NSURLがファイルURLであることを確認してください。つまり、[url isFileURL]
が保証されますあなたの場合にtrueを返すには、単に使用することができます:
[UIImage imageWithContentsOfFile:url.path]
このコードを試して、イメージをロードするように設定できます。これにより、ユーザーはアプリがURLからイメージをロードしていることがわかります。
UIImageView *yourImageView = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"loading.png"]];
[yourImageView setContentMode:UIViewContentModeScaleAspectFit];
//Request image data from the URL:
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
NSData *imgData = [NSData dataWithContentsOfURL:[NSURL URLWithString:@"http://yourdomain.com/yourimg.png"]];
dispatch_async(dispatch_get_main_queue(), ^{
if (imgData)
{
//Load the data into an UIImage:
UIImage *image = [UIImage imageWithData:imgData];
//Check if your image loaded successfully:
if (image)
{
yourImageView.image = image;
}
else
{
//Failed to load the data into an UIImage:
yourImageView.image = [UIImage imageNamed:@"no-data-image.png"];
}
}
else
{
//Failed to get the image data:
yourImageView.image = [UIImage imageNamed:@"no-data-image.png"];
}
});
});
AFNetworking は、プレースホルダーをサポートするUIImageViewへの非同期画像の読み込みを提供します。また、一般的なAPIを操作するための非同期ネットワーキングもサポートしています。
here で提供されるAsyncImageView
を確認してください。優れたサンプルコードがいくつかあり、「箱から出して」すぐに使用できる場合もあります。
IOS 9からこの設定を有効にしてください:
App Transport Security SettingsinInfo.plistを使用して、URLから画像を確実にロードし、ダウンロード画像を許可して設定します。
そして、このコードを書きます:
NSURL *url = [[NSURL alloc]initWithString:@"http://feelgrafix.com/data/images/images-1.jpg"];
NSData *data =[NSData dataWithContentsOfURL:url];
quickViewImage.image = [UIImage imageWithData:data];
Swift Extension to UIImageView
(ソースコード ここ )を使用する方法:
UIActivityIndicatorView
の計算プロパティの作成import Foundation
import UIKit
import ObjectiveC
private var activityIndicatorAssociationKey: UInt8 = 0
extension UIImageView {
//Associated Object as Computed Property
var activityIndicator: UIActivityIndicatorView! {
get {
return objc_getAssociatedObject(self, &activityIndicatorAssociationKey) as? UIActivityIndicatorView
}
set(newValue) {
objc_setAssociatedObject(self, &activityIndicatorAssociationKey, newValue, UInt(OBJC_ASSOCIATION_RETAIN))
}
}
private func ensureActivityIndicatorIsAnimating() {
if (self.activityIndicator == nil) {
self.activityIndicator = UIActivityIndicatorView(activityIndicatorStyle: UIActivityIndicatorViewStyle.Gray)
self.activityIndicator.hidesWhenStopped = true
let size = self.frame.size;
self.activityIndicator.center = CGPoint(x: size.width/2, y: size.height/2);
NSOperationQueue.mainQueue().addOperationWithBlock({ () -> Void in
self.addSubview(self.activityIndicator)
self.activityIndicator.startAnimating()
})
}
}
convenience init(URL: NSURL, errorImage: UIImage? = nil) {
self.init()
self.setImageFromURL(URL)
}
func setImageFromURL(URL: NSURL, errorImage: UIImage? = nil) {
self.ensureActivityIndicatorIsAnimating()
let downloadTask = NSURLSession.sharedSession().dataTaskWithURL(URL) {(data, response, error) in
if (error == nil) {
NSOperationQueue.mainQueue().addOperationWithBlock({ () -> Void in
self.activityIndicator.stopAnimating()
self.image = UIImage(data: data)
})
}
else {
self.image = errorImage
}
}
downloadTask.resume()
}
}
Urlを介して画像を読み込むための最良かつ簡単な方法は、このコードによるものです。
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
NSData *data =[NSData dataWithContentsOfURL:[NSURL URLWithString:imgUrl]];
dispatch_async(dispatch_get_main_queue(), ^{
imgView.image= [UIImage imageWithData:data];
});
});
imgUrl
をImageURL
に置き換えますimgView
をUIImageView
に置き換えます。
別のスレッドにイメージをロードするため、アプリのロードが遅くなることはありません。