• 作者:老汪软件技巧
  • 发表时间:2024-10-03 04:01
  • 浏览量:

将手机系统升级到 iOS18,App 内访问系统相册获取到视频播放链接,使用 AVPlayer 播放该提示,系统提示:

Error Domain=NSCocoaErrorDomain Code=257 "未能打开该文件,因为你没有查看它的权限。" UserInfo={NSUnderlyingError=0x30338fe70 {Error Domain=NSOSStatusErrorDomain Code=-12203 "(null)

在 iOS18 之前的系统都是播放正常,多操作几次,发现了一个神奇的问题:

打开相册播放一个在 App 中播放失败的视频后, 重新回到 App,重新播放该视频,视频竟然又神奇的播放成功了,可能还附带其他的一些视频播放成功

然后经过一系列的摸索,在一下代码中大概明白了什么原因


- (void)setAssetModel:(YXCAssetModel *)assetModel {
    
    _assetModel = assetModel;
    
    NSLog(@"%@ - %@", self, assetModel);
    YXCWeakSelf(self)
    PHImageRequestOptions *option = [PHImageRequestOptions new];
    [[PHImageManager defaultManager] requestImageForAsset:assetModel.asset
                                               targetSize:self.imageView.bounds.size
                                              contentMode:PHImageContentModeDefault
                                                  options:option
                                            resultHandler:^(UIImage * _Nullable result, NSDictionary * _Nullable info) {
        weakself.imageView.image = result;
    }];
    
    [self getVideoWithAsset:assetModel.asset];
}
- (void)getVideoWithAsset:(PHAsset *)asset {
    
    [YXCPhotoHandler getVideoWithAsset:asset complete:^(AVAsset * _Nullable asset, NSDictionary * _Nullable info) {
        if ([asset isKindOfClass:[AVURLAsset class]]) {
            dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
                NSString *videoUrl = [(AVURLAsset *)asset URL].absoluteString;
                [self p_playVideoWithURL:videoUrl];
            });
        }
    }];
}
// 播放视频
- (void)p_playVideoWithURL:(NSString *)urlString {
    NSLog(@"视频播放地址 : %@", urlString);
    
    if (self.player) {
        [self.player pause];
        self.player = nil;
    }
    
    if (self.playerItem) {
        self.playerItem = nil;
    }
    
    if (self.playerLayer) {
        [self.playerLayer removeFromSuperlayer];
        self.playerLayer = nil;
    }
    
    // 初始化 AVPlayer 并播放视频
    self.playerItem = [[AVPlayerItem alloc] initWithURL:[NSURL URLWithString:urlString]];
    self.player = [AVPlayer playerWithPlayerItem:self.playerItem];
    self.playerLayer = [AVPlayerLayer playerLayerWithPlayer:self.player];
    
    // 设置 playerLayer 尺寸
    self.playerLayer.frame = self.bounds;
    self.playerLayer.videoGravity = AVLayerVideoGravityResizeAspect;
    
    // 将 playerLayer 添加到视图中
    [self.layer addSublayer:self.playerLayer];
    
    // 播放视频
    [self.player play];
}

以上代码视频在 App 中是能正常播放的, 其中 getVideoWithAsset: 方法中,动一行代码后,就无法播放成功

- (void)getVideoWithAsset:(PHAsset *)asset {
    
    [YXCPhotoHandler getVideoWithAsset:asset complete:^(AVAsset * _Nullable asset, NSDictionary * _Nullable info) {
        if ([asset isKindOfClass:[AVURLAsset class]]) {
            // 对比上面的代码,只是将该行代码放在了 block 外面
            NSString *videoUrl = [(AVURLAsset *)asset URL].absoluteString;
            dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
                [self p_playVideoWithURL:videoUrl];
            });
        }
    }];
}

可能此时,好奇为什么要加延时 3s 播放,这里只是为了让播放异常现象更明显

通过两个对比分析之后,猜测原因是:

系统获取到了视频资源 AVAsset 对象,block 因为会持有对象的特性,导致在 block 中和 block 外面去获取 videoUrl,对 AVAsset 对象的生命周期影响不一样。block 中会持有 AVAsset 对象,所以播放视频的时候 AVAsset 对象还没有被释放;如果 block 外面去获取的 videoUrl,这样 block 就不会持有 AVAsset 对象,因为异步的原因,大概率导致 AVAsset 对象被释放了,这样就无法播放视频,提示没有权限。

总结:

在 iOS 18 中,苹果可能针对 AVAsset 进行了安全性优化,如果说 AVAsset 对象被释放了,就无法通过获取到的 url 进行 访问,想要正常播放的话,就必须保证 AVAsset 对象在播放的时候没被释放。