将手机系统升级到 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 对象在播放的时候没被释放。
网友评论