2012-07-12 22 views
5

Khi tôi sử dụng ALAssetsLibrary để tải ảnh cục bộ, nó hoạt động tốt. Nhưng sau khi tôi xóa một số ảnh bằng ứng dụng 'Ảnh', ứng dụng của tôi gặp sự cố.ALAssetsLibrary dường như trả lại số lượng sai ảnh của tôi

thông tin về lỗi là:

"Terminating app due to uncaught exception 'NSRangeException', reason: '*** -[NSOrderedSet enumerateObjectsAtIndexes:options:usingBlock:]: index 14 beyond bounds [0 .. 9]'".'14'

Dường như số lượng ảnh trong nước vẫn giữ nguyên như befoore. Và ngay cả sau khi tôi thoát khỏi ứng dụng và khởi động lại nó, nó vẫn bị treo.

mã truy cập hình ảnh địa phương:

dispatch_async(dispatch_get_main_queue(),^
{ 
    @autoreleasepool 
    { 
     ALAssetsLibraryAccessFailureBlock failureblock = ^(NSError *myerror) 
     { 
      NSLog(@"error occour =%@", [myerror localizedDescription]); 
     }; 

     ALAssetsGroupEnumerationResultsBlock groupEnumerAtion = ^(ALAsset *result, NSUInteger index, BOOL *stop) 
     { 
      if (result!=NULL) 
      { 
       if ([[result valueForProperty:ALAssetPropertyType] isEqualToString:ALAssetTypePhoto]) 
       { 
        [self.g_imageArray addObject:result]; 
       } 
      } 
     }; 

     ALAssetsLibraryGroupsEnumerationResultsBlock 
     libraryGroupsEnumeration = ^(ALAssetsGroup* group, BOOL* stop) 
     { 
      if (group == nil) 
      { 
       return; 
      } 

      if (group!=nil) { 
       [group enumerateAssetsUsingBlock:groupEnumerAtion]; 
      } 
     [self updatephotoList]; 
     }; 

     self.library = [[ALAssetsLibrary alloc] init]; 
     [self.library enumerateGroupsWithTypes:ALAssetsGroupSavedPhotos 
           usingBlock:libraryGroupsEnumeration 
          failureBlock:failureblock]; 
    } 
}); 

Nếu tôi lấy một bức ảnh với máy ảnh hệ thống, ứng dụng của tôi không OK lần nữa.

+0

Một câu hỏi: bạn có chắc rằng '[tự updatephotoList]' sẽ được gọi _after_ khối liệt kê cuối cùng của '[nhóm enumerateAssetsUsingBlock ... ]? Tôi có hàng tá sự cố trong ứng dụng của mình nên tôi sẽ kiểm tra cách tiếp cận này và xem cách thức hoạt động: http://stackoverflow.com/a/23378441/649379 – SoftDesigner

Trả lời

2

Điều này có vẻ là một lỗi iOS, như bạn đã nói ALAssetsLibrary trả lại số lượng sai ảnh của bạn để bạn có chỉ mục ngoài giới hạn lỗi. Giải pháp thay thế là tải lại ảnh của bạn một lần nữa như sau:

ALAssetsLibraryGroupsEnumerationResultsBlock 
libraryGroupsEnumeration = ^(ALAssetsGroup* group, BOOL* stop) 
{ 
     if (group == nil) 
     { 
      return; 
     } 
     //Force to reload photo as numberOfAssets is broken 
     NSLog(@"how many picture I have in this group: %d",[group numberOfAssets]); 
     [group setAssetsFilter:[ALAssetsFilter allPhotos]];//this will cause group to reload 
     NSLog(@"how many picture I have in this group: %d",[group numberOfAssets]); 

     if (group!=nil) { 
      [group enumerateAssetsUsingBlock:groupEnumerAtion]; 
     } 
    [self updatephotoList]; 
}; 
+0

cảm ơn sự giúp đỡ của bạn – zhangwx

0

Những gì bạn cần là đăng ký người quan sát cho ALAssetsLibraryChangedNotification để nhận các thay đổi trong thư viện. Khi thông báo kích hoạt, hãy reenumerate các nhóm và nội dung của AssetsLibrary. Nếu bạn không đăng ký thông báo, ứng dụng của bạn sẽ nhận được ảnh chụp nhanh cũ của thư viện và liệt kê sẽ không thành công. Cũng xin lưu ý rằng có lỗi liên quan đến ALAssetsLibraryChangedNotification trong iOS 5.x như được ghi ở đây: http://www.openradar.me/10484334

+0

Sẽ không trợ giúp nếu album bạn hiện đang xem bị xóa. Có, bạn sẽ nhận được thông báo nhưng ứng dụng sẽ gặp sự cố với ngoại lệ 'NSRangeException' khi cố gắng liệt kê lại. –

+0

Ngay cả khi không thay đổi nền, 'numberOfAssets' có thể sai trên iOS 8 (GM). – Rivera

+0

Đúng, đó là lỗi. các numberOfAssets không may bao gồm số lượng ảnh đã chuyển vào thùng rác/xóa trên iOS 8 (GM) – holtmann

1

Đăng ký người quan sát không giúp ích gì trong trường hợp của tôi. Người dùng vẫn bị rơi, nhưng tôi thì không. Cho đến ngày nay.

Tôi đã tìm ra cách để giải quyết sự cố này. Đó là một lỗi trong thư viện ảnh của Apple cuối cùng, nhưng có một cách giải quyết khác. Những gì bạn làm là, bạn đặt bộ lọc thành ảnh và sau đó là video, thay vì để nó ở 'nội dung' mặc định. Bạn sau đó liệt kê nó một lần cho mỗi, và làm một số trickery để đảm bảo bạn nhận được rằng cuối cùng 'tại null' điểm để làm bất cứ thông tin cập nhật bạn cần. Cách tiếp cận hiện tại của tôi hơi lộn xộn nhưng bạn có ý tưởng:

// pending is used to tell the block we're not done, even if result is NULL 
BOOL pending = YES; 
// resorted is just a flag I use in case there are no videos; if there are none, the block is never called at all, and thus the == NULL part never triggers 
__block BOOL resorted = NO; 

ALAssetsGroupEnumerationResultsBlock assetEnumerator = ^(ALAsset *result, NSUInteger index, BOOL *stop) { 
    if(result != NULL) { 
     [assets addObject:result]; 
    } else if (! pending) { 
     // ready!! 
     resorted = YES; 
     [self resort]; // my own method; replace with e.g. tableView reload! 
    } 
}; 

// there are two types of assets - photos and videos; we start with photo 
[group setAssetsFilter:[ALAssetsFilter allPhotos]]; 
NSLog(@"assets = %d", group.numberOfAssets); 
[group enumerateAssetsUsingBlock:assetEnumerator]; 
// we then set pending to NO; even though the enumeration happens in a separate thread, it seems like pending is not caught by the above enumeration (I have 105 images in the group I am enumerating, FWIW; it may be better to set pending in the == NULL part of the enumeration though 
pending = NO; 
// now we switch to video and do the same thing 
[group setAssetsFilter:[ALAssetsFilter allVideos]]; 
BriefLog(@"assets = %d", group.numberOfAssets); 
[group enumerateAssetsUsingBlock:assetEnumerator]; 

// if there are 0 vids, the above block is not ever called so we flip to a background thread and then back (probably not necessary) and then check if resorted is set; if it isn't we call resort 
if (! resorted) dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, 0), ^{ 
    dispatch_async(dispatch_get_main_queue(), ^{ 
     if (! resorted) { 
      [self resort]; // my own method; replace with e.g. tableView reload! 
     } 
    }); 
}); 

Vậy đó. Các NSRangeExceptions biến mất. Ít nhất trong trường hợp của tôi họ đã làm.

+1

'pending = NO;' của bạn không có hiệu lực. Các khối nắm bắt các biến cục bộ không '' chặn' theo giá trị tại thời điểm chúng được tạo. – user102008

+0

Tôi khá chắc chắn một trong các cuộc gọi có một có và khác có một không. Sẽ phải kiểm tra gấp đôi tho. – Kalle

0

Tôi bắt đầu với câu trả lời Qiulang, nhưng nó không hiệu quả với tôi. Điều gì làm việc cho tôi là gọi setAssetsFilter 3 lần liên tiếp với tất cả các kết hợp bộ lọc, trước khi bắt đầu đếm.

[group setAssetsFilter:[ALAssetsFilter allPhotos]]; 
[group setAssetsFilter:[ALAssetsFilter allVideos]]; 
[group setAssetsFilter:[ALAssetsFilter allAssets]]; 
+0

Không phải điều này hoặc thêm 'numberOfAssets' vào giữa các cuộc gọi đã làm việc cho tôi để có được số lượng ảnh phù hợp. – Rivera

0

Tại iOS 8 Tôi cũng nhận thấy số lượng ảnh đó bị xóa và nằm trong album 'Đã xóa gần đây'.

+1

Không thực sự là câu trả lời nhưng là một nhận xét. – Rivera

+0

Tôi gặp sự cố tương tự, sự cố đã được khắc phục trong iOS 8.0.2 – Ponf

+0

Đối với tôi, numberOfAssets vẫn gặp lỗi trên iOS8.0.2 – SarpErdag

0

Đối với những người bạn của những người điền vào một số NSArray với tất cả các ALAssetRepresentations và không muốn thay đổi bạn mã quá nhiều, chỉ cần lọc mảng của bạn với Checker này - AssetURLChecker

Cheers!

0

thư viện ALAssetsLibrary được depriciated trên PHAssetsLibrary nên sử dụng mã này:

__block PHAssetCollection *collection; 
    _arr_downloadedWallpaper = [[NSMutableArray alloc] init]; 

    // Find the album 
    PHFetchOptions *fetchOptions = [[PHFetchOptions alloc] init]; 
    fetchOptions.predicate = [NSPredicate predicateWithFormat:@"title = %@", @"Bhakti Ras"]; 
    collection = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeAlbum 
                  subtype:PHAssetCollectionSubtypeAny 
                  options:fetchOptions].firstObject; 

    PHFetchResult *collectionResult = [PHAsset fetchAssetsInAssetCollection:collection options:nil]; 

    [collectionResult enumerateObjectsUsingBlock:^(PHAsset *asset, NSUInteger idx, BOOL *stop) { 

     //add assets to an array for later use in the uicollectionviewcell 
     NSLog(@"asset is =%@",asset); 


     if (asset) { 
      [self.arr_downloadedWallpaper addObject:asset]; 
     } 
    }];