前两天在网上看到一篇博客,介绍AVPlayer的使用,但是只简单介绍了一下单个的本地文件如何播放,心血来潮,就想着做一个类似于播放器的东西,能够实现播放网络歌曲,循环播放多首音乐,下面我们来实现一下

  首先明确一下,在本文中需要讲到的几点:

  实现网络歌曲的播放 实现在后台也能播放歌曲 实现多首歌曲的循环播放 需要有播放/暂停和下一首的功能 需要在播放期间能够得知该首歌曲的总时长和当前播放时长

  本文中就暂时将这名多,后面还会丰富,例如实现缓存下载,实现歌曲缓存的进度查看,实现能够使用耳机按钮控制播放等等。

  播放网络歌曲

  因为需要播放网络歌曲,我就往七牛云上传了几首歌,就不用再自己到处去找歌曲了

  首先,明确我们播放歌曲使用的是AVPlayer,至于为什么使用它不使用其他的,因为它好用啊,苹果封装了强大的功能,让我们使用,干嘛不用!其实还有其他原因,这个就等着你自己去搜索了。

  AVQueuePlayer

  AVQueuePlayer是AVPlayer的一个子类,他可以实现多首歌曲播放,所以我们直接使用它了

  //传入多个AVPlayerItem来初始化AVQueuePlayer

  + (instancetype)queuePlayerWithItems:(NSArray<AVPlayerItem *> *)items;

  复制代码

  AVPlayerItem

  AVPlayerItem是一个资源对象,我们加载歌曲的时候都是使用它,它提供了两种初始化方法

  //初始化网络资源

  + (instancetype)playerItemWithURL:(NSURL *)URL;

  //初始化本地资源,本地的音乐或者影片资源都是通过AVAsset拿出来

  + (instancetype)playerItemWithAsset:(AVAsset *)asset;

  复制代码

  先来试一下:

  //初始化AVPlayerItem

  NSMutableArray *items = [NSMutableArray array];

  NSArray *urls = @[MUSIC_URL1,MUSIC_URL2,MUSIC_URL3,MUSIC_URL4,MUSIC_URL5];

  for (NSString *url in urls) {

  AVPlayerItem *item = [[AVPlayerItem alloc] initWithURL:[NSURL URLWithString:url]];

  [items addObject:item];

  }

  //初始化AVQueuePlayer

  AVQueuePlayer *player = [AVQueuePlayer queuePlayerWithItems: items];

  //测试播放

  if(player.status == AVPlayerStatusReadyToPlay){

  [player play];

  }

  复制代码

  上面的代码看起来没有错,但是我在做的时候,却遇到一个问题,第一次点击的时候,并不会播放,第二次第三次就会开始播放了。

  其实这里是有一个缓冲的原因,因为是网络资源,涉及到一个缓冲,后面我们会对这个做处理,歌曲确实是能够播放的

  就这样,简单实现了多首歌曲的播放,但是我们还需要实现循环播放,这个就相对麻烦一点了。

  要实现循环播放,我们就需要知道AVQueuePlayer的播放机制,对于AVQueuePlayer播放,是有一个队列,每次播放完成一首歌曲过后,这首歌曲就会从队列中删除,即这个item会从队列中删除,并且如果我们想直接再将这个item再次加入队列,是不能够加入的,我们必须要在new 一个item,再次加载到这个队列当中,才能够实现再次播放。这个也是挺蛋疼的。

  知道了这个,我们就有想法了,我们能够在player最后一首歌曲即将播放完成后,再来新建一个队列啊。思路是正确的,但是我们不能够直接得到player正在播放最后一首歌曲,这时候我想到的是一个timer检测,通过timer去检测player的播放队列是否还剩下一首歌曲,如果是的话,我们就新建队列,加入到player的播放序列中

  首先,我们在开始播放歌曲的时候,就需要将timer启动,监测player

  self.checkMusicTimer = [NSTimer scheduledTimerWithTimeInterval:3 target:self selector:@selector(checkMusic) userInfo:nil repeats:YES];

  复制代码

  在checkMusic我们判断当前是否队列中只有一首歌曲

  - (void)checkMusic

  {

  if (self.player.items.count == 1){

  [self prepareItems];//这个方法即是再次创建队列,加入到player播放序列中

  [self play];

  }

  }

  复制代码

  // 准备歌曲

  // 因为需要歌曲循环播放,每次AVQueuePlayer播放完成一首歌曲,就会将其从队列中移除

  // 所以我们需要在歌曲最后一首播放完之前重新为AVQueuePlayer创建一个播放队列,这样就能够实现循环播放

  //

  //

  - (void)prepareItems{

  NSMutableArray *items = [NSMutableArray array];

  NSArray *urls = @[MUSIC_URL1,MUSIC_URL2,MUSIC_URL3,MUSIC_URL4,MUSIC_URL5];

  for (NSString *url in urls) {

  AVPlayerItem *item = [[AVPlayerItem alloc] initWithURL:[NSURL URLWithString:url]];

  [items addObject:item];

  //这里是添加每首歌曲的监测,我们后面会讲到

  [item addObserver:self forKeyPath:@"loadedTimeRanges" options:NSKeyValueObservingOptionNew context:nil];

  [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(playbackFinished:) name:AVPlayerItemDidPlayToEndTimeNotification object:item];

  }

  self.playerItems = items;

  for (AVPlayerItem *item in items) {

  if ([self.player canInsertItem:item afterItem:self.player.items.lastObject]) {

  [self.player insertItem:item afterItem:self.player.items.lastObject];

  }

  }

  }

  复制代码

  这样一来,我们就能够实现循环播放了,这里的代码和后面要讲到的有关联,所以这里看不清晰也没关系,接着往后看

  上面我们讲了,有个缓冲的原因,导致首次点击播放的时候,不能够成功播放,在AVPlayerItem中有一个属性loadedTimeRanges,表示的是缓存状态,我们可以对他进行观察,来进行播放

  //上面的代码已经写出了对缓冲的检测

  [item addObserver:self forKeyPath:@"loadedTimeRanges" options:NSKeyValueObservingOptionNew context:nil];

  复制代码

  然后我们在观察者中

  -(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{

  if ([keyPath isEqualToString:@"loadedTimeRanges"]) {

  NSLog(@"缓冲");

  [self play];

  }

  }

  复制代码

  我们在每个item中加入了观察者,在什么时候移除呢,当然是在每首歌曲播放完成后移除,如果不移除将会崩溃

  再次对每个item进行观测,播放结束时

  [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(playbackFinished:) name:AVPlayerItemDidPlayToEndTimeNotification object:item];

  复制代码

  在播放结束,移除观察者

  - (void)playbackFinished:(NSNotification *)notice {

  NSLog(@"播放完成");

  [self.currentItem removeObserver:self forKeyPath:@"loadedTimeRanges"];

  }

  复制代码实现后台播放

  要实现后台播放,很简单只需要加入几行代码

  //设置可后台播放

  NSError *error = nil;

  [[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayback error:&error];

  [[AVAudioSession sharedInstance] setActive:YES error:nil];

  复制代码

  然后我们还需要在项目里设置

  播放暂停

  这个就很简单了

  直接调方法就行

  上一首下一首也是直接调用方法就行

  /*!

  @method play

  @abstract Signals the desire to begin playback at the current item's natural rate.

  @discussion Equivalent to setting the value of rate to 1.0.

  */

  - (void)play;

  - /*!

  @method pause

  @abstract Pauses playback.

  @discussion Equivalent to setting the value of rate to 0.0.

  */

  - (void)pause;

  /*!

  @method advanceToNextItem

  @abstract Ends playback of the current item and initiates playback of the next item in the player's queue.

  @discussion Removes the current item from the play queue.

  */

  - (void)advanceToNextItem;

  复制代码时长计算

  为player加一个观察者就行

  -(void)playerDidPlay{

  // //添加播放进度观察者

  __weak typeof(self) weakSelf = self;

  self.timeObserver = [self.manager.player addPeriodicTimeObserverForInterval:CMTimeMake(1.0,1.0) queue:dispatch_get_main_queue() usingBlock:^(CMTime time) {

  float current = CMTimeGetSeconds(time);

  float total = CMTimeGetSeconds(weakSelf.manager.currentItem.duration);

  weakSelf.total = [NSString stringWithFormat:@"%.2f",total];

  weakSelf.current = [NSString stringWithFormat:@"%.f",current];

  weakSelf.label.text = [NSString stringWithFormat:@"%@/%@",weakSelf.current,weakSelf.total];

  }];

  _isPlaying = YES;

  [self.play setTitle:@"暂停" forState:UIControlStateNormal];

  }

  复制代码

  其中的CMTime指的是帧数

iOS AVPlayer 的使用实践的更多相关文章

  1. iOS应用开发最佳实践

    <iOS应用开发最佳实践> 基本信息 作者: 王浩    出版社:电子工业出版社 ISBN:9787121207679 上架时间:2013-7-22 出版日期:2013 年8月 开本:16 ...

  2. iOS AVPlayer视频播放器

    代码地址如下:http://www.demodashi.com/demo/11168.html 一.运行效果 二.实现过程 ①.创建播放器avPlayer //创建播放器 url = [url str ...

  3. iOS代码瘦身实践

    1 分析当前ipa的组成 一般一个ipa会包含: 1) 资源文件 本地文件:数据.配置.数据库等等 字体文件 图片资源 2)  源代码 通过生成linkmap文件,分析源代码生成的编译文件的大小.在B ...

  4. 奇艺iOS移动端网络优化实践 | 请求成功率优化篇 原创 Charles 爱奇艺技术

    奇艺iOS移动端网络优化实践 | 请求成功率优化篇 原创 Charles 爱奇艺技术

  5. iOS - AVPlayer 音视频播放

    前言 NS_CLASS_AVAILABLE(10_7, 4_0) @interface AVPlayer : NSObject @available(iOS 4.0, *) public class ...

  6. iOS AvPlayer AvAudioPlayer音频的后台播放问题

    iOS 4开始引入的multitask,我们可以实现像ipod程序那样在后台播放音频了.如果音频操作是用苹果官方的AVFoundation.framework实现,像用AvAudioPlayer,Av ...

  7. 视频云SDK iOS持续集成项目实践

    1. 前言 2016年, 我们维护的 iOS推流播放融合SDK KSYLive_iOS 在github上发布了40多个版本, 平均两周发布一个新版本, 经历了最初痛苦的全手动版本构建和维护, 到后来慢 ...

  8. 学习笔记TF066:TensorFlow移动端应用,iOS、Android系统实践

    TensorFlow对Android.iOS.树莓派都提供移动端支持. 移动端应用原理.移动端.嵌入式设备应用深度学习方式,一模型运行在云端服务器,向服务器发送请求,接收服务器响应:二在本地运行模型, ...

  9. 【转】iOS:AvPlayer设置播放速度不生效的解决办法

    现象: 项目有一个需求是实现视频的慢速播放,使用的是封装的AvPlayer,但是设置时发现比如设置rate为0.5,0.1,0.01都是一样的速度,非常疑惑.后来经过查找资料,发现iOS10对这个AP ...

随机推荐

  1. Linux下的命令,删除文件夹下的所有文件,而不删除文件夹本身

    Linux下的命令,删除文件夹下的所有文件,而不删除文件夹本身 rm -rf *

  2. Jenkins之自动发送git变更到微信

    当我们通过Jenkins构建job的时候,是可以获取到git Change Log 的信息, 即本次上线修改了什么功能,我们将这个信息发送到微信群相关人员可直接获取到上线变更信息, 这样就不需要人为的 ...

  3. DOS & UNIX文件格式转换

    1.使用vi编辑器 vi xxxx :set fileformat=unix(or dos) :wq 2.使用 dos2unix 这个只能把DOS转换成UNIX文件 . sudo apt-get in ...

  4. 外带IP 防火墙限制导致 IP不通

    案例: 业务报障,一台设备配了20个IP,跳板机测试都通,但从外边访问,发现部分IP通,部分不通. 排雷: 1. 从跳板机测试都通,说明所有IP 本身应该都没问题的,都可以用,2. 从其他设备测试,部 ...

  5. 02.01Linux中软件的安装、环境搭建

    图1 图2 图3 redis安装 图4 =====================linux下的软件的安装====================安装方式:Yum/rpm/源码安装yum:通过分析rp ...

  6. python3 正则表达式 re模块之辣眼睛 计算器

    额...学到几个常用模块了,也要其中考试了,每天晚上敲一点,敲得脑壳疼,不过又想到好一点的办法了,有时间再改吧. 此非吾所欲也,实属无奈也....复习之路漫漫,吾将到书上求索,在此不多逗留,我挥一挥衣 ...

  7. 使用C++11原子量实现自旋锁

    一.自旋锁 自旋锁是一种基础的同步原语,用于保障对共享数据的互斥访问.与互斥锁的相比,在获取锁失败的时候不会使得线程阻塞而是一直自旋尝试获取锁.当线程等待自旋锁的时候,CPU不能做其他事情,而是一直处 ...

  8. 小程序部分机型上一个诡异的偶现bug

    如上图所示:开始的时候进到下单页面,价格是0,当选中了商品产生价格的时候,生成的价格如 ¥150,这个时候会只露出¥1以及一小半的5,后面的都被遮挡住了. wxml里是这样的写的 <view w ...

  9. (模板)poj1681 高斯消元法求异或方程组(无解、唯一解、多解)

    题目链接:https://vjudge.net/problem/POJ-1681 题意:类似于poj1222,有n×n的01矩阵,翻转一个点会翻转其上下左右包括自己的点,求最少翻转多少点能使得矩阵全0 ...

  10. Spark Scala当中reduceByKey(_+_) reduceByKey((x,y) => x+y)的用法

    [学习笔记] reduceByKey(_+_)是reduceByKey((x,y) => x+y)的一个 简洁的形式*/ val rdd08 = sc.parallelize(List((1, ...