【播放器】播放器对比

news/2024/11/24 11:07:49/

前言###

由于什么直播、小视频火爆之后,我们开发者也要准备我们的播放器了,最常用到的肯定是自定义AVPlayer播放器,虽然本文也主要研究的也是AVPlayer,但是你知道吗?苹果其实为我们提供了三种播放器。

播放两种视频###

1、播放本地视频
播放本地视频其实就是播放本地的视频文件路径,[NSURL fileURLWithPath:videoPath];找到文件路径并且播放
2、播放网络视频
简单来讲就是我们设置播放器url地址等参数后,播放器就会向远程url所在服务器发送请求(一般请求包括两个值,一个是offset偏移量,另一个是length长度),然后服务器就会根据参数返回数据给播放器。
3、播放器一般支持播放格式
AVPlayer支持WMV,AVI,MKV,RMVB,RM,XVID,MP4,3GP,MPG等
MPMoviePlayerController与MPMoviePlayerViewController支持MOV, MP4, .MPV、3GP等

苹果的三种播放器播放视频###

我写了一个实例demo,包括
1、MPMoviePlayerController播放视频
2、MPMoviePlayerViewController播放视频
3、AVPlayer播放视频当个视频
4、AVPlayer播放视频多个视频
项目下载地址:三种视频播放器大总结
项目效果图

三种播放器大总结.gif

一、MPMoviePlayerController播放视频,但是iOS3.2之后不建议使用
这个播放器在MediaPlayer中,所以在使用的时候,必须先导入
#import <MediaPlayer/MediaPlayer.h>
MPMoviePlayerController播放器是继承于NSObject,但是内部有个view可以展示视频内容,如果将该视图添加其他控制器的view上,即可显示视频内容
播放状态需要增加观察者来监听

  • (MPMoviePlayerController *)player
    {
    if(_player == nil)
    {
    // 1.创建播放器
    NSURL url = [NSURL URLWithString:self.videoPath];
    _player = [[MPMoviePlayerController alloc]initWithContentURL:url];
    // 2.给播放器内部的View设置frame
    _player.view.frame = CGRectMake(0, 64, self.view.bounds.size.width, self.view.bounds.size.width * 9 / 16);
    // 3.添加到控制器View中
    [self.view addSubview:_player.view];
    //监听当前视频播放状态
    [[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(loadStateDidChange:) name:MPMoviePlayerLoadStateDidChangeNotification object:nil];
    // 4.设置控制面板的显示
    //_player.controlStyle = MPMovieControlStyleFullscreen;
    }
    return _player;
    }
    -(void)loadStateDidChange:(NSNotification
    )sender
    {
    switch (self.player.loadState) {
    case MPMovieLoadStatePlayable:
    {
    NSLog(@“加载完成,可以播放”);
    }
    break;
    case MPMovieLoadStatePlaythroughOK:
    {
    NSLog(@“缓冲完成,可以连续播放”);
    }
    break;
    case MPMovieLoadStateStalled:
    {
    NSLog(@“缓冲中”);
    }
    break;
    case MPMovieLoadStateUnknown:
    {
    NSLog(@“未知状态”);
    }
    break;
    default:
    break;
    }
    }
    二、MPMoviePlayerViewController播放视频,但是iOS3.2之后不建议使用
    其实MPMoviePlayerViewController播放视频与MPMoviePlayerController播放一样,只是MPMoviePlayerViewController只能全屏播放视频
    并且MPMoviePlayerViewController包含MPMoviePlayerController属性

  • (instancetype)initWithContentURL:(NSURL *)contentURL NS_DESIGNATED_INITIALIZER;
    @property (nonatomic, readonly) MPMoviePlayerController *moviePlayer;
    并且弹出方式也是modal方式
    -(void)presentMoviePlayerViewControllerAnimated:(MPMoviePlayerViewController *)moviePlayerViewController
    -(void)dismissMoviePlayerViewControllerAnimated

3、AVPlayer播放视频当个视频(本文重点讲解)
由于播放器可能在以后我们经常会用到,所以我建议封装成工具类。
首先我们来看看AVPlayer播放器设置一些什么鬼?

@property (nonatomic, strong) AVPlayerItem *currentPlayerItem; //当前正在播放视频的Item
@property (nonatomic, strong) AVPlayer *player;
@property (nonatomic, strong) AVPlayerLayer *currentPlayerLayer; //当前图像层
AVPlayerItem:播放器的播放model,管理资源的对象,这个东东很重要,不仅可以可以获得到播放状态status,还可以看到缓冲进度loadedTimeRange、playbackBufferEmpty、playbackLikelyToKeepUp
等属性。
AVPlayer:播放器自身的播放工具类,可以播放play、暂停pause等
AVPlayerLayer:播放器播放的View,预览层,把这个View增加到播放页面显示区域,就可以看到播放的画面

第一步:
导入AVPlayer播放器框架

#import <AVFoundation/AVFoundation.h>
第二步
使用url初始化播放器AVPlayerItem、AVPlayer、AVPlayerLayer

//加载视频的方式
if([url.absoluteString hasPrefix:@“http”])
{
//网络加载数据
AVPlayerItem *playerItem=[AVPlayerItem playerItemWithURL:[NSURL URLWithString:[url.absoluteString stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]]];
self.currentPlayerItem = playerItem;
}
else
{
//本地视频
self.currentPlayerItem = [AVPlayerItem playerItemWithURL:url];
}
self.player = [AVPlayer playerWithPlayerItem:self.currentPlayerItem];
self.currentPlayerLayer = [AVPlayerLayer playerLayerWithPlayer:self.player];
self.currentPlayerLayer.frame = CGRectMake(0, 0, superView.bounds.size.width, superView.bounds.size.height);
第三步
初始化完成,那么什么时候播放视频呢,查看AVPlayerItem的属性@property (nonatomic, readonly) AVPlayerItemStatus status;但是是只读的,如果我们想要知道播放器的状态,那么就利用KVO监听状态、同时监听播放器的缓冲进度playbackBufferEmpty、是否加载完成(注意监听者可以在初始化的时候就增加)playbackLikelyToKeepUp

#define LZBVideoPlayerStatus @“status”
#define LZBVideoPlayerPlaybackBufferEmpty @“playbackBufferEmpty”
#define LZBVideoPlayerStatusplaybackLikelyToKeepUp @“playbackLikelyToKeepUp”
//增加播放器播放监听

  • (void)addPlayerObserver
    {
    //播放状态监听
    [_currentPlayerItem addObserver:self forKeyPath:LZBVideoPlayerStatus options:NSKeyValueObservingOptionNew context:nil];
    [_currentPlayerItem addObserver:self forKeyPath:LZBVideoPlayerPlaybackBufferEmpty options:NSKeyValueObservingOptionNew context:nil];
    [_currentPlayerItem addObserver:self forKeyPath:LZBVideoPlayerStatusplaybackLikelyToKeepUp options:NSKeyValueObservingOptionNew context:nil];
    }
    //移除播放器播放监听
  • (void)removePlayerObserver
    {
    [_currentPlayerItem removeObserver:self forKeyPath:LZBVideoPlayerStatus];
    [_currentPlayerItem removeObserver:self forKeyPath:LZBVideoPlayerPlaybackBufferEmpty];
    [_currentPlayerItem removeObserver:self forKeyPath:LZBVideoPlayerStatusplaybackLikelyToKeepUp];
    }
    第四步
    监听播放器的状态
  • (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context
    {
    if([keyPath isEqualToString:LZBVideoPlayerStatus])
    {
    [self processObserveValueStatusWithItem:(AVPlayerItem *)object];
    }
    else if ([keyPath isEqualToString:LZBVideoPlayerPlaybackBufferEmpty])
    {
    [self processObserveBuffering];
    }
    else if([keyPath isEqualToString:LZBVideoPlayerStatusplaybackLikelyToKeepUp])
    {
    [self processObserveBuffered];
    }
    }
    //监听status
  • (void)processObserveValueStatusWithItem:(AVPlayerItem *)playerItem
    {
    AVPlayerItemStatus status = playerItem.status;
    switch (status) {
    case AVPlayerItemStatusUnknown:{
    }
    break;
    //监听到准备播放
    case AVPlayerItemStatusReadyToPlay:{
    [self.player play]; //开始准备播放
    [self.showSuperView.layer insertSublayer:self.currentPlayerLayer atIndex:0]; //把预览层增加到父类显示区域
    }
    break;
    case AVPlayerItemStatusFailed:{
    }
    break;
    }
    }

第五步
监听什么时候缓冲完成。主要逻辑:监听是否正在缓冲processObserveBuffering,如果正在缓冲,那么调用bufferingForSeconds,暂停播放、等过了2s之后再去播放一次并且检查时候缓冲完成,如果缓冲完成就播放,如果没有就继续等2s,这样循环到缓存完成。(只是适合播放小视频)

//缓冲进度

  • (void)processObserveBuffering
    {
    if(self.currentPlayerItem.playbackBufferEmpty)
    {
    [self startLoadingViewInSuperView:self.showSuperView];
    self.isBuffering = YES;
    [self bufferingForSeconds];
    }
    }
    //缓冲完成
  • (void)processObserveBuffered
    {
    if (self.currentPlayerItem.playbackLikelyToKeepUp){
    [self stopLoading];
    [self startPlayerTimeProgress];
    self.coverImageView.hidden = YES;
    self.isBuffering = NO;
    }
    }
    -(void)bufferingForSeconds{
    if (self.isBuffering == NO) return;
    [self.player pause];
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
    [self.player play];
    if (!self.currentPlayerItem.isPlaybackLikelyToKeepUp) {
    [self bufferingForSeconds];
    }
    });
    }
    第六步
    在播放过程中还可以获取播放时长和已经播放的时间

float totalDuration = CMTimeGetSeconds(self.currentPlayerItem.duration);
float currentTime = CMTimeGetSeconds(self.currentPlayerItem.currentTime);
NSInteger residueTime =(NSInteger)(totalDuration - currentTime);
第七步
播放过程中还要考虑,app的状态,并且监听内存等

//增加app运行状态监听
-(void)addObserverOnce{
if (!self.isAddObserver) {
// 添加监听,只能增加一次
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(appDidEnterBackground) name:UIApplicationWillResignActiveNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(appDidEnterPlayGround) name:UIApplicationDidBecomeActiveNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(playerItemDidPlayToEnd:) name:AVPlayerItemDidPlayToEndTimeNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(receiveMemoryWarning) name:UIApplicationDidReceiveMemoryWarningNotification object:nil];
}
self.isAddObserver = YES;
}
小细节:本文demo中还增加了视频加载过程的默认默认背景图等。
AVPlayer播放多个视频留到下一篇文章,未完待续。。。。
下一站
AVPlayer播放多个小视频切换

详情代码请直接下载demo查看:
项目下载地址:三种视频播放器大总结

作者:weixin_34288121
原文:https://blog.csdn.net/weixin_34288121/article/details/86856634


http://www.ppmy.cn/news/171723.html

相关文章

android-音乐播放器实现及源码下载(一)

从本文开始&#xff0c;详细讲述一个音乐播放器的实现&#xff0c;以及从网络解析数据获取最新推荐歌曲以及歌曲下载的功能。 功能介绍如下&#xff1a; 1、获取本地歌曲列表&#xff0c;实现歌曲播放功能。 2、利用硬件加速感应器&#xff0c;摇动手机实现切换歌曲的功能 …

Android开发本地及网络Mp3音乐播放器之网络音乐下载功能实现

实现功能&#xff1a; 实现网络音乐搜索功能 实现网络音乐下载功能 下载好的音乐目前不在播放器内&#xff0c;可以通过文件浏览器查看。 后续将博文&#xff0c;将实现歌词和下载音乐扫描功能。 实现效果如图&#xff1a; 实现代码如下&#xff1a; NetMusicListFragment如…

教你用三种方式打造一款简单的网络播放器

1-前言 视频类、直播类APP最近几年一直都比较火爆&#xff0c;占据下载排行榜的前列。华为视频服务提供视频播放服务&#xff0c;助力开发者快速构建视频特性&#xff0c;帮助开发者向用户提供视频媒体体验。本文介绍了几种简单实现网络视频播放器的方式&#xff0c;包括使用An…

Android基础课程:原生视频播放器(播放网络资源)

在之前的博客中&#xff0c;我分析过本地音乐播放器的逻辑和写法&#xff0c;需要使用MediaPlayer类&#xff0c;对于多媒体音频可进行播放&#xff0c;暂停&#xff0c;切换&#xff0c;停止等操作。在本篇博客中&#xff0c;将继续使用MediaPlayer类&#xff0c;将其放置于Su…

基于QT的网络音乐播放器(四)

关于歌词的显示&#xff0c;其实我的主要思想就是解析歌词部分的字符串。歌词显示分为两部分&#xff0c;一部分是播放器右侧的歌词显示以及下面的桌面歌词的显示。其中桌面歌词让我很难受&#xff0c;想了很久&#xff0c;后面看到一个大佬的一篇文章后才有了思路。 先看效果…

网络控制播放器(局域网内通过TCP和UDP控制视频播放器)

分享一个视频播放器&#xff0c;这个播放器支持能过网络TCP和UDP协议控制&#xff0c;同时还支持RS232和485串口控制。 可以通过指令控制播放器播放指定的视频&#xff0c;音量控制&#xff0c;图片浏览等。 播放器支持与所有的中控软件对接。 控制指令&#xff1a; 进入视频…

基于android的网络音乐播放器-网络音乐的搜索和展示(五)

作为android初学者&#xff0c;最近把疯狂android讲义和疯狂Java讲义看了一遍&#xff0c;看到书中介绍的知识点非常多&#xff0c;很难全部记住&#xff0c;为了更好的掌握基础知识点&#xff0c;我将开发一个网络音乐播放器-EasyMusic来巩固下&#xff0c;也当作是练练手。感…

易语言在线播放器源码php,易语言视频播放器源代码

用简单的语言编写视频播放器的源代码 源代码包含以下功能: *调用Thunder APlayer SDK进行视频播放*自动检测是否下载Thunderbolt解码库,如果不存在,则自动下载并解压缩(新手可以学习)新手可以学习)*支持快捷键可快进和快退以及将音量最多增加或减小至1000 *自动截取本地视频缩…