iOS开发系列--音频播放、录音、视频播放、拍照、视频录制

--iOS多媒体

概览

随着移动互联网的发展,如今的手机早已不是打电话、发短信那么简单了,播放音乐、视频、录音、拍照等都是很常用的功能。在iOS中对于多媒体的支持是非常强大的,无论是音视频播放、录制,还是对麦克风、摄像头的操作都提供了多套API。在今天的文章中将会对这些内容进行一一介绍:

  1. 音频
    1. 音效
    2. 音乐
    3. 音频会话
    4. 录音
    5. 音频队列服务
  2. 视频
    1. MPMoviePlayerController
    2. MPMoviePlayerViewController
    3. AVPlayer
  3. 摄像头
    1. UIImagePickerController拍照和视频录制
    2. AVFoundation拍照和录制视频
  4. 总结

音频

在iOS中音频播放从形式上可以分为音效播放和音乐播放。前者主要指的是一些短音频播放,通常作为点缀音频,对于这类音频不需要进行进度、循环等控制。后者指的是一些较长的音频,通常是主音频,对于这些音频的播放通常需要进行精确的控制。在iOS中播放两类音频分别使用AudioToolbox.framework和AVFoundation.framework来完成音效和音乐播放。

音效

AudioToolbox.framework是一套基于C语言的框架,使用它来播放音效其本质是将短音频注册到系统声音服务(System Sound Service)。System Sound Service是一种简单、底层的声音播放服务,但是它本身也存在着一些限制:

  • 音频播放时间不能超过30s 
  • 数据必须是PCM或者IMA4格式 
  • 音频文件必须打包成.caf、.aif、.wav中的一种(注意这是官方文档的说法,实际测试发现一些.mp3也可以播放)

使用System Sound Service 播放音效的步骤如下:

  1. 调用AudioServicesCreateSystemSoundID(   CFURLRef  inFileURL, SystemSoundID*   outSystemSoundID)函数获得系统声音ID。 
  2. 如果需要监听播放完成操作,则使用AudioServicesAddSystemSoundCompletion(  SystemSoundID inSystemSoundID,
    CFRunLoopRef  inRunLoop, CFStringRef  inRunLoopMode, AudioServicesSystemSoundCompletionProc  inCompletionRoutine, void*  inClientData)
    方法注册回调函数。 
  3. 调用AudioServicesPlaySystemSound(SystemSoundID inSystemSoundID) 或者AudioServicesPlayAlertSound(SystemSoundID inSystemSoundID) 方法播放音效(后者带有震动效果)。

下面是一个简单的示例程序:

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  KCMainViewController.m
//  Audio
//
//  Created by Kenshin Cui on 14/03/30.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//  音效播放

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"KCMainViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><AudioToolbox/AudioToolbox.h>

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">KCMainViewController ()

@end

@implementation KCMainViewController

- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];
    
    [self playSoundEffect:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"videoRing.caf"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  播放完成回调函数
 *
 *  @param soundID    系统声音ID
 *  @param clientData 回调时传递的数据
 */
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">soundCompleteCallback(SystemSoundID soundID,</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">* clientData){
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"播放完成..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  播放音效文件
 *
 *  @param name 音频文件名称
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)playSoundEffect:(NSString *)name{
    NSString *audioFile=[[NSBundle mainBundle] pathForResource:name ofType:nil];
    NSURL *fileUrl=[NSURL fileURLWithPath:audioFile];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//1.获得系统声音ID
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">SystemSoundID soundID=0;
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
     * inFileUrl:音频文件url
     * outSystemSoundID:声音id(此函数会将音效文件加入到系统音频服务中并返回一个长整形ID)
     */
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AudioServicesCreateSystemSoundID((__bridge CFURLRef)(fileUrl), &soundID);
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//如果需要在播放完之后执行某些操作,可以调用如下方法注册一个播放完成回调函数
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AudioServicesAddSystemSoundCompletion(soundID, NULL, NULL, soundCompleteCallback, NULL);
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//2.播放音频
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AudioServicesPlaySystemSound(soundID);</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放音效
//    AudioServicesPlayAlertSound(soundID);//播放音效并震动
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}

@end</span>

音乐

如果播放较大的音频或者要对音频有精确的控制则System Sound Service可能就很难满足实际需求了,通常这种情况会选择使用AVFoundation.framework中的AVAudioPlayer来实现。AVAudioPlayer可以看成一个播放器,它支持多种音频格式,而且能够进行进度、音量、播放速度等控制。首先简单看一下AVAudioPlayer常用的属性和方法:

属性说明
@property(readonly, getter=isPlaying) BOOL playing是否正在播放,只读
@property(readonly) NSUInteger numberOfChannels音频声道数,只读
@property(readonly) NSTimeInterval duration音频时长
@property(readonly) NSURL *url音频文件路径,只读
@property(readonly) NSData *data音频数据,只读
@property float pan立体声平衡,如果为-1.0则完全左声道,如果0.0则左右声道平衡,如果为1.0则完全为右声道
@property float volume音量大小,范围0-1.0
@property BOOL enableRate是否允许改变播放速率
@property float rate播放速率,范围0.5-2.0,如果为1.0则正常播放,如果要修改播放速率则必须设置enableRate为YES
@property NSTimeInterval currentTime当前播放时长
@property(readonly) NSTimeInterval deviceCurrentTime输出设备播放音频的时间,注意如果播放中被暂停此时间也会继续累加
@property NSInteger numberOfLoops循环播放次数,如果为0则不循环,如果小于0则无限循环,大于0则表示循环次数
@property(readonly) NSDictionary *settings音频播放设置信息,只读
@property(getter=isMeteringEnabled) BOOL meteringEnabled是否启用音频测量,默认为NO,一旦启用音频测量可以通过updateMeters方法更新测量值
对象方法说明
- (instancetype)initWithContentsOfURL:(NSURL *)url error:(NSError **)outError使用文件URL初始化播放器,注意这个URL不能是HTTP URL,AVAudioPlayer不支持加载网络媒体流,只能播放本地文件
- (instancetype)initWithData:(NSData *)data error:(NSError **)outError使用NSData初始化播放器,注意使用此方法时必须文件格式和文件后缀一致,否则出错,所以相比此方法更推荐使用上述方法或- (instancetype)initWithData:(NSData *)data fileTypeHint:(NSString *)utiString error:(NSError **)outError方法进行初始化
- (BOOL)prepareToPlay;加载音频文件到缓冲区,注意即使在播放之前音频文件没有加载到缓冲区程序也会隐式调用此方法。
- (BOOL)play;播放音频文件
- (BOOL)playAtTime:(NSTimeInterval)time在指定的时间开始播放音频
- (void)pause;暂停播放
- (void)stop;停止播放
- (void)updateMeters更新音频测量值,注意如果要更新音频测量值必须设置meteringEnabled为YES,通过音频测量值可以即时获得音频分贝等信息
- (float)peakPowerForChannel:(NSUInteger)channelNumber; 获得指定声道的分贝峰值,注意如果要获得分贝峰值必须在此之前调用updateMeters方法
- (float)averagePowerForChannel:(NSUInteger)channelNumber获得指定声道的分贝平均值,注意如果要获得分贝平均值必须在此之前调用updateMeters方法
@property(nonatomic, copy) NSArray *channelAssignments获得或设置播放声道
代理方法说明
- (void)audioPlayerDidFinishPlaying:(AVAudioPlayer *)player successfully:(BOOL)flag音频播放完成
- (void)audioPlayerDecodeErrorDidOccur:(AVAudioPlayer *)player error:(NSError *)error音频解码发生错误

AVAudioPlayer的使用比较简单:

  1. 初始化AVAudioPlayer对象,此时通常指定本地文件路径。 
  2. 设置播放器属性,例如重复次数、音量大小等。 
  3. 调用play方法播放。

下面就使用AVAudioPlayer实现一个简单播放器,在这个播放器中实现了播放、暂停、显示播放进度功能,当然例如调节音量、设置循环模式、甚至是声波图像(通过分析音频分贝值)等功能都可以实现,这里就不再一一演示。界面效果如下:

AudioPlayerScreen

当然由于AVAudioPlayer一次只能播放一个音频文件,所有上一曲、下一曲其实可以通过创建多个播放器对象来完成,这里暂不实现。播放进度的实现主要依靠一个定时器实时计算当前播放时长和音频总时长的比例,另外为了演示委托方法,下面的代码中也实现了播放完成委托方法,通常如果有下一曲功能的话播放完可以触发下一曲音乐播放。下面是主要代码:

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  KCAVAudioPlayer
//
//  Created by Kenshin Cui on 14/03/30.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><AVFoundation/AVFoundation.h>
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#define </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">kMusicFile @</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"刘若英 - 原来你也在这里.mp3"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#define </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">kMusicSinger @</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"刘若英"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#define </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">kMusicTitle @</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"原来你也在这里"

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()<AVAudioPlayerDelegate>

@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(nonatomic,strong) AVAudioPlayer *audioPlayer;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放器
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UILabel *controlPanel; </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//控制面板
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIProgressView *playProgress;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放进度
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UILabel *musicSinger; </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//演唱者
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *playOrPause; </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放/暂停按钮(如果tag为0认为是暂停状态,1是播放状态)

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak ,nonatomic) NSTimer *timer;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//进度更新定时器

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@end

@implementation ViewController

- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];

    [self setupUI];
    
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  初始化UI
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setupUI{
    self.title=kMusicTitle;
    self.musicSinger.text=kMusicSinger;
}

-(NSTimer *)timer{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_timer) {
        _timer=[NSTimer scheduledTimerWithTimeInterval:0.5 target:self selector:@selector(updateProgress) userInfo:nil repeats:</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">true</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">];
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_timer;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  创建播放器
 *
 *  @return 音频播放器
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(AVAudioPlayer *)audioPlayer{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_audioPlayer) {
        NSString *urlStr=[[NSBundle mainBundle]pathForResource:kMusicFile ofType:nil];
        NSURL *url=[NSURL fileURLWithPath:urlStr];
        NSError *error=nil;
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//初始化播放器,注意这里的Url参数只能时文件路径,不支持HTTP Url
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_audioPlayer=[[AVAudioPlayer alloc]initWithContentsOfURL:url error:&error];
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置播放器属性
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_audioPlayer.numberOfLoops=0;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置为0不循环
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_audioPlayer.</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">delegate</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">=self;
        [_audioPlayer prepareToPlay];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//加载音频文件到缓存
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(error){
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"初始化播放器过程发生错误,错误信息:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">nil;
        }
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_audioPlayer;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  播放音频
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)play{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(![self.audioPlayer isPlaying]) {
        [self.audioPlayer play];
        self.timer.fireDate=[NSDate distantPast];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//恢复定时器
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  暂停播放
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)pause{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([self.audioPlayer isPlaying]) {
        [self.audioPlayer pause];
        self.timer.fireDate=[NSDate distantFuture];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//暂停定时器,注意不能调用invalidate方法,此方法会取消,之后无法恢复
        
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  点击播放/暂停按钮
 *
 *  @param sender 播放/暂停按钮
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">- (IBAction)playClick:(UIButton *)sender {
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(sender.tag){
        sender.tag=0;
        [sender setImage:[UIImage imageNamed:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"playing_btn_play_n"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">] forState:UIControlStateNormal];
        [sender setImage:[UIImage imageNamed:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"playing_btn_play_h"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">] forState:UIControlStateHighlighted];
        [self pause];
    }</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">{
        sender.tag=1;
        [sender setImage:[UIImage imageNamed:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"playing_btn_pause_n"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">] forState:UIControlStateNormal];
        [sender setImage:[UIImage imageNamed:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"playing_btn_pause_h"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">] forState:UIControlStateHighlighted];
        [self play];
    }
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  更新播放进度
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)updateProgress{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">float </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">progress= self.audioPlayer.currentTime /self.audioPlayer.duration;
    [self.playProgress setProgress:progress animated:</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">true</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">];
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 播放器代理方法
-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)audioPlayerDidFinishPlaying:(AVAudioPlayer *)player successfully:(BOOL)flag{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"音乐播放完成..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}

@end</span>
运行效果: 

AVAudioPlayer

音频会话

事实上上面的播放器还存在一些问题,例如通常我们看到的播放器即使退出到后台也是可以播放的,而这个播放器如果退出到后台它会自动暂停。如果要支持后台播放需要做下面几件事情:

1.设置后台运行模式:在plist文件中添加Required background modes,并且设置item 0=App plays audio or streams audio/video using AirPlay(其实可以直接通过Xcode在Project Targets-Capabilities-Background Modes中设置)

BackgroundModes

2.设置AVAudioSession的类型为AVAudioSessionCategoryPlayback并且调用setActive::方法启动会话。

<span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">    AVAudioSession *audioSession=[AVAudioSession sharedInstance];
    [audioSession setCategory:AVAudioSessionCategoryPlayback error:nil];
    [audioSession setActive:YES error:nil];</span>

3.为了能够让应用退到后台之后支持耳机控制,建议添加远程控制事件(这一步不是后台播放必须的)

前两步是后台播放所必须设置的,第三步主要用于接收远程事件,这部分内容之前的文章中有详细介绍,如果这一步不设置虽让也能够在后台播放,但是无法获得音频控制权(如果在使用当前应用之前使用其他播放器播放音乐的话,此时如果按耳机播放键或者控制中心的播放按钮则会播放前一个应用的音频),并且不能使用耳机进行音频控制。第一步操作相信大家都很容易理解,如果应用程序要允许运行到后台必须设置,正常情况下应用如果进入后台会被挂起,通过该设置可以上应用程序继续在后台运行。但是第二步使用的AVAudioSession有必要进行一下详细的说明。

在iOS中每个应用都有一个音频会话,这个会话就通过AVAudioSession来表示。AVAudioSession同样存在于AVFoundation框架中,它是单例模式设计,通过sharedInstance进行访问。在使用Apple设备时大家会发现有些应用只要打开其他音频播放就会终止,而有些应用却可以和其他应用同时播放,在多种音频环境中如何去控制播放的方式就是通过音频会话来完成的。下面是音频会话的几种会话模式:

会话类型说明是否要求输入是否要求输出是否遵从静音键
AVAudioSessionCategoryAmbient混音播放,可以与其他音频应用同时播放
AVAudioSessionCategorySoloAmbient独占播放
AVAudioSessionCategoryPlayback后台播放,也是独占的
AVAudioSessionCategoryRecord录音模式,用于录音时使用
AVAudioSessionCategoryPlayAndRecord播放和录音,此时可以录音也可以播放
AVAudioSessionCategoryAudioProcessing硬件解码音频,此时不能播放和录制 
AVAudioSessionCategoryMultiRoute多种输入输出,例如可以耳机、USB设备同时播放

注意:是否遵循静音键表示在播放过程中如果用户通过硬件设置为静音是否能关闭声音。

根据前面对音频会话的理解,相信大家开发出能够在后台播放的音频播放器并不难,但是注意一下,在前面的代码中也提到设置完音频会话类型之后需要调用setActive::方法将会话激活才能起作用。类似的,如果一个应用已经在播放音频,打开我们的应用之后设置了在后台播放的会话类型,此时其他应用的音频会停止而播放我们的音频,如果希望我们的程序音频播放完之后(关闭或退出到后台之后)能够继续播放其他应用的音频的话则可以调用setActive::方法关闭会话。代码如下:

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  KCAVAudioPlayer
//
//  Created by Kenshin Cui on 14/03/30.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//  AVAudioSession 音频会话

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><AVFoundation/AVFoundation.h>
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#define </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">kMusicFile @</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"刘若英 - 原来你也在这里.mp3"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#define </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">kMusicSinger @</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"刘若英"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#define </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">kMusicTitle @</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"原来你也在这里"

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()<AVAudioPlayerDelegate>

@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(nonatomic,strong) AVAudioPlayer *audioPlayer;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放器
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UILabel *controlPanel; </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//控制面板
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIProgressView *playProgress;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放进度
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UILabel *musicSinger; </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//演唱者
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *playOrPause; </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放/暂停按钮(如果tag为0认为是暂停状态,1是播放状态)

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak ,nonatomic) NSTimer *timer;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//进度更新定时器

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@end

@implementation ViewController

- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];

    [self setupUI];
    
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  显示当面视图控制器时注册远程事件
 *
 *  @param animated 是否以动画的形式显示
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewWillAppear:(BOOL)animated{
    [super viewWillAppear:animated];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//开启远程控制
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[[UIApplication sharedApplication] beginReceivingRemoteControlEvents];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//作为第一响应者
    //[self becomeFirstResponder];
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  当前控制器视图不显示时取消远程控制
 *
 *  @param animated 是否以动画的形式消失
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewWillDisappear:(BOOL)animated{
    [super viewWillDisappear:animated];
    [[UIApplication sharedApplication] endReceivingRemoteControlEvents];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//[self resignFirstResponder];
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  初始化UI
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setupUI{
    self.title=kMusicTitle;
    self.musicSinger.text=kMusicSinger;
}

-(NSTimer *)timer{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_timer) {
        _timer=[NSTimer scheduledTimerWithTimeInterval:0.5 target:self selector:@selector(updateProgress) userInfo:nil repeats:</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">true</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">];
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_timer;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  创建播放器
 *
 *  @return 音频播放器
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(AVAudioPlayer *)audioPlayer{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_audioPlayer) {
        NSString *urlStr=[[NSBundle mainBundle]pathForResource:kMusicFile ofType:nil];
        NSURL *url=[NSURL fileURLWithPath:urlStr];
        NSError *error=nil;
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//初始化播放器,注意这里的Url参数只能时文件路径,不支持HTTP Url
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_audioPlayer=[[AVAudioPlayer alloc]initWithContentsOfURL:url error:&error];
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置播放器属性
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_audioPlayer.numberOfLoops=0;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置为0不循环
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_audioPlayer.</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">delegate</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">=self;
        [_audioPlayer prepareToPlay];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//加载音频文件到缓存
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(error){
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"初始化播放器过程发生错误,错误信息:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">nil;
        }
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置后台播放模式
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVAudioSession *audioSession=[AVAudioSession sharedInstance];
        [audioSession setCategory:AVAudioSessionCategoryPlayback error:nil];
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//        [audioSession setCategory:AVAudioSessionCategoryPlayback withOptions:AVAudioSessionCategoryOptionAllowBluetooth error:nil];
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[audioSession setActive:YES error:nil];
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//添加通知,拔出耳机后暂停播放
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(routeChange:) name:AVAudioSessionRouteChangeNotification object:nil];
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_audioPlayer;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  播放音频
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)play{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(![self.audioPlayer isPlaying]) {
        [self.audioPlayer play];
        self.timer.fireDate=[NSDate distantPast];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//恢复定时器
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  暂停播放
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)pause{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([self.audioPlayer isPlaying]) {
        [self.audioPlayer pause];
        self.timer.fireDate=[NSDate distantFuture];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//暂停定时器,注意不能调用invalidate方法,此方法会取消,之后无法恢复
        
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  点击播放/暂停按钮
 *
 *  @param sender 播放/暂停按钮
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">- (IBAction)playClick:(UIButton *)sender {
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(sender.tag){
        sender.tag=0;
        [sender setImage:[UIImage imageNamed:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"playing_btn_play_n"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">] forState:UIControlStateNormal];
        [sender setImage:[UIImage imageNamed:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"playing_btn_play_h"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">] forState:UIControlStateHighlighted];
        [self pause];
    }</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">{
        sender.tag=1;
        [sender setImage:[UIImage imageNamed:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"playing_btn_pause_n"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">] forState:UIControlStateNormal];
        [sender setImage:[UIImage imageNamed:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"playing_btn_pause_h"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">] forState:UIControlStateHighlighted];
        [self play];
    }
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  更新播放进度
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)updateProgress{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">float </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">progress= self.audioPlayer.currentTime /self.audioPlayer.duration;
    [self.playProgress setProgress:progress animated:</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">true</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  一旦输出改变则执行此方法
 *
 *  @param notification 输出改变通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)routeChange:(NSNotification *)notification{
    NSDictionary *dic=notification.userInfo;
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">int </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">changeReason= [dic[AVAudioSessionRouteChangeReasonKey] intValue];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//等于AVAudioSessionRouteChangeReasonOldDeviceUnavailable表示旧输出不可用
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(changeReason==AVAudioSessionRouteChangeReasonOldDeviceUnavailable) {
        AVAudioSessionRouteDescription *routeDescription=dic[AVAudioSessionRouteChangePreviousRouteKey];
        AVAudioSessionPortDescription *portDescription= [routeDescription.outputs firstObject];
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//原设备为耳机则暂停
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([portDescription.portType isEqualToString:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"Headphones"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">]) {
            [self pause];
        }
    }
    
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//    [dic enumerateKeysAndObjectsUsingBlock:^(id key, id obj, BOOL *stop) {
//        NSLog(@"%@:%@",key,obj);
//    }];
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)dealloc{
    [[NSNotificationCenter defaultCenter] removeObserver:self name:AVAudioSessionRouteChangeNotification object:nil];
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 播放器代理方法
-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)audioPlayerDidFinishPlaying:(AVAudioPlayer *)player successfully:(BOOL)flag{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"音乐播放完成..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//根据实际情况播放完成可以将会话关闭,其他音频应用继续播放
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[[AVAudioSession sharedInstance]setActive:NO error:nil];
}

@end</span>

在上面的代码中还实现了拔出耳机暂停音乐播放的功能,这也是一个比较常见的功能。在iOS7及以后的版本中可以通过通知获得输出改变的通知,然后拿到通知对象后根据userInfo获得是何种改变类型,进而根据情况对音乐进行暂停操作。

扩展--播放音乐库中的音乐

众所周知音乐是iOS的重要组成播放,无论是iPod、iTouch、iPhone还是iPad都可以在iTunes购买音乐或添加本地音乐到音乐库中同步到你的iOS设备。在MediaPlayer.frameowork中有一个MPMusicPlayerController用于播放音乐库中的音乐。

下面先来看一下MPMusicPlayerController的常用属性和方法:

属性说明
@property (nonatomic, readonly) MPMusicPlaybackState playbackState播放器状态,枚举类型:
MPMusicPlaybackStateStopped:停止播放 MPMusicPlaybackStatePlaying:正在播放
MPMusicPlaybackStatePaused:暂停播放
MPMusicPlaybackStateInterrupted:播放中断
MPMusicPlaybackStateSeekingForward:向前查找
MPMusicPlaybackStateSeekingBackward:向后查找
@property (nonatomic) MPMusicRepeatMode repeatMode重复模式,枚举类型:
MPMusicRepeatModeDefault:默认模式,使用用户的首选项(系统音乐程序设置)
MPMusicRepeatModeNone:不重复
MPMusicRepeatModeOne:单曲循环
MPMusicRepeatModeAll:在当前列表内循环
@property (nonatomic) MPMusicShuffleMode shuffleMode随机播放模式,枚举类型:
MPMusicShuffleModeDefault:默认模式,使用用户首选项(系统音乐程序设置)
MPMusicShuffleModeOff:不随机播放
MPMusicShuffleModeSongs:按歌曲随机播放
MPMusicShuffleModeAlbums:按专辑随机播放
@property (nonatomic, copy) MPMediaItem *nowPlayingItem正在播放的音乐项
@property (nonatomic, readonly) NSUInteger indexOfNowPlayingItem 当前正在播放的音乐在播放队列中的索引
@property(nonatomic, readonly) BOOL isPreparedToPlay是否准好播放准备
@property(nonatomic) NSTimeInterval currentPlaybackTime当前已播放时间,单位:秒
@property(nonatomic) float currentPlaybackRate当前播放速度,是一个播放速度倍率,0表示暂停播放,1代表正常速度
类方法说明
+ (MPMusicPlayerController *)applicationMusicPlayer;获取应用播放器,注意此类播放器无法在后台播放
+ (MPMusicPlayerController *)systemMusicPlayer获取系统播放器,支持后台播放
对象方法说明
- (void)setQueueWithQuery:(MPMediaQuery *)query使用媒体队列设置播放源媒体队列
- (void)setQueueWithItemCollection:(MPMediaItemCollection *)itemCollection使用媒体项集合设置播放源媒体队列
- (void)skipToNextItem下一曲
- (void)skipToBeginning从起始位置播放
- (void)skipToPreviousItem上一曲
- (void)beginGeneratingPlaybackNotifications开启播放通知,注意不同于其他播放器,MPMusicPlayerController要想获得通知必须首先开启,默认情况无法获得通知
- (void)endGeneratingPlaybackNotifications关闭播放通知
- (void)prepareToPlay做好播放准备(加载音频到缓冲区),在使用play方法播放时如果没有做好准备回自动调用该方法
- (void)play开始播放
- (void)pause暂停播放
- (void)stop停止播放
- (void)beginSeekingForward开始向前查找(快进)
- (void)beginSeekingBackward开始向后查找(快退)
- (void)endSeeking结束查找
通知说明
(注意:要想获得MPMusicPlayerController通知必须首先调用beginGeneratingPlaybackNotifications开启通知)
MPMusicPlayerControllerPlaybackStateDidChangeNotification播放状态改变
MPMusicPlayerControllerNowPlayingItemDidChangeNotification当前播放音频改变
MPMusicPlayerControllerVolumeDidChangeNotification声音大小改变
MPMediaPlaybackIsPreparedToPlayDidChangeNotification准备好播放
  • MPMusicPlayerController有两种播放器:applicationMusicPlayer和systemMusicPlayer,前者在应用退出后音乐播放会自动停止,后者在应用停止后不会退出播放状态。 
  • MPMusicPlayerController加载音乐不同于前面的AVAudioPlayer是通过一个文件路径来加载,而是需要一个播放队列。在MPMusicPlayerController中提供了两个方法来加载播放队列:- (void)setQueueWithQuery:(MPMediaQuery *)query- (void)setQueueWithItemCollection:(MPMediaItemCollection *)itemCollection,正是由于它的播放音频来源是一个队列,因此MPMusicPlayerController支持上一曲、下一曲等操作。

那么接下来的问题就是如何获取MPMediaQueue或者MPMediaItemCollection?MPMediaQueue对象有一系列的类方法来获得媒体队列:

+ (MPMediaQuery *)albumsQuery;
+ (MPMediaQuery *)artistsQuery;
+ (MPMediaQuery *)songsQuery;
+ (MPMediaQuery *)playlistsQuery;
+ (MPMediaQuery *)podcastsQuery;
+ (MPMediaQuery *)audiobooksQuery;
+ (MPMediaQuery *)compilationsQuery;
+ (MPMediaQuery *)composersQuery;
+ (MPMediaQuery *)genresQuery;

有了这些方法,就可以很容易获到歌曲、播放列表、专辑媒体等媒体队列了,这样就可以通过:- (void)setQueueWithQuery:(MPMediaQuery *)query方法设置音乐来源了又或者得到MPMediaQueue之后创建MPMediaItemCollection,使用- (void)setQueueWithItemCollection:(MPMediaItemCollection *)itemCollection设置音乐来源。

有时候可能希望用户自己来选择要播放的音乐,这时可以使用MPMediaPickerController,它是一个视图控制器,类似于UIImagePickerController,选择完播放来源后可以在其代理方法中获得MPMediaItemCollection对象。

无论是通过哪种方式获得MPMusicPlayerController的媒体源,可能都希望将每个媒体的信息显示出来,这时候可以通过MPMediaItem对象获得。一个MPMediaItem代表一个媒体文件,通过它可以访问媒体标题、专辑名称、专辑封面、音乐时长等等。无论是MPMediaQueue还是MPMediaItemCollection都有一个items属性,它是MPMediaItem数组,通过这个属性可以获得MPMediaItem对象。

下面就简单看一下MPMusicPlayerController的使用,在下面的例子中简单演示了音乐的选择、播放、暂停、通知、下一曲、上一曲功能,相信有了上面的概念,代码读起来并不复杂(示例中是直接通过MPMeidaPicker进行音乐选择的,但是仍然提供了两个方法getLocalMediaQuery和getLocalMediaItemCollection来演示如何直接通过MPMediaQueue获得媒体队列或媒体集合):

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  MPMusicPlayerController
//
//  Created by Kenshin Cui 14/03/30
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><MediaPlayer/MediaPlayer.h>

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()<MPMediaPickerControllerDelegate>

@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(nonatomic,strong) MPMediaPickerController *mediaPicker;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//媒体选择控制器
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(nonatomic,strong) MPMusicPlayerController *musicPlayer; </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//音乐播放器

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@end

@implementation ViewController

- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)dealloc{
    [self.musicPlayer endGeneratingPlaybackNotifications];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  获得音乐播放器
 *
 *  @return 音乐播放器
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(MPMusicPlayerController *)musicPlayer{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_musicPlayer) {
        _musicPlayer=[MPMusicPlayerController systemMusicPlayer];
        [_musicPlayer beginGeneratingPlaybackNotifications];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//开启通知,否则监控不到MPMusicPlayerController的通知
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self addNotification];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//添加通知
        //如果不使用MPMediaPickerController可以使用如下方法获得音乐库媒体队列
        //[_musicPlayer setQueueWithItemCollection:[self getLocalMediaItemCollection]];
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_musicPlayer;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  创建媒体选择器
 *
 *  @return 媒体选择器
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(MPMediaPickerController *)mediaPicker{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_mediaPicker) {
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//初始化媒体选择器,这里设置媒体类型为音乐,其实这里也可以选择视频、广播等
//        _mediaPicker=[[MPMediaPickerController alloc]initWithMediaTypes:MPMediaTypeMusic];
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_mediaPicker=[[MPMediaPickerController alloc]initWithMediaTypes:MPMediaTypeAny];
        _mediaPicker.allowsPickingMultipleItems=YES;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//允许多选
//        _mediaPicker.showsCloudItems=YES;//显示icloud选项
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_mediaPicker.prompt=@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"请选择要播放的音乐"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        _mediaPicker.</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">delegate</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">=self;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置选择器代理
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_mediaPicker;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得媒体队列
 *
 *  @return 媒体队列
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(MPMediaQuery *)getLocalMediaQuery{
    MPMediaQuery *mediaQueue=[MPMediaQuery songsQuery];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">for </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(MPMediaItem *item in mediaQueue.items) {
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"标题:%@,%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,item.title,item.albumTitle);
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mediaQueue;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得媒体集合
 *
 *  @return 媒体集合
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(MPMediaItemCollection *)getLocalMediaItemCollection{
    MPMediaQuery *mediaQueue=[MPMediaQuery songsQuery];
    NSMutableArray *</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">array</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">=[NSMutableArray </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">array</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">for </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(MPMediaItem *item in mediaQueue.items) {
        [</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">array </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">addObject:item];
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"标题:%@,%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,item.title,item.albumTitle);
    }
    MPMediaItemCollection *mediaItemCollection=[[MPMediaItemCollection alloc]initWithItems:[</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">array </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">copy]];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mediaItemCollection;
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - MPMediaPickerController代理方法
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//选择完成
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)mediaPicker:(MPMediaPickerController *)mediaPicker didPickMediaItems:(MPMediaItemCollection *)mediaItemCollection{
    MPMediaItem *mediaItem=[mediaItemCollection.items firstObject];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//第一个播放音乐
    //注意很多音乐信息如标题、专辑、表演者、封面、时长等信息都可以通过MPMediaItem的valueForKey:方法得到,但是从iOS7开始都有对应的属性可以直接访问
//    NSString *title= [mediaItem valueForKey:MPMediaItemPropertyAlbumTitle];
//    NSString *artist= [mediaItem valueForKey:MPMediaItemPropertyAlbumArtist];
//    MPMediaItemArtwork *artwork= [mediaItem valueForKey:MPMediaItemPropertyArtwork];
    //UIImage *image=[artwork imageWithSize:CGSizeMake(100, 100)];//专辑图片
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"标题:%@,表演者:%@,专辑:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,mediaItem.title ,mediaItem.artist,mediaItem.albumTitle);
    [self.musicPlayer setQueueWithItemCollection:mediaItemCollection];
    [self dismissViewControllerAnimated:YES completion:nil];
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//取消选择
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)mediaPickerDidCancel:(MPMediaPickerController *)mediaPicker{
    [self dismissViewControllerAnimated:YES completion:nil];
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 通知
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  添加通知
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addNotification{
    NSNotificationCenter *notificationCenter=[NSNotificationCenter defaultCenter];
    [notificationCenter addObserver:self selector:@selector(playbackStateChange:) name:MPMusicPlayerControllerPlaybackStateDidChangeNotification object:self.musicPlayer];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  播放状态改变通知
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)playbackStateChange:(NSNotification *)notification{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">switch </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(self.musicPlayer.playbackState) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">MPMusicPlaybackStatePlaying:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"正在播放..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">MPMusicPlaybackStatePaused:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"播放暂停."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">MPMusicPlaybackStateStopped:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"播放停止."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">default</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">:
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    }
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - UI事件
- (IBAction)selectClick:(UIButton *)sender {
    [self presentViewController:self.mediaPicker animated:YES completion:nil];
}

- (IBAction)playClick:(UIButton *)sender {
    [self.musicPlayer play];
}

- (IBAction)puaseClick:(UIButton *)sender {
    [self.musicPlayer pause];
}

- (IBAction)stopClick:(UIButton *)sender {
    [self.musicPlayer stop];
}

- (IBAction)nextClick:(UIButton *)sender {
    [self.musicPlayer skipToNextItem];
}

- (IBAction)prevClick:(UIButton *)sender {
    [self.musicPlayer skipToPreviousItem];
}

@end</span>

录音

除了上面说的,在AVFoundation框架中还要一个AVAudioRecorder类专门处理录音操作,它同样支持多种音频格式。与AVAudioPlayer类似,你完全可以将它看成是一个录音机控制类,下面是常用的属性和方法:

属性说明
@property(readonly, getter=isRecording) BOOL recording;是否正在录音,只读
@property(readonly) NSURL *url录音文件地址,只读
@property(readonly) NSDictionary *settings录音文件设置,只读
@property(readonly) NSTimeInterval currentTime录音时长,只读,注意仅仅在录音状态可用
@property(readonly) NSTimeInterval deviceCurrentTime输入设置的时间长度,只读,注意此属性一直可访问
@property(getter=isMeteringEnabled) BOOL meteringEnabled;是否启用录音测量,如果启用录音测量可以获得录音分贝等数据信息
@property(nonatomic, copy) NSArray *channelAssignments当前录音的通道
对象方法说明
- (instancetype)initWithURL:(NSURL *)url settings:(NSDictionary *)settings error:(NSError **)outError录音机对象初始化方法,注意其中的url必须是本地文件url,settings是录音格式、编码等设置
- (BOOL)prepareToRecord准备录音,主要用于创建缓冲区,如果不手动调用,在调用record录音时也会自动调用
- (BOOL)record开始录音
- (BOOL)recordAtTime:(NSTimeInterval)time在指定的时间开始录音,一般用于录音暂停再恢复录音
- (BOOL)recordForDuration:(NSTimeInterval) duration按指定的时长开始录音
- (BOOL)recordAtTime:(NSTimeInterval)time forDuration:(NSTimeInterval) duration在指定的时间开始录音,并指定录音时长
- (void)pause;暂停录音
- (void)stop;停止录音
- (BOOL)deleteRecording;删除录音,注意要删除录音此时录音机必须处于停止状态
- (void)updateMeters;更新测量数据,注意只有meteringEnabled为YES此方法才可用
- (float)peakPowerForChannel:(NSUInteger)channelNumber;指定通道的测量峰值,注意只有调用完updateMeters才有值
- (float)averagePowerForChannel:(NSUInteger)channelNumber指定通道的测量平均值,注意只有调用完updateMeters才有值
代理方法说明
- (void)audioRecorderDidFinishRecording:(AVAudioRecorder *)recorder successfully:(BOOL)flag完成录音
- (void)audioRecorderEncodeErrorDidOccur:(AVAudioRecorder *)recorder error:(NSError *)error录音编码发生错误

AVAudioRecorder很多属性和方法跟AVAudioPlayer都是类似的,但是它的创建有所不同,在创建录音机时除了指定路径外还必须指定录音设置信息,因为录音机必须知道录音文件的格式、采样率、通道数、每个采样点的位数等信息,但是也并不是所有的信息都必须设置,通常只需要几个常用设置。关于录音设置详见帮助文档中的“AV Foundation Audio Settings Constants”。

下面就使用AVAudioRecorder创建一个录音机,实现了录音、暂停、停止、播放等功能,实现效果大致如下:

AVAudioRecorderSnapshot

在这个示例中将实行一个完整的录音控制,包括录音、暂停、恢复、停止,同时还会实时展示用户录音的声音波动,当用户点击完停止按钮还会自动播放录音文件。程序的构建主要分为以下几步:

  1. 设置音频会话类型为AVAudioSessionCategoryPlayAndRecord,因为程序中牵扯到录音和播放操作。 
  2. 创建录音机AVAudioRecorder,指定录音保存的路径并且设置录音属性,注意对于一般的录音文件要求的采样率、位数并不高,需要适当设置以保证录音文件的大小和效果。 
  3. 设置录音机代理以便在录音完成后播放录音,打开录音测量保证能够实时获得录音时的声音强度。(注意声音强度范围-160到0,0代表最大输入) 
  4. 创建音频播放器AVAudioPlayer,用于在录音完成之后播放录音。 
  5. 创建一个定时器以便实时刷新录音测量值并更新录音强度到UIProgressView中显示。 
  6. 添加录音、暂停、恢复、停止操作,需要注意录音的恢复操作其实是有音频会话管理的,恢复时只要再次调用record方法即可,无需手动管理恢复时间等。

下面是主要代码:

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  AVAudioRecorder
//
//  Created by Kenshin Cui on 14/03/30.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><AVFoundation/AVFoundation.h>
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#define </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">kRecordAudioFile @</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"myRecord.caf"

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()<AVAudioRecorderDelegate>

@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(nonatomic,strong) AVAudioRecorder *audioRecorder;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//音频录音机
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(nonatomic,strong) AVAudioPlayer *audioPlayer;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//音频播放器,用于播放录音文件
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(nonatomic,strong) NSTimer *timer;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//录音声波监控(注意这里暂时不对播放进行监控)

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *record;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//开始录音
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *pause;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//暂停录音
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *resume;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//恢复录音
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *stop;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//停止录音
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIProgressView *audioPower;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//音频波动

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@end

@implementation ViewController

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 控制器视图方法
- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];
    
    [self setAudioSession];
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 私有方法
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置音频会话
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setAudioSession{
    AVAudioSession *audioSession=[AVAudioSession sharedInstance];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置为播放和录音状态,以便可以在录制完之后播放录音
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[audioSession setCategory:AVAudioSessionCategoryPlayAndRecord error:nil];
    [audioSession setActive:YES error:nil];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得录音文件保存路径
 *
 *  @return 录音文件路径
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(NSURL *)getSavePath{
    NSString *urlStr=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];
    urlStr=[urlStr stringByAppendingPathComponent:kRecordAudioFile];
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"file path:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,urlStr);
    NSURL *url=[NSURL fileURLWithPath:urlStr];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">url;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得录音文件设置
 *
 *  @return 录音设置
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(NSDictionary *)getAudioSetting{
    NSMutableDictionary *dicM=[NSMutableDictionary dictionary];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置录音格式
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[dicM setObject:@(kAudioFormatLinearPCM) forKey:AVFormatIDKey];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置录音采样率,8000是电话采样率,对于一般录音已经够了
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[dicM setObject:@(8000) forKey:AVSampleRateKey];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置通道,这里采用单声道
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[dicM setObject:@(1) forKey:AVNumberOfChannelsKey];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//每个采样点位数,分为8、16、24、32
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[dicM setObject:@(8) forKey:AVLinearPCMBitDepthKey];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//是否使用浮点数采样
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[dicM setObject:@(YES) forKey:AVLinearPCMIsFloatKey];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//....其他设置等
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">dicM;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  获得录音机对象
 *
 *  @return 录音机对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(AVAudioRecorder *)audioRecorder{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_audioRecorder) {
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//创建录音文件保存路径
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">NSURL *url=[self getSavePath];
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//创建录音格式设置
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">NSDictionary *setting=[self getAudioSetting];
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//创建录音机
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">NSError *error=nil;
        _audioRecorder=[[AVAudioRecorder alloc]initWithURL:url settings:setting error:&error];
        _audioRecorder.</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">delegate</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">=self;
        _audioRecorder.meteringEnabled=YES;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//如果要监控声波则必须设置为YES
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(error) {
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"创建录音机对象时发生错误,错误信息:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">nil;
        }
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_audioRecorder;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  创建播放器
 *
 *  @return 播放器
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(AVAudioPlayer *)audioPlayer{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_audioPlayer) {
        NSURL *url=[self getSavePath];
        NSError *error=nil;
        _audioPlayer=[[AVAudioPlayer alloc]initWithContentsOfURL:url error:&error];
        _audioPlayer.numberOfLoops=0;
        [_audioPlayer prepareToPlay];
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(error) {
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"创建播放器过程中发生错误,错误信息:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">nil;
        }
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_audioPlayer;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  录音声波监控定制器
 *
 *  @return 定时器
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(NSTimer *)timer{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_timer) {
        _timer=[NSTimer scheduledTimerWithTimeInterval:0.1f target:self selector:@selector(audioPowerChange) userInfo:nil repeats:YES];
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_timer;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  录音声波状态设置
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)audioPowerChange{
    [self.audioRecorder updateMeters];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//更新测量值
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">float </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">power= [self.audioRecorder averagePowerForChannel:0];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//取得第一个通道的音频,注意音频强度范围时-160到0
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">CGFloat progress=(1.0/160.0)*(power+160.0);
    [self.audioPower setProgress:progress];
}
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - UI事件
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  点击录音按钮
 *
 *  @param sender 录音按钮
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">- (IBAction)recordClick:(UIButton *)sender {
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(![self.audioRecorder isRecording]) {
        [self.audioRecorder record];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//首次使用应用时如果调用record方法会询问用户是否允许使用麦克风
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">self.timer.fireDate=[NSDate distantPast];
    }
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  点击暂定按钮
 *
 *  @param sender 暂停按钮
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">- (IBAction)pauseClick:(UIButton *)sender {
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([self.audioRecorder isRecording]) {
        [self.audioRecorder pause];
        self.timer.fireDate=[NSDate distantFuture];
    }
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  点击恢复按钮
 *  恢复录音只需要再次调用record,AVAudioSession会帮助你记录上次录音位置并追加录音
 *
 *  @param sender 恢复按钮
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">- (IBAction)resumeClick:(UIButton *)sender {
    [self recordClick:sender];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  点击停止按钮
 *
 *  @param sender 停止按钮
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">- (IBAction)stopClick:(UIButton *)sender {
    [self.audioRecorder stop];
    self.timer.fireDate=[NSDate distantFuture];
    self.audioPower.progress=0.0;
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 录音机代理方法
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  录音完成,录音完成后播放录音
 *
 *  @param recorder 录音机对象
 *  @param flag     是否成功
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)audioRecorderDidFinishRecording:(AVAudioRecorder *)recorder successfully:(BOOL)flag{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(![self.audioPlayer isPlaying]) {
        [self.audioPlayer play];
    }
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"录音完成!"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}

@end</span>

运行效果:

AVAudioRecorder

音频队列服务

大家应该已经注意到了,无论是前面的录音还是音频播放均不支持网络流媒体播放,当然对于录音来说这种需求可能不大,但是对于音频播放来说有时候就很有必要了。AVAudioPlayer只能播放本地文件,并且是一次性加载所以音频数据,初始化AVAudioPlayer时指定的URL也只能是File URL而不能是HTTP URL。当然,将音频文件下载到本地然后再调用AVAudioPlayer来播放也是一种播放网络音频的办法,但是这种方式最大的弊端就是必须等到整个音频播放完成才能播放,而不能使用流式播放,这往往在实际开发中是不切实际的。那么在iOS中如何播放网络流媒体呢?就是使用AudioToolbox框架中的音频队列服务Audio Queue Services。

使用音频队列服务完全可以做到音频播放和录制,首先看一下录音音频服务队列:

recording_architecture_2x

一个音频服务队列Audio Queue有三部分组成:

三个缓冲器Buffers:每个缓冲器都是一个存储音频数据的临时仓库。

一个缓冲队列Buffer Queue:一个包含音频缓冲器的有序队列。

一个回调Callback:一个自定义的队列回调函数。

声音通过输入设备进入缓冲队列中,首先填充第一个缓冲器;当第一个缓冲器填充满之后自动填充下一个缓冲器,同时会调用回调函数;在回调函数中需要将缓冲器中的音频数据写入磁盘,同时将缓冲器放回到缓冲队列中以便重用。下面是Apple官方关于音频队列服务的流程示意图:

recording_callback_function_2x

类似的,看一下音频播放缓冲队列,其组成部分和录音缓冲队列类似。

playback_architecture_2x

但是在音频播放缓冲队列中,回调函数调用的时机不同于音频录制缓冲队列,流程刚好相反。将音频读取到缓冲器中,一旦一个缓冲器填充满之后就放到缓冲队列中,然后继续填充其他缓冲器;当开始播放时,则从第一个缓冲器中读取音频进行播放;一旦播放完之后就会触发回调函数,开始播放下一个缓冲器中的音频,同时填充第一个缓冲器放;填充满之后再次放回到缓冲队列。下面是详细的流程:

playback_callback_function_2x

当然,要明白音频队列服务的原理并不难,问题是如何实现这个自定义的回调函数,这其中我们有大量的工作要做,控制播放状态、处理异常中断、进行音频编码等等。由于牵扯内容过多,而且不是本文目的,如果以后有时间将另开一篇文章重点介绍,目前有很多第三方优秀框架可以直接使用,例如AudioStreamerFreeStreamer。由于前者当前只有非ARC版本,所以下面不妨使用FreeStreamer来简单演示在线音频播放的过程,当然在使用之前要做如下准备工作:

1.拷贝FreeStreamer中的Reachability.h、Reachability.m和Common、astreamer两个文件夹中的内容到项目中。

2.添加FreeStreamer使用的类库:CFNetwork.framework、AudioToolbox.framework、AVFoundation.framework
、libxml2.dylib、MediaPlayer.framework。

3.如果引用libxml2.dylib编译不通过,需要在Xcode的Targets-Build Settings-Header Build Path中添加$(SDKROOT)/usr/include/libxml2。

4.将FreeStreamer中的FreeStreamerMobile-Prefix.pch文件添加到项目中并将Targets-Build Settings-Precompile Prefix Header设置为YES,在Targets-Build Settings-Prefix Header设置为$(SRCROOT)/项目名称/FreeStreamerMobile-Prefix.pch(因为Xcode6默认没有pch文件)

然后就可以编写代码播放网络音频了:

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  AudioQueueServices
//
//  Created by Kenshin Cui on 14/03/30.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//  使用FreeStreamer实现网络音频播放

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"FSAudioStream.h"

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()

@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(nonatomic,strong) FSAudioStream *audioStream;

@end

@implementation ViewController

- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];

    [self.audioStream play];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得本地文件路径
 *
 *  @return 文件路径
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(NSURL *)getFileUrl{
    NSString *urlStr=[[NSBundle mainBundle]pathForResource:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"刘若英 - 原来你也在这里.mp3" </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ofType:nil];
    NSURL *url=[NSURL fileURLWithPath:urlStr];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">url;
}
-(NSURL *)getNetworkUrl{
    NSString *urlStr=@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"http://192.168.1.102/liu.mp3"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    NSURL *url=[NSURL URLWithString:urlStr];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">url;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  创建FSAudioStream对象
 *
 *  @return FSAudioStream对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(FSAudioStream *)audioStream{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_audioStream) {
        NSURL *url=[self getNetworkUrl];
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//创建FSAudioStream对象
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_audioStream=[[FSAudioStream alloc]initWithUrl:url];
        _audioStream.onFailure=^(FSAudioStreamError error,NSString *description){
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"播放过程中发生错误,错误信息:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,description);
        };
        _audioStream.onCompletion=^(){
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"播放完成!"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
        };
        [_audioStream setVolume:0.5];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置声音
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_audioStream;
}

@end</span>
其实FreeStreamer的功能很强大,不仅仅是播放本地、网络音频那么简单,它还支持播放列表、检查包内容、RSS订阅、播放中断等很多强大的功能,甚至还包含了一个音频分析器,有兴趣的朋友可以访问 官网 查看详细用法 

视频

MPMoviePlayerController

在iOS中播放视频可以使用MediaPlayer.framework种的MPMoviePlayerController类来完成,它支持本地视频和网络视频播放。这个类实现了MPMediaPlayback协议,因此具备一般的播放器控制功能,例如播放、暂停、停止等。但是MPMediaPlayerController自身并不是一个完整的视图控制器,如果要在UI中展示视频需要将view属性添加到界面中。下面列出了MPMoviePlayerController的常用属性和方法:

属性说明
@property (nonatomic, copy) NSURL *contentURL播放媒体URL,这个URL可以是本地路径,也可以是网络路径
@property (nonatomic, readonly) UIView *view播放器视图,如果要显示视频必须将此视图添加到控制器视图中
@property (nonatomic, readonly) UIView *backgroundView播放器背景视图
@property (nonatomic, readonly) MPMoviePlaybackState playbackState媒体播放状态,枚举类型:
MPMoviePlaybackStateStopped:停止播放
MPMoviePlaybackStatePlaying:正在播放
MPMoviePlaybackStatePaused:暂停
MPMoviePlaybackStateInterrupted:中断
MPMoviePlaybackStateSeekingForward:向前定位
MPMoviePlaybackStateSeekingBackward:向后定位
@property (nonatomic, readonly) MPMovieLoadState loadState网络媒体加载状态,枚举类型:
MPMovieLoadStateUnknown:位置类型
MPMovieLoadStatePlayable:
MPMovieLoadStatePlaythroughOK:这种状态如果shouldAutoPlay为YES将自动播放
MPMovieLoadStateStalled:停滞状态
@property (nonatomic) MPMovieControlStyle controlStyle控制面板风格,枚举类型:
MPMovieControlStyleNone:无控制面板 
MPMovieControlStyleEmbedded:嵌入视频风格 
MPMovieControlStyleFullscreen:全屏 
MPMovieControlStyleDefault:默认风格
@property (nonatomic) MPMovieRepeatMode repeatMode;重复播放模式,枚举类型:
MPMovieRepeatModeNone:不重复,默认值
MPMovieRepeatModeOne:重复播放
@property (nonatomic) BOOL shouldAutoplay当网络媒体缓存到一定数据时是否自动播放,默认为YES
@property (nonatomic, getter=isFullscreen) BOOL fullscreen是否全屏展示,默认为NO,注意如果要通过此属性设置全屏必须在视图显示完成后设置,否则无效
@property (nonatomic) MPMovieScalingMode scalingMode视频缩放填充模式,枚举类型:
MPMovieScalingModeNone:不进行任何缩放
MPMovieScalingModeAspectFit:固定缩放比例并且尽量全部展示视频,不会裁切视频
MPMovieScalingModeAspectFill:固定缩放比例并填充满整个视图展示,可能会裁切视频
MPMovieScalingModeFill:不固定缩放比例压缩填充整个视图,视频不会被裁切但是比例失衡
@property (nonatomic, readonly) BOOL readyForDisplay是否有相关媒体被播放
@property (nonatomic, readonly) MPMovieMediaTypeMask movieMediaTypes媒体类别,枚举类型:
MPMovieMediaTypeMaskNone:未知类型
MPMovieMediaTypeMaskVideo:视频
MPMovieMediaTypeMaskAudio:音频
@property (nonatomic) MPMovieSourceType movieSourceType媒体源,枚举类型:
MPMovieSourceTypeUnknown:未知来源
MPMovieSourceTypeFile:本地文件
MPMovieSourceTypeStreaming:流媒体(直播或点播)
@property (nonatomic, readonly) NSTimeInterval duration媒体时长,如果未知则返回0
@property (nonatomic, readonly) NSTimeInterval playableDuration媒体可播放时长,主要用于表示网络媒体已下载视频时长
@property (nonatomic, readonly) CGSize naturalSize视频实际尺寸,如果未知则返回CGSizeZero
@property (nonatomic) NSTimeInterval initialPlaybackTime起始播放时间
@property (nonatomic) NSTimeInterval endPlaybackTime终止播放时间
@property (nonatomic) BOOL allowsAirPlay是否允许无线播放,默认为YES
@property (nonatomic, readonly, getter=isAirPlayVideoActive) BOOL airPlayVideoActive当前媒体是否正在通过AirPlay播放
@property(nonatomic, readonly) BOOL isPreparedToPlay是否准备好播放
@property(nonatomic) NSTimeInterval currentPlaybackTime当前播放时间,单位:秒
@property(nonatomic) float currentPlaybackRate当前播放速度,如果暂停则为0,正常速度为1.0,非0数据表示倍率
对象方法说明
- (instancetype)initWithContentURL:(NSURL *)url使用指定的URL初始化媒体播放控制器对象
- (void)setFullscreen:(BOOL)fullscreen animated:(BOOL)animated设置视频全屏,注意如果要通过此方法设置全屏则必须在其视图显示之后设置,否则无效
- (void)requestThumbnailImagesAtTimes:(NSArray *)playbackTimes timeOption:(MPMovieTimeOption)option获取在指定播放时间的视频缩略图,第一个参数是获取缩略图的时间点数组;第二个参数代表时间点精度,枚举类型:
MPMovieTimeOptionNearestKeyFrame:时间点附近
MPMovieTimeOptionExact:准确时间
- (void)cancelAllThumbnailImageRequests取消所有缩略图获取请求
- (void)prepareToPlay准备播放,加载视频数据到缓存,当调用play方法时如果没有准备好会自动调用此方法
- (void)play开始播放
- (void)pause暂停播放
- (void)stop停止播放
- (void)beginSeekingForward向前定位
- (void)beginSeekingBackward向后定位
- (void)endSeeking停止快进/快退
通知说明
MPMoviePlayerScalingModeDidChangeNotification视频缩放填充模式发生改变
MPMoviePlayerPlaybackDidFinishNotification媒体播放完成或用户手动退出,具体完成原因可以通过通知userInfo中的key为MPMoviePlayerPlaybackDidFinishReasonUserInfoKey的对象获取
MPMoviePlayerPlaybackStateDidChangeNotification播放状态改变,可配合playbakcState属性获取具体状态
MPMoviePlayerLoadStateDidChangeNotification媒体网络加载状态改变
MPMoviePlayerNowPlayingMovieDidChangeNotification当前播放的媒体内容发生改变
MPMoviePlayerWillEnterFullscreenNotification将要进入全屏
MPMoviePlayerDidEnterFullscreenNotification进入全屏后
MPMoviePlayerWillExitFullscreenNotification将要退出全屏
MPMoviePlayerDidExitFullscreenNotification退出全屏后
MPMoviePlayerIsAirPlayVideoActiveDidChangeNotification当媒体开始通过AirPlay播放或者结束AirPlay播放
MPMoviePlayerReadyForDisplayDidChangeNotification视频显示状态改变
MPMovieMediaTypesAvailableNotification确定了媒体可用类型后
MPMovieSourceTypeAvailableNotification确定了媒体来源后
MPMovieDurationAvailableNotification确定了媒体播放时长后
MPMovieNaturalSizeAvailableNotification确定了媒体的实际尺寸后
MPMoviePlayerThumbnailImageRequestDidFinishNotification缩略图请求完成之后
MPMediaPlaybackIsPreparedToPlayDidChangeNotification做好播放准备后

注意MPMediaPlayerController的状态等信息并不是通过代理来和外界交互的,而是通过通知中心,因此从上面的列表中可以看到常用的一些通知。由于MPMoviePlayerController本身对于媒体播放做了深度的封装,使用起来就相当简单:创建MPMoviePlayerController对象,设置frame属性,将MPMoviePlayerController的view添加到控制器视图中。下面的示例中将创建一个播放控制器并添加播放状态改变及播放完成的通知:

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  MPMoviePlayerController
//
//  Created by Kenshin Cui on 14/03/30.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><MediaPlayer/MediaPlayer.h>

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()

@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(nonatomic,strong) MPMoviePlayerController *moviePlayer;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//视频播放控制器

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@end

@implementation ViewController

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 控制器视图方法
- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.moviePlayer play];
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//添加通知
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self addNotification];
    
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)dealloc{
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//移除所有通知监控
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[[NSNotificationCenter defaultCenter] removeObserver:self];
}


</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 私有方法
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得本地文件路径
 *
 *  @return 文件路径
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(NSURL *)getFileUrl{
    NSString *urlStr=[[NSBundle mainBundle] pathForResource:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"The New Look of OS X Yosemite.mp4" </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ofType:nil];
    NSURL *url=[NSURL fileURLWithPath:urlStr];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">url;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得网络文件路径
 *
 *  @return 文件路径
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(NSURL *)getNetworkUrl{
    NSString *urlStr=@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"http://192.168.1.161/The New Look of OS X Yosemite.mp4"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    urlStr=[urlStr stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
    NSURL *url=[NSURL URLWithString:urlStr];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">url;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  创建媒体播放控制器
 *
 *  @return 媒体播放控制器
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(MPMoviePlayerController *)moviePlayer{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_moviePlayer) {
        NSURL *url=[self getNetworkUrl];
        _moviePlayer=[[MPMoviePlayerController alloc]initWithContentURL:url];
        _moviePlayer.view.frame=self.view.bounds;
        _moviePlayer.view.autoresizingMask=UIViewAutoresizingFlexibleWidth|UIViewAutoresizingFlexibleHeight;
        [self.view addSubview:_moviePlayer.view];
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_moviePlayer;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  添加通知监控媒体播放控制器状态
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addNotification{
    NSNotificationCenter *notificationCenter=[NSNotificationCenter defaultCenter];
    [notificationCenter addObserver:self selector:@selector(mediaPlayerPlaybackStateChange:) name:MPMoviePlayerPlaybackStateDidChangeNotification object:self.moviePlayer];
    [notificationCenter addObserver:self selector:@selector(mediaPlayerPlaybackFinished:) name:MPMoviePlayerPlaybackDidFinishNotification object:self.moviePlayer];
    
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  播放状态改变,注意播放完成时的状态是暂停
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)mediaPlayerPlaybackStateChange:(NSNotification *)notification{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">switch </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(self.moviePlayer.playbackState) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">MPMoviePlaybackStatePlaying:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"正在播放..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">MPMoviePlaybackStatePaused:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"暂停播放."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">MPMoviePlaybackStateStopped:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"停止播放."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">default</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"播放状态:%li"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,self.moviePlayer.playbackState);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    }
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  播放完成
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)mediaPlayerPlaybackFinished:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"播放完成.%li"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,self.moviePlayer.playbackState);
}


@end</span>

运行效果:

MPMoviePlayerController
从上面的API大家也不难看出其实MPMoviePlayerController功能相当强大,日常开发中作为一般的媒体播放器也完全没有问题。MPMoviePlayerController除了一般的视频播放和控制外还有一些强大的功能,例如截取视频缩略图。请求视频缩略图时只要调用- (void)requestThumbnailImagesAtTimes:(NSArray *)playbackTimes timeOption:(MPMovieTimeOption)option方法指定获得缩略图的时间点,然后监控MPMoviePlayerThumbnailImageRequestDidFinishNotification通知,每个时间点的缩略图请求完成就会调用通知,在通知调用方法中可以通过MPMoviePlayerThumbnailImageKey获得UIImage对象处理即可。例如下面的程序演示了在程序启动后获得两个时间点的缩略图的过程,截图成功后保存到相册:

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  MPMoviePlayerController
//
//  Created by Kenshin Cui on 14/03/30.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//  视频截图

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><MediaPlayer/MediaPlayer.h>

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()

@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(nonatomic,strong) MPMoviePlayerController *moviePlayer;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//视频播放控制器

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@end

@implementation ViewController

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 控制器视图方法
- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.moviePlayer play];
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//添加通知
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self addNotification];
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//获取缩略图
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self thumbnailImageRequest];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)dealloc{
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//移除所有通知监控
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[[NSNotificationCenter defaultCenter] removeObserver:self];
}


</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 私有方法
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得本地文件路径
 *
 *  @return 文件路径
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(NSURL *)getFileUrl{
    NSString *urlStr=[[NSBundle mainBundle] pathForResource:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"The New Look of OS X Yosemite.mp4" </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ofType:nil];
    NSURL *url=[NSURL fileURLWithPath:urlStr];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">url;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得网络文件路径
 *
 *  @return 文件路径
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(NSURL *)getNetworkUrl{
    NSString *urlStr=@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"http://192.168.1.161/The New Look of OS X Yosemite.mp4"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    urlStr=[urlStr stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
    NSURL *url=[NSURL URLWithString:urlStr];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">url;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  创建媒体播放控制器
 *
 *  @return 媒体播放控制器
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(MPMoviePlayerController *)moviePlayer{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_moviePlayer) {
        NSURL *url=[self getNetworkUrl];
        _moviePlayer=[[MPMoviePlayerController alloc]initWithContentURL:url];
        _moviePlayer.view.frame=self.view.bounds;
        _moviePlayer.view.autoresizingMask=UIViewAutoresizingFlexibleWidth|UIViewAutoresizingFlexibleHeight;
        [self.view addSubview:_moviePlayer.view];
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_moviePlayer;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  获取视频缩略图
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)thumbnailImageRequest{
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//获取13.0s、21.5s的缩略图
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.moviePlayer requestThumbnailImagesAtTimes:@[@13.0,@21.5] timeOption:MPMovieTimeOptionNearestKeyFrame];
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 控制器通知
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  添加通知监控媒体播放控制器状态
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addNotification{
    NSNotificationCenter *notificationCenter=[NSNotificationCenter defaultCenter];
    [notificationCenter addObserver:self selector:@selector(mediaPlayerPlaybackStateChange:) name:MPMoviePlayerPlaybackStateDidChangeNotification object:self.moviePlayer];
    [notificationCenter addObserver:self selector:@selector(mediaPlayerPlaybackFinished:) name:MPMoviePlayerPlaybackDidFinishNotification object:self.moviePlayer];
    [notificationCenter addObserver:self selector:@selector(mediaPlayerThumbnailRequestFinished:) name:MPMoviePlayerThumbnailImageRequestDidFinishNotification object:self.moviePlayer];
    
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  播放状态改变,注意播放完成时的状态是暂停
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)mediaPlayerPlaybackStateChange:(NSNotification *)notification{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">switch </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(self.moviePlayer.playbackState) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">MPMoviePlaybackStatePlaying:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"正在播放..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">MPMoviePlaybackStatePaused:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"暂停播放."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">MPMoviePlaybackStateStopped:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"停止播放."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">default</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"播放状态:%li"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,self.moviePlayer.playbackState);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    }
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  播放完成
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)mediaPlayerPlaybackFinished:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"播放完成.%li"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,self.moviePlayer.playbackState);
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  缩略图请求完成,此方法每次截图成功都会调用一次
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)mediaPlayerThumbnailRequestFinished:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"视频截图完成."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
    UIImage *image=notification.userInfo[MPMoviePlayerThumbnailImageKey];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//保存图片到相册(首次调用会请求用户获得访问相册权限)
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">UIImageWriteToSavedPhotosAlbum(image, nil, nil, nil);
}

@end</span>

截图效果:

MPMoviePlayerController_Thumbnail1     MPMoviePlayerController_Thumbnail2

扩展--使用AVFoundation生成缩略图

通过前面的方法大家应该已经看到,使用MPMoviePlayerController来生成缩略图足够简单,但是如果仅仅是是为了生成缩略图而不进行视频播放的话,此刻使用MPMoviePlayerController就有点大材小用了。其实使用AVFundation框架中的AVAssetImageGenerator就可以获取视频缩略图。使用AVAssetImageGenerator获取缩略图大致分为三个步骤:

  1. 创建AVURLAsset对象(此类主要用于获取媒体信息,包括视频、声音等)。 
  2. 根据AVURLAsset创建AVAssetImageGenerator对象。 
  3. 使用AVAssetImageGenerator的copyCGImageAtTime::方法获得指定时间点的截图。
<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  AVAssetImageGenerator
//
//  Created by Kenshin Cui on 14/03/30.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><AVFoundation/AVFoundation.h>

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()

@end

@implementation ViewController

- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//获取第13.0s的缩略图
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self thumbnailImageRequest:13.0];
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 私有方法
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得本地文件路径
 *
 *  @return 文件路径
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(NSURL *)getFileUrl{
    NSString *urlStr=[[NSBundle mainBundle] pathForResource:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"The New Look of OS X Yosemite.mp4" </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ofType:nil];
    NSURL *url=[NSURL fileURLWithPath:urlStr];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">url;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得网络文件路径
 *
 *  @return 文件路径
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(NSURL *)getNetworkUrl{
    NSString *urlStr=@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"http://192.168.1.161/The New Look of OS X Yosemite.mp4"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    urlStr=[urlStr stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
    NSURL *url=[NSURL URLWithString:urlStr];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">url;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  截取指定时间的视频缩略图
 *
 *  @param timeBySecond 时间点
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)thumbnailImageRequest:(CGFloat )timeBySecond{
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//创建URL
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">NSURL *url=[self getNetworkUrl];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//根据url创建AVURLAsset
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVURLAsset *urlAsset=[AVURLAsset assetWithURL:url];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//根据AVURLAsset创建AVAssetImageGenerator
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVAssetImageGenerator *imageGenerator=[AVAssetImageGenerator assetImageGeneratorWithAsset:urlAsset];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/*截图
     * requestTime:缩略图创建时间
     * actualTime:缩略图实际生成的时间
     */
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">NSError *error=nil;
    CMTime time=CMTimeMakeWithSeconds(timeBySecond, 10);</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//CMTime是表示电影时间信息的结构体,第一个参数表示是视频第几秒,第二个参数表示每秒帧数.(如果要活的某一秒的第几帧可以使用CMTimeMake方法)
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">CMTime actualTime;
    CGImageRef cgImage= [imageGenerator copyCGImageAtTime:time actualTime:&actualTime error:&error];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(error){
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"截取视频缩略图时发生错误,错误信息:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    }
    CMTimeShow(actualTime);
    UIImage *image=[UIImage imageWithCGImage:cgImage];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//转化为UIImage
    //保存到相册
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">UIImageWriteToSavedPhotosAlbum(image,nil, nil, nil);
    CGImageRelease(cgImage);
}

@end</span>

生成的缩略图效果:

AVAssetImageGenerator_Thumbnail

MPMoviePlayerViewController

其实MPMoviePlayerController如果不作为嵌入视频来播放(例如在新闻中嵌入一个视频),通常在播放时都是占满一个屏幕的,特别是在iPhone、iTouch上。因此从iOS3.2以后苹果也在思考既然MPMoviePlayerController在使用时通常都是将其视图view添加到另外一个视图控制器中作为子视图,那么何不直接创建一个控制器视图内部创建一个MPMoviePlayerController属性并且默认全屏播放,开发者在开发的时候直接使用这个视图控制器。这个内部有一个MPMoviePlayerController的视图控制器就是MPMoviePlayerViewController,它继承于UIViewController。MPMoviePlayerViewController内部多了一个moviePlayer属性和一个带有url的初始化方法,同时它内部实现了一些作为模态视图展示所特有的功能,例如默认是全屏模式展示、弹出后自动播放、作为模态窗口展示时如果点击“Done”按钮会自动退出模态窗口等。在下面的示例中就不直接将播放器放到主视图控制器,而是放到一个模态视图控制器中,简单演示MPMoviePlayerViewController的使用。

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  MPMoviePlayerViewController
//
//  Created by Kenshin Cui on 14/03/30.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//  MPMoviePlayerViewController使用

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><MediaPlayer/MediaPlayer.h>

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放器视图控制器
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(nonatomic,strong) MPMoviePlayerViewController *moviePlayerViewController;

@end

@implementation ViewController

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 控制器视图方法
- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];

}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)dealloc{
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//移除所有通知监控
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[[NSNotificationCenter defaultCenter] removeObserver:self];
}


</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 私有方法
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得本地文件路径
 *
 *  @return 文件路径
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(NSURL *)getFileUrl{
    NSString *urlStr=[[NSBundle mainBundle] pathForResource:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"The New Look of OS X Yosemite.mp4" </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ofType:nil];
    NSURL *url=[NSURL fileURLWithPath:urlStr];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">url;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得网络文件路径
 *
 *  @return 文件路径
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(NSURL *)getNetworkUrl{
    NSString *urlStr=@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"http://192.168.1.161/The New Look of OS X Yosemite.mp4"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    urlStr=[urlStr stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
    NSURL *url=[NSURL URLWithString:urlStr];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">url;
}

-(MPMoviePlayerViewController *)moviePlayerViewController{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_moviePlayerViewController) {
        NSURL *url=[self getNetworkUrl];
        _moviePlayerViewController=[[MPMoviePlayerViewController alloc]initWithContentURL:url];
        [self addNotification];
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_moviePlayerViewController;
}
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - UI事件
- (IBAction)playClick:(UIButton *)sender {
    self.moviePlayerViewController=nil;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//保证每次点击都重新创建视频播放控制器视图,避免再次点击时由于不播放的问题
//    [self presentViewController:self.moviePlayerViewController animated:YES completion:nil];
    //注意,在MPMoviePlayerViewController.h中对UIViewController扩展两个用于模态展示和关闭MPMoviePlayerViewController的方法,增加了一种下拉展示动画效果
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self presentMoviePlayerViewControllerAnimated:self.moviePlayerViewController];
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 控制器通知
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  添加通知监控媒体播放控制器状态
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addNotification{
    NSNotificationCenter *notificationCenter=[NSNotificationCenter defaultCenter];
    [notificationCenter addObserver:self selector:@selector(mediaPlayerPlaybackStateChange:) name:MPMoviePlayerPlaybackStateDidChangeNotification object:self.moviePlayerViewController.moviePlayer];
    [notificationCenter addObserver:self selector:@selector(mediaPlayerPlaybackFinished:) name:MPMoviePlayerPlaybackDidFinishNotification object:self.moviePlayerViewController.moviePlayer];
    
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  播放状态改变,注意播放完成时的状态是暂停
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)mediaPlayerPlaybackStateChange:(NSNotification *)notification{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">switch </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(self.moviePlayerViewController.moviePlayer.playbackState) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">MPMoviePlaybackStatePlaying:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"正在播放..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">MPMoviePlaybackStatePaused:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"暂停播放."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">MPMoviePlaybackStateStopped:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"停止播放."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">default</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">:
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"播放状态:%li"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,self.moviePlayerViewController.moviePlayer.playbackState);
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    }
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  播放完成
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)mediaPlayerPlaybackFinished:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"播放完成.%li"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,self.moviePlayerViewController.moviePlayer.playbackState);
}

@end</span>

运行效果:

MPMoviePlayerViewController

这里需要强调一下,由于MPMoviePlayerViewController的初始化方法做了大量工作(例如设置URL、自动播放、添加点击Done完成的监控等),所以当再次点击播放弹出新的模态窗口的时如果不销毁之前的MPMoviePlayerViewController,那么新的对象就无法完成初始化,这样也就不能再次进行播放。

AVPlayer

MPMoviePlayerController足够强大,几乎不用写几行代码就能完成一个播放器,但是正是由于它的高度封装使得要自定义这个播放器变得很复杂,甚至是不可能完成。例如有些时候需要自定义播放器的样式,那么如果要使用MPMoviePlayerController就不合适了,如果要对视频有自由的控制则可以使用AVPlayer。AVPlayer存在于AVFoundation中,它更加接近于底层,所以灵活性也更强:

AVFoundation_Framework

AVPlayer本身并不能显示视频,而且它也不像MPMoviePlayerController有一个view属性。如果AVPlayer要显示必须创建一个播放器层AVPlayerLayer用于展示,播放器层继承于CALayer,有了AVPlayerLayer之添加到控制器视图的layer中即可。要使用AVPlayer首先了解一下几个常用的类:

AVAsset:主要用于获取多媒体信息,是一个抽象类,不能直接使用。

AVURLAsset:AVAsset的子类,可以根据一个URL路径创建一个包含媒体信息的AVURLAsset对象。

AVPlayerItem:一个媒体资源管理对象,管理者视频的一些基本信息和状态,一个AVPlayerItem对应着一个视频资源。

下面简单通过一个播放器来演示AVPlayer的使用,播放器的效果如下:

AVPlayer_Thumbnail

在这个自定义的播放器中实现了视频播放、暂停、进度展示和视频列表功能,下面将对这些功能一一介绍。

首先说一下视频的播放、暂停功能,这也是最基本的功能,AVPlayer对应着两个方法play、pause来实现。但是关键问题是如何判断当前视频是否在播放,在前面的内容中无论是音频播放器还是视频播放器都有对应的状态来判断,但是AVPlayer却没有这样的状态属性,通常情况下可以通过判断播放器的播放速度来获得播放状态。如果rate为0说明是停止状态,1是则是正常播放状态。

其次要展示播放进度就没有其他播放器那么简单了。在前面的播放器中通常是使用通知来获得播放器的状态,媒体加载状态等,但是无论是AVPlayer还是AVPlayerItem(AVPlayer有一个属性currentItem是AVPlayerItem类型,表示当前播放的视频对象)都无法获得这些信息。当然AVPlayerItem是有通知的,但是对于获得播放状态和加载状态有用的通知只有一个:播放完成通知AVPlayerItemDidPlayToEndTimeNotification。在播放视频时,特别是播放网络视频往往需要知道视频加载情况、缓冲情况、播放情况,这些信息可以通过KVO监控AVPlayerItem的status、loadedTimeRanges属性来获得。当AVPlayerItem的status属性为AVPlayerStatusReadyToPlay是说明正在播放,只有处于这个状态时才能获得视频时长等信息;当loadedTimeRanges的改变时(每缓冲一部分数据就会更新此属性)可以获得本次缓冲加载的视频范围(包含起始时间、本次加载时长),这样一来就可以实时获得缓冲情况。然后就是依靠AVPlayer的- (id)addPeriodicTimeObserverForInterval:(CMTime)interval queue:(dispatch_queue_t)queue usingBlock:(void (^)(CMTime time))block方法获得播放进度,这个方法会在设定的时间间隔内定时更新播放进度,通过time参数通知客户端。相信有了这些视频信息播放进度就不成问题了,事实上通过这些信息就算是平时看到的其他播放器的缓冲进度显示以及拖动播放的功能也可以顺利的实现。

最后就是视频切换的功能,在前面介绍的所有播放器中每个播放器对象一次只能播放一个视频,如果要切换视频只能重新创建一个对象,但是AVPlayer却提供了- (void)replaceCurrentItemWithPlayerItem:(AVPlayerItem *)item方法用于在不同的视频之间切换(事实上在AVFoundation内部还有一个AVQueuePlayer专门处理播放列表切换,有兴趣的朋友可以自行研究,这里不再赘述)。

下面附上代码:

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  AVPlayer
//
//  Created by Kenshin Cui on 14/03/30.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><AVFoundation/AVFoundation.h>

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()

@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(nonatomic,strong) AVPlayer *player;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放器对象

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIView *container; </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放器容器
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *playOrPause; </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放/暂停按钮
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIProgressView *progress;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放进度

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@end

@implementation ViewController

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 控制器视图方法
- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];
    [self setupUI];
    [self.player play];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)dealloc{
    [self removeObserverFromPlayerItem:self.player.currentItem];
    [self removeNotification];
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 私有方法
-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setupUI{
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//创建播放器层
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVPlayerLayer *playerLayer=[AVPlayerLayer playerLayerWithPlayer:self.player];
    playerLayer.frame=self.container.frame;
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//playerLayer.videoGravity=AVLayerVideoGravityResizeAspect;//视频填充模式
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.container.layer addSublayer:playerLayer];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  截取指定时间的视频缩略图
 *
 *  @param timeBySecond 时间点
 */

/**
 *  初始化播放器
 *
 *  @return 播放器对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(AVPlayer *)player{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_player) {
        AVPlayerItem *playerItem=[self getPlayItem:0];
        _player=[AVPlayer playerWithPlayerItem:playerItem];
        [self addProgressObserver];
        [self addObserverToPlayerItem:playerItem];
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_player;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  根据视频索引取得AVPlayerItem对象
 *
 *  @param videoIndex 视频顺序索引
 *
 *  @return AVPlayerItem对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(AVPlayerItem *)getPlayItem:(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">int</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)videoIndex{
    NSString *urlStr=[NSString stringWithFormat:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"http://192.168.1.161/%i.mp4"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,videoIndex];
    urlStr =[urlStr stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
    NSURL *url=[NSURL URLWithString:urlStr];
    AVPlayerItem *playerItem=[AVPlayerItem playerItemWithURL:url];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">playerItem;
}
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 通知
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  添加播放器通知
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addNotification{
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//给AVPlayerItem添加播放完成通知
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(playbackFinished:) name:AVPlayerItemDidPlayToEndTimeNotification object:self.player.currentItem];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)removeNotification{
    [[NSNotificationCenter defaultCenter] removeObserver:self];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  播放完成通知
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)playbackFinished:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"视频播放完成."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 监控
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  给播放器添加进度更新
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addProgressObserver{
    AVPlayerItem *playerItem=self.player.currentItem;
    UIProgressView *progress=self.progress;
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//这里设置每秒执行一次
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.player addPeriodicTimeObserverForInterval:CMTimeMake(1.0, 1.0) queue:dispatch_get_main_queue() usingBlock:^(CMTime time) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">float </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">current=CMTimeGetSeconds(time);
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">float </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">total=CMTimeGetSeconds([playerItem duration]);
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"当前已经播放%.2fs."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,current);
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(current) {
            [progress setProgress:(current/total) animated:YES];
        }
    }];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  给AVPlayerItem添加监控
 *
 *  @param playerItem AVPlayerItem对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addObserverToPlayerItem:(AVPlayerItem *)playerItem{
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//监控状态属性,注意AVPlayer也有一个status属性,通过监控它的status也可以获得播放状态
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[playerItem addObserver:self forKeyPath:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"status" </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">options:NSKeyValueObservingOptionNew context:nil];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//监控网络加载情况属性
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[playerItem addObserver:self forKeyPath:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"loadedTimeRanges" </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">options:NSKeyValueObservingOptionNew context:nil];
}
-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)removeObserverFromPlayerItem:(AVPlayerItem *)playerItem{
    [playerItem removeObserver:self forKeyPath:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"status"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">];
    [playerItem removeObserver:self forKeyPath:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"loadedTimeRanges"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">];
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  通过KVO监控播放器状态
 *
 *  @param keyPath 监控属性
 *  @param object  监视器
 *  @param change  状态改变
 *  @param context 上下文
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">*)context{
    AVPlayerItem *playerItem=object;
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([keyPath isEqualToString:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"status"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">]) {
        AVPlayerStatus status= [[change objectForKey:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"new"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">] intValue];
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(status==AVPlayerStatusReadyToPlay){
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"正在播放...,视频总长度:%.2f"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,CMTimeGetSeconds(playerItem.duration));
        }
    }</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else if</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([keyPath isEqualToString:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"loadedTimeRanges"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">]){
        NSArray *</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">array</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">=playerItem.loadedTimeRanges;
        CMTimeRange timeRange = [</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">array</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">.firstObject CMTimeRangeValue];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//本次缓冲时间范围
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">float </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">startSeconds = CMTimeGetSeconds(timeRange.start);
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">float </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">durationSeconds = CMTimeGetSeconds(timeRange.duration);
        NSTimeInterval totalBuffer = startSeconds + durationSeconds;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//缓冲总长度
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"共缓冲:%.2f"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,totalBuffer);
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - UI事件
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  点击播放/暂停按钮
 *
 *  @param sender 播放/暂停按钮
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">- (IBAction)playClick:(UIButton *)sender {
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//    AVPlayerItemDidPlayToEndTimeNotification
    //AVPlayerItem *playerItem= self.player.currentItem;
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(self.player.rate==0){ </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//说明时暂停
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[sender setImage:[UIImage imageNamed:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"player_pause"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">] forState:UIControlStateNormal];
        [self.player play];
    }</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else if</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(self.player.rate==1){</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//正在播放
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.player pause];
        [sender setImage:[UIImage imageNamed:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"player_play"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">] forState:UIControlStateNormal];
    }
}


</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  切换选集,这里使用按钮的tag代表视频名称
 *
 *  @param sender 点击按钮对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">- (IBAction)navigationButtonClick:(UIButton *)sender {
    [self removeNotification];
    [self removeObserverFromPlayerItem:self.player.currentItem];
    AVPlayerItem *playerItem=[self getPlayItem:sender.tag];
    [self addObserverToPlayerItem:playerItem];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//切换视频
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.player replaceCurrentItemWithPlayerItem:playerItem];
    [self addNotification];
}

@end</span>

运行效果:

AVPlayer

到目前为止无论是MPMoviePlayerController还是AVPlayer来播放视频都相当强大,但是它也存在着一些不可回避的问题,那就是支持的视频编码格式很有限:H.264、MPEG-4,扩展名(压缩格式):.mp4、.mov、.m4v、.m2v、.3gp、.3g2等。但是无论是MPMoviePlayerController还是AVPlayer它们都支持绝大多数音频编码,所以大家如果纯粹是为了播放音乐的话也可以考虑使用这两个播放器。那么如何支持更多视频编码格式呢?目前来说主要还是依靠第三方框架,在iOS上常用的视频编码、解码框架有:VLCffmpeg, 具体使用方式今天就不再做详细介绍。

摄像头

UIImagePickerController拍照和视频录制

下面看一下在iOS如何拍照和录制视频。在iOS中要拍照和录制视频最简单的方法就是使用UIImagePickerController。UIImagePickerController继承于UINavigationController,前面的文章中主要使用它来选取照片,其实UIImagePickerController的功能不仅如此,它还可以用来拍照和录制视频。首先看一下这个类常用的属性和方法:

属性说明
@property(nonatomic)           UIImagePickerControllerSourceType     sourceType拾取源类型,sourceType是枚举类型:
UIImagePickerControllerSourceTypePhotoLibrary:照片库
,默认值
UIImagePickerControllerSourceTypeCamera:摄像头
UIImagePickerControllerSourceTypeSavedPhotosAlbum:相簿
@property(nonatomic,copy)      NSArray                              *mediaTypes媒体类型,默认情况下此数组包含kUTTypeImage,所以拍照时可以不用设置;但是当要录像的时候必须设置,可以设置为kUTTypeVideo(视频,但不带声音)或者kUTTypeMovie(视频并带有声音)
@property(nonatomic)           NSTimeInterval                        videoMaximumDuration视频最大录制时长,默认为10 s
@property(nonatomic)           UIImagePickerControllerQualityType    videoQuality视频质量,枚举类型:
UIImagePickerControllerQualityTypeHigh:高清质量
UIImagePickerControllerQualityTypeMedium:中等质量,适合WiFi传输
UIImagePickerControllerQualityTypeLow:低质量,适合蜂窝网传输
UIImagePickerControllerQualityType640x480:640*480
UIImagePickerControllerQualityTypeIFrame1280x720:1280*720
UIImagePickerControllerQualityTypeIFrame960x540:960*540
@property(nonatomic)           BOOL                                  showsCameraControls是否显示摄像头控制面板,默认为YES
@property(nonatomic,retain)    UIView                                *cameraOverlayView 摄像头上覆盖的视图,可用通过这个视频来自定义拍照或录像界面
@property(nonatomic)           CGAffineTransform                     cameraViewTransform摄像头形变
@property(nonatomic) UIImagePickerControllerCameraCaptureMode cameraCaptureMode摄像头捕获模式,捕获模式是枚举类型:
UIImagePickerControllerCameraCaptureModePhoto:拍照模式
UIImagePickerControllerCameraCaptureModeVideo:视频录制模式
@property(nonatomic) UIImagePickerControllerCameraDevice      cameraDevice摄像头设备,cameraDevice是枚举类型:
UIImagePickerControllerCameraDeviceRear:前置摄像头
UIImagePickerControllerCameraDeviceFront:后置摄像头
@property(nonatomic) UIImagePickerControllerCameraFlashMode   cameraFlashMode 闪光灯模式,枚举类型:
UIImagePickerControllerCameraFlashModeOff:关闭闪光灯
UIImagePickerControllerCameraFlashModeAuto:闪光灯自动
UIImagePickerControllerCameraFlashModeOn:打开闪光灯
类方法说明
+ (BOOL)isSourceTypeAvailable:(UIImagePickerControllerSourceType)sourceType指定的源类型是否可用,sourceType是枚举类型:
UIImagePickerControllerSourceTypePhotoLibrary:照片库
UIImagePickerControllerSourceTypeCamera:摄像头
UIImagePickerControllerSourceTypeSavedPhotosAlbum:相簿
+ (NSArray *)availableMediaTypesForSourceType:(UIImagePickerControllerSourceType)sourceType指定的源设备上可用的媒体类型,一般就是图片和视频
+ (BOOL)isCameraDeviceAvailable:(UIImagePickerControllerCameraDevice)cameraDevice指定的摄像头是否可用,cameraDevice是枚举类型:
UIImagePickerControllerCameraDeviceRear:前置摄像头
UIImagePickerControllerCameraDeviceFront:后置摄像头
+ (BOOL)isFlashAvailableForCameraDevice:(UIImagePickerControllerCameraDevice)cameraDevice指定摄像头的闪光灯是否可用
+ (NSArray *)availableCaptureModesForCameraDevice:(UIImagePickerControllerCameraDevice)cameraDevice获得指定摄像头上的可用捕获模式,捕获模式是枚举类型:
UIImagePickerControllerCameraCaptureModePhoto:拍照模式
UIImagePickerControllerCameraCaptureModeVideo:视频录制模式
对象方法说明
- (void)takePicture 编程方式拍照
- (BOOL)startVideoCapture 编程方式录制视频
- (void)stopVideoCapture编程方式停止录制视频
代理方法说明
- (void)imagePickerController:(UIImagePickerController *)picker didFinishPickingMediaWithInfo:(NSDictionary *)info媒体拾取完成
- (void)imagePickerControllerDidCancel:(UIImagePickerController *)picker取消拾取
扩展方法(主要用于保存照片、视频到相簿)说明
UIImageWriteToSavedPhotosAlbum(UIImage *image, id completionTarget, SEL completionSelector, void *contextInfo)保存照片到相簿
UIVideoAtPathIsCompatibleWithSavedPhotosAlbum(NSString *videoPath)能否将视频保存到相簿
void UISaveVideoAtPathToSavedPhotosAlbum(NSString *videoPath, id completionTarget, SEL completionSelector, void *contextInfo)保存视频到相簿

要用UIImagePickerController来拍照或者录制视频通常可以分为如下步骤:

  1. 创建UIImagePickerController对象。 
  2. 指定拾取源,平时选择照片时使用的拾取源是照片库或者相簿,此刻需要指定为摄像头类型。 
  3. 指定摄像头,前置摄像头或者后置摄像头。 
  4. 设置媒体类型mediaType,注意如果是录像必须设置,如果是拍照此步骤可以省略,因为mediaType默认包含kUTTypeImage(注意媒体类型定义在MobileCoreServices.framework中) 
  5. 指定捕获模式,拍照或者录制视频。(视频录制时必须先设置媒体类型再设置捕获模式 
  6. ) 
  7. 展示UIImagePickerController(通常以模态窗口形式打开)。 
  8. 拍照和录制视频结束后在代理方法中展示/保存照片或视频。

当然这个过程中有很多细节可以设置,例如是否显示拍照控制面板,拍照后是否允许编辑等等,通过上面的属性/方法列表相信并不难理解。下面就以一个示例展示如何使用UIImagePickerController来拍照和录制视频,下面的程序中只要将_isVideo设置为YES就是视频录制模式,录制完后在主视图控制器中自动播放;如果将_isVideo设置为NO则为拍照模式,拍照完成之后在主视图控制器中显示拍摄的照片:

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  UIImagePickerController
//
//  Created by Kenshin Cui on 14/04/05.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><MobileCoreServices/MobileCoreServices.h>
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><AVFoundation/AVFoundation.h>

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()<UIImagePickerControllerDelegate,UINavigationControllerDelegate>
@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(assign,nonatomic) </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">int </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">isVideo;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//是否录制视频,如果为1表示录制视频,0代表拍照
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) UIImagePickerController *imagePicker;
@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIImageView *photo;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//照片展示视图
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong ,nonatomic) AVPlayer *player;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//播放器,用于录制完视频后播放视频

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@end

@implementation ViewController

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 控制器视图事件
- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//通过这里设置当前程序是拍照还是录制视频
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_isVideo=YES;
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - UI事件
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//点击拍照按钮
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">- (IBAction)takeClick:(UIButton *)sender {
    [self presentViewController:self.imagePicker animated:YES completion:nil];
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - UIImagePickerController代理方法
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//完成
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)imagePickerController:(UIImagePickerController *)picker didFinishPickingMediaWithInfo:(NSDictionary *)info{
    NSString *mediaType=[info objectForKey:UIImagePickerControllerMediaType];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([mediaType isEqualToString:(NSString *)kUTTypeImage]) {</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//如果是拍照
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">UIImage *image;
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//如果允许编辑则获得编辑后的照片,否则获取原始照片
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(self.imagePicker.allowsEditing) {
            image=[info objectForKey:UIImagePickerControllerEditedImage];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//获取编辑后的照片
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">{
            image=[info objectForKey:UIImagePickerControllerOriginalImage];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//获取原始照片
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
        [self.photo setImage:image];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//显示照片
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">UIImageWriteToSavedPhotosAlbum(image, nil, nil, nil);</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//保存到相簿
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else if</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([mediaType isEqualToString:(NSString *)kUTTypeMovie]){</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//如果是录制视频
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"video..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
        NSURL *url=[info objectForKey:UIImagePickerControllerMediaURL];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//视频路径
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">NSString *urlStr=[url path];
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(UIVideoAtPathIsCompatibleWithSavedPhotosAlbum(urlStr)) {
            </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//保存视频到相簿,注意也可以使用ALAssetsLibrary来保存
            </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">UISaveVideoAtPathToSavedPhotosAlbum(urlStr, self, @selector(video:didFinishSavingWithError:contextInfo:), nil);</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//保存视频到相簿
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
        
    }

    [self dismissViewControllerAnimated:YES completion:nil];
}
-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)imagePickerControllerDidCancel:(UIImagePickerController *)picker{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"取消"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 私有方法
-(UIImagePickerController *)imagePicker{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!_imagePicker) {
        _imagePicker=[[UIImagePickerController alloc]init];
        _imagePicker.sourceType=UIImagePickerControllerSourceTypeCamera;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置image picker的来源,这里设置为摄像头
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_imagePicker.cameraDevice=UIImagePickerControllerCameraDeviceRear;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置使用哪个摄像头,这里设置为后置摄像头
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(self.isVideo) {
            _imagePicker.mediaTypes=@[(NSString *)kUTTypeMovie];
            _imagePicker.videoQuality=UIImagePickerControllerQualityTypeIFrame1280x720;
            _imagePicker.cameraCaptureMode=UIImagePickerControllerCameraCaptureModeVideo;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置摄像头模式(拍照,录制视频)
            
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">{
            _imagePicker.cameraCaptureMode=UIImagePickerControllerCameraCaptureModePhoto;
        }
        _imagePicker.allowsEditing=YES;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//允许编辑
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_imagePicker.</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">delegate</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">=self;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置代理,检测操作
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_imagePicker;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//视频保存后的回调
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)video:(NSString *)videoPath didFinishSavingWithError:(NSError *)error contextInfo:(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">*)contextInfo{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(error) {
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"保存视频过程中发生错误,错误信息:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
    }</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">{
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"视频保存成功."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//录制完之后自动播放
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">NSURL *url=[NSURL fileURLWithPath:videoPath];
        _player=[AVPlayer playerWithURL:url];
        AVPlayerLayer *playerLayer=[AVPlayerLayer playerLayerWithPlayer:_player];
        playerLayer.frame=self.photo.frame;
        [self.photo.layer addSublayer:playerLayer];
        [_player play];
        
    }
}
@end</span>

运行效果(视频录制):

UIImagePickerController

AVFoundation拍照和录制视频

不得不说UIImagePickerController确实强大,但是与MPMoviePlayerController类似,由于它的高度封装性,要进行某些自定义工作就比较复杂了。例如要做出一款类似于美颜相机的拍照界面就比较难以实现了,此时就可以考虑使用AVFoundation来实现。AVFoundation中提供了很多现成的播放器和录音机,但是事实上它还有更加底层的内容可以供开发者使用。因为AVFoundation中抽了很多和底层输入、输出设备打交道的类,依靠这些类开发人员面对的不再是封装好的音频播放器AVAudioPlayer、录音机(AVAudioRecorder)、视频(包括音频)播放器AVPlayer,而是输入设备(例如麦克风、摄像头)、输出设备(图片、视频)等。首先了解一下使用AVFoundation做拍照和视频录制开发用到的相关类:

AVCaptureSession:媒体(音、视频)捕获会话,负责把捕获的音视频数据输出到输出设备中。一个AVCaptureSession可以有多个输入输出:

AVCaptureSession_InputAndOutput

AVCaptureDevice:输入设备,包括麦克风、摄像头,通过该对象可以设置物理设备的一些属性(例如相机聚焦、白平衡等)。

AVCaptureDeviceInput:设备输入数据管理对象,可以根据AVCaptureDevice创建对应的AVCaptureDeviceInput对象,该对象将会被添加到AVCaptureSession中管理。

AVCaptureOutput:输出数据管理对象,用于接收各类输出数据,通常使用对应的子类AVCaptureAudioDataOutput、AVCaptureStillImageOutput、AVCaptureVideoDataOutput、AVCaptureFileOutput,该对象将会被添加到AVCaptureSession中管理。注意:前面几个对象的输出数据都是NSData类型,而AVCaptureFileOutput代表数据以文件形式输出,类似的,AVCcaptureFileOutput也不会直接创建使用,通常会使用其子类:AVCaptureAudioFileOutput、AVCaptureMovieFileOutput。当把一个输入或者输出添加到AVCaptureSession之后AVCaptureSession就会在所有相符的输入、输出设备之间建立连接(AVCaptionConnection):

AVCaptureSession_Relation

AVCaptureVideoPreviewLayer:相机拍摄预览图层,是CALayer的子类,使用该对象可以实时查看拍照或视频录制效果,创建该对象需要指定对应的AVCaptureSession对象。

使用AVFoundation拍照和录制视频的一般步骤如下:

  1. 创建AVCaptureSession对象。 
  2. 使用AVCaptureDevice的静态方法获得需要使用的设备,例如拍照和录像就需要获得摄像头设备,录音就要获得麦克风设备。 
  3. 利用输入设备AVCaptureDevice初始化AVCaptureDeviceInput对象。 
  4. 初始化输出数据管理对象,如果要拍照就初始化AVCaptureStillImageOutput对象;如果拍摄视频就初始化AVCaptureMovieFileOutput对象。 
  5. 将数据输入对象AVCaptureDeviceInput、数据输出对象AVCaptureOutput添加到媒体会话管理对象AVCaptureSession中。 
  6. 创建视频预览图层AVCaptureVideoPreviewLayer并指定媒体会话,添加图层到显示容器中,调用AVCaptureSession的startRuning方法开始捕获。 
  7. 将捕获的音频或视频数据输出到指定文件。

拍照

下面看一下如何使用AVFoundation实现一个拍照程序,在这个程序中将实现摄像头预览、切换前后摄像头、闪光灯设置、对焦、拍照保存等功能。应用大致效果如下:

AVFoundation_CameraRunEffect

在程序中定义会话、输入、输出等相关对象。

<span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()
@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) AVCaptureSession *captureSession;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//负责输入和输出设备之间的数据传递
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) AVCaptureDeviceInput *captureDeviceInput;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//负责从AVCaptureDevice获得输入数据
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) AVCaptureStillImageOutput *captureStillImageOutput;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//照片输出流
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) AVCaptureVideoPreviewLayer *captureVideoPreviewLayer;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//相机拍摄预览图层
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIView *viewContainer;
@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *takeButton;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//拍照按钮
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *flashAutoButton;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//自动闪光灯按钮
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *flashOnButton;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//打开闪光灯按钮
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *flashOffButton;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//关闭闪光灯按钮
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIImageView *focusCursor; </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//聚焦光标
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@end</span>

在控制器视图将要展示时创建并初始化会话、摄像头设备、输入、输出、预览图层,并且添加预览图层到视图中,除此之外还做了一些初始化工作,例如添加手势(点击屏幕进行聚焦)、初始化界面等。

<span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewWillAppear:(BOOL)animated{
    [super viewWillAppear:animated];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//初始化会话
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureSession=[[AVCaptureSession alloc]init];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([_captureSession canSetSessionPreset:AVCaptureSessionPreset1280x720]) {</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置分辨率
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureSession.sessionPreset=AVCaptureSessionPreset1280x720;
    }
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//获得输入设备
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureDevice *captureDevice=[self getCameraDeviceWithPosition:AVCaptureDevicePositionBack];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//取得后置摄像头
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!captureDevice) {
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"取得后置摄像头时出现问题."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    }
    
    NSError *error=nil;
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//根据输入设备初始化设备输入对象,用于获得输入数据
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureDeviceInput=[[AVCaptureDeviceInput alloc]initWithDevice:captureDevice error:&error];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(error) {
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"取得设备输入对象时出错,错误原因:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    }
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//初始化设备输出对象,用于获得输出数据
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureStillImageOutput=[[AVCaptureStillImageOutput alloc]init];
    NSDictionary *outputSettings = @{AVVideoCodecKey:AVVideoCodecJPEG};
    [_captureStillImageOutput setOutputSettings:outputSettings];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//输出设置
    
    //将设备输入添加到会话中
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([_captureSession canAddInput:_captureDeviceInput]) {
        [_captureSession addInput:_captureDeviceInput];
    }
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//将设备输出添加到会话中
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([_captureSession canAddOutput:_captureStillImageOutput]) {
        [_captureSession addOutput:_captureStillImageOutput];
    }
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//创建视频预览层,用于实时展示摄像头状态
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureVideoPreviewLayer=[[AVCaptureVideoPreviewLayer alloc]initWithSession:self.captureSession];
    
    CALayer *layer=self.viewContainer.layer;
    layer.masksToBounds=YES;
    
    _captureVideoPreviewLayer.frame=layer.bounds;
    _captureVideoPreviewLayer.videoGravity=AVLayerVideoGravityResizeAspectFill;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//填充模式
    //将视频预览层添加到界面中
    //[layer addSublayer:_captureVideoPreviewLayer];
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[layer insertSublayer:_captureVideoPreviewLayer below:self.focusCursor.layer];
    
    [self addNotificationToCaptureDevice:captureDevice];
    [self addGenstureRecognizer];
    [self setFlashModeButtonStatus];
}</span>

在控制器视图展示和视图离开界面时启动、停止会话。

<span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidAppear:(BOOL)animated{
    [super viewDidAppear:animated];
    [self.captureSession startRunning];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidDisappear:(BOOL)animated{
    [super viewDidDisappear:animated];
    [self.captureSession stopRunning];
}</span>

定义闪光灯开闭及自动模式功能,注意无论是设置闪光灯、白平衡还是其他输入设备属性,在设置之前必须先锁定配置,修改完后解锁。

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  改变设备属性的统一操作方法
 *
 *  @param propertyChange 属性改变操作
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)changeDeviceProperty:(PropertyChangeBlock)propertyChange{
    AVCaptureDevice *captureDevice= [self.captureDeviceInput device];
    NSError *error;
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//注意改变设备属性前一定要首先调用lockForConfiguration:调用完之后使用unlockForConfiguration方法解锁
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice lockForConfiguration:&error]) {
        propertyChange(captureDevice);
        [captureDevice unlockForConfiguration];
    }</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">{
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"设置设备属性过程发生错误,错误信息:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
    }
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置闪光灯模式
 *
 *  @param flashMode 闪光灯模式
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setFlashMode:(AVCaptureFlashMode )flashMode{
    [self changeDeviceProperty:^(AVCaptureDevice *captureDevice) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isFlashModeSupported:flashMode]) {
            [captureDevice setFlashMode:flashMode];
        }
    }];
}</span>

定义切换摄像头功能,切换摄像头的过程就是将原有输入移除,在会话中添加新的输入,但是注意动态修改会话需要首先开启配置,配置成功后提交配置。

<span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark 切换前后摄像头
- (IBAction)toggleButtonClick:(UIButton *)sender {
    AVCaptureDevice *currentDevice=[self.captureDeviceInput device];
    AVCaptureDevicePosition currentPosition=[currentDevice position];
    [self removeNotificationFromCaptureDevice:currentDevice];
    AVCaptureDevice *toChangeDevice;
    AVCaptureDevicePosition toChangePosition=AVCaptureDevicePositionFront;
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(currentPosition==AVCaptureDevicePositionUnspecified||currentPosition==AVCaptureDevicePositionFront) {
        toChangePosition=AVCaptureDevicePositionBack;
    }
    toChangeDevice=[self getCameraDeviceWithPosition:toChangePosition];
    [self addNotificationToCaptureDevice:toChangeDevice];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//获得要调整的设备输入对象
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureDeviceInput *toChangeDeviceInput=[[AVCaptureDeviceInput alloc]initWithDevice:toChangeDevice error:nil];
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//改变会话的配置前一定要先开启配置,配置完成后提交配置改变
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.captureSession beginConfiguration];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//移除原有输入对象
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.captureSession removeInput:self.captureDeviceInput];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//添加新的输入对象
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([self.captureSession canAddInput:toChangeDeviceInput]) {
        [self.captureSession addInput:toChangeDeviceInput];
        self.captureDeviceInput=toChangeDeviceInput;
    }
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//提交会话配置
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.captureSession commitConfiguration];
    
    [self setFlashModeButtonStatus];
}</span>

添加点击手势操作,点按预览视图时进行聚焦、白平衡设置。

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置聚焦点
 *
 *  @param point 聚焦点
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)focusWithMode:(AVCaptureFocusMode)focusMode exposureMode:(AVCaptureExposureMode)exposureMode atPoint:(CGPoint)point{
    [self changeDeviceProperty:^(AVCaptureDevice *captureDevice) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isFocusModeSupported:focusMode]) {
            [captureDevice setFocusMode:AVCaptureFocusModeAutoFocus];
        }
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isFocusPointOfInterestSupported]) {
            [captureDevice setFocusPointOfInterest:point];
        }
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isExposureModeSupported:exposureMode]) {
            [captureDevice setExposureMode:AVCaptureExposureModeAutoExpose];
        }
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isExposurePointOfInterestSupported]) {
            [captureDevice setExposurePointOfInterest:point];
        }
    }];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  添加点按手势,点按时聚焦
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addGenstureRecognizer{
    UITapGestureRecognizer *tapGesture=[[UITapGestureRecognizer alloc]initWithTarget:self action:@selector(tapScreen:)];
    [self.viewContainer addGestureRecognizer:tapGesture];
}
-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)tapScreen:(UITapGestureRecognizer *)tapGesture{
    CGPoint point= [tapGesture locationInView:self.viewContainer];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//将UI坐标转化为摄像头坐标
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">CGPoint cameraPoint= [self.captureVideoPreviewLayer captureDevicePointOfInterestForPoint:point];
    [self setFocusCursorWithPoint:point];
    [self focusWithMode:AVCaptureFocusModeAutoFocus exposureMode:AVCaptureExposureModeAutoExpose atPoint:cameraPoint];
}</span>

定义拍照功能,拍照的过程就是获取连接,从连接中获得捕获的输出数据并做保存操作。

<span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark 拍照
- (IBAction)takeButtonClick:(UIButton *)sender {
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//根据设备输出获得连接
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureConnection *captureConnection=[self.captureStillImageOutput connectionWithMediaType:AVMediaTypeVideo];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//根据连接取得设备输出的数据
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.captureStillImageOutput captureStillImageAsynchronouslyFromConnection:captureConnection completionHandler:^(CMSampleBufferRef imageDataSampleBuffer, NSError *error) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(imageDataSampleBuffer) {
            NSData *imageData=[AVCaptureStillImageOutput jpegStillImageNSDataRepresentation:imageDataSampleBuffer];
            UIImage *image=[UIImage imageWithData:imageData];
            UIImageWriteToSavedPhotosAlbum(image, nil, nil, nil);
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//            ALAssetsLibrary *assetsLibrary=[[ALAssetsLibrary alloc]init];
//            [assetsLibrary writeImageToSavedPhotosAlbum:[image CGImage] orientation:(ALAssetOrientation)[image imageOrientation] completionBlock:nil];
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
        
    }];
}</span>

最后附上完整代码:

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  AVFoundationCamera
//
//  Created by Kenshin Cui on 14/04/05.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><AVFoundation/AVFoundation.h>
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><AssetsLibrary/AssetsLibrary.h>
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">typedef void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(^PropertyChangeBlock)(AVCaptureDevice *captureDevice);

@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()

@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) AVCaptureSession *captureSession;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//负责输入和输出设备之间的数据传递
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) AVCaptureDeviceInput *captureDeviceInput;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//负责从AVCaptureDevice获得输入数据
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) AVCaptureStillImageOutput *captureStillImageOutput;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//照片输出流
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) AVCaptureVideoPreviewLayer *captureVideoPreviewLayer;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//相机拍摄预览图层
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIView *viewContainer;
@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *takeButton;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//拍照按钮
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *flashAutoButton;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//自动闪光灯按钮
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *flashOnButton;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//打开闪光灯按钮
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *flashOffButton;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//关闭闪光灯按钮
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIImageView *focusCursor; </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//聚焦光标



</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@end

@implementation ViewController

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 控制器视图方法
- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];
    
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewWillAppear:(BOOL)animated{
    [super viewWillAppear:animated];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//初始化会话
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureSession=[[AVCaptureSession alloc]init];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([_captureSession canSetSessionPreset:AVCaptureSessionPreset1280x720]) {</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置分辨率
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureSession.sessionPreset=AVCaptureSessionPreset1280x720;
    }
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//获得输入设备
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureDevice *captureDevice=[self getCameraDeviceWithPosition:AVCaptureDevicePositionBack];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//取得后置摄像头
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!captureDevice) {
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"取得后置摄像头时出现问题."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    }
    
    NSError *error=nil;
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//根据输入设备初始化设备输入对象,用于获得输入数据
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureDeviceInput=[[AVCaptureDeviceInput alloc]initWithDevice:captureDevice error:&error];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(error) {
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"取得设备输入对象时出错,错误原因:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    }
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//初始化设备输出对象,用于获得输出数据
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureStillImageOutput=[[AVCaptureStillImageOutput alloc]init];
    NSDictionary *outputSettings = @{AVVideoCodecKey:AVVideoCodecJPEG};
    [_captureStillImageOutput setOutputSettings:outputSettings];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//输出设置
    
    //将设备输入添加到会话中
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([_captureSession canAddInput:_captureDeviceInput]) {
        [_captureSession addInput:_captureDeviceInput];
    }
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//将设备输出添加到会话中
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([_captureSession canAddOutput:_captureStillImageOutput]) {
        [_captureSession addOutput:_captureStillImageOutput];
    }
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//创建视频预览层,用于实时展示摄像头状态
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureVideoPreviewLayer=[[AVCaptureVideoPreviewLayer alloc]initWithSession:self.captureSession];
    
    CALayer *layer=self.viewContainer.layer;
    layer.masksToBounds=YES;
    
    _captureVideoPreviewLayer.frame=layer.bounds;
    _captureVideoPreviewLayer.videoGravity=AVLayerVideoGravityResizeAspectFill;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//填充模式
    //将视频预览层添加到界面中
    //[layer addSublayer:_captureVideoPreviewLayer];
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[layer insertSublayer:_captureVideoPreviewLayer below:self.focusCursor.layer];
    
    [self addNotificationToCaptureDevice:captureDevice];
    [self addGenstureRecognizer];
    [self setFlashModeButtonStatus];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidAppear:(BOOL)animated{
    [super viewDidAppear:animated];
    [self.captureSession startRunning];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidDisappear:(BOOL)animated{
    [super viewDidDisappear:animated];
    [self.captureSession stopRunning];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)dealloc{
    [self removeNotification];
}
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - UI方法
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark 拍照
- (IBAction)takeButtonClick:(UIButton *)sender {
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//根据设备输出获得连接
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureConnection *captureConnection=[self.captureStillImageOutput connectionWithMediaType:AVMediaTypeVideo];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//根据连接取得设备输出的数据
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.captureStillImageOutput captureStillImageAsynchronouslyFromConnection:captureConnection completionHandler:^(CMSampleBufferRef imageDataSampleBuffer, NSError *error) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(imageDataSampleBuffer) {
            NSData *imageData=[AVCaptureStillImageOutput jpegStillImageNSDataRepresentation:imageDataSampleBuffer];
            UIImage *image=[UIImage imageWithData:imageData];
            UIImageWriteToSavedPhotosAlbum(image, nil, nil, nil);
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//            ALAssetsLibrary *assetsLibrary=[[ALAssetsLibrary alloc]init];
//            [assetsLibrary writeImageToSavedPhotosAlbum:[image CGImage] orientation:(ALAssetOrientation)[image imageOrientation] completionBlock:nil];
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
        
    }];
}
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark 切换前后摄像头
- (IBAction)toggleButtonClick:(UIButton *)sender {
    AVCaptureDevice *currentDevice=[self.captureDeviceInput device];
    AVCaptureDevicePosition currentPosition=[currentDevice position];
    [self removeNotificationFromCaptureDevice:currentDevice];
    AVCaptureDevice *toChangeDevice;
    AVCaptureDevicePosition toChangePosition=AVCaptureDevicePositionFront;
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(currentPosition==AVCaptureDevicePositionUnspecified||currentPosition==AVCaptureDevicePositionFront) {
        toChangePosition=AVCaptureDevicePositionBack;
    }
    toChangeDevice=[self getCameraDeviceWithPosition:toChangePosition];
    [self addNotificationToCaptureDevice:toChangeDevice];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//获得要调整的设备输入对象
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureDeviceInput *toChangeDeviceInput=[[AVCaptureDeviceInput alloc]initWithDevice:toChangeDevice error:nil];
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//改变会话的配置前一定要先开启配置,配置完成后提交配置改变
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.captureSession beginConfiguration];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//移除原有输入对象
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.captureSession removeInput:self.captureDeviceInput];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//添加新的输入对象
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([self.captureSession canAddInput:toChangeDeviceInput]) {
        [self.captureSession addInput:toChangeDeviceInput];
        self.captureDeviceInput=toChangeDeviceInput;
    }
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//提交会话配置
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.captureSession commitConfiguration];
    
    [self setFlashModeButtonStatus];
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark 自动闪光灯开启
- (IBAction)flashAutoClick:(UIButton *)sender {
    [self setFlashMode:AVCaptureFlashModeAuto];
    [self setFlashModeButtonStatus];
}
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark 打开闪光灯
- (IBAction)flashOnClick:(UIButton *)sender {
    [self setFlashMode:AVCaptureFlashModeOn];
    [self setFlashModeButtonStatus];
}
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark 关闭闪光灯
- (IBAction)flashOffClick:(UIButton *)sender {
    [self setFlashMode:AVCaptureFlashModeOff];
    [self setFlashModeButtonStatus];
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 通知
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  给输入设备添加通知
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addNotificationToCaptureDevice:(AVCaptureDevice *)captureDevice{
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//注意添加区域改变捕获通知必须首先设置设备允许捕获
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self changeDeviceProperty:^(AVCaptureDevice *captureDevice) {
        captureDevice.subjectAreaChangeMonitoringEnabled=YES;
    }];
    NSNotificationCenter *notificationCenter= [NSNotificationCenter defaultCenter];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//捕获区域发生改变
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[notificationCenter addObserver:self selector:@selector(areaChange:) name:AVCaptureDeviceSubjectAreaDidChangeNotification object:captureDevice];
}
-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)removeNotificationFromCaptureDevice:(AVCaptureDevice *)captureDevice{
    NSNotificationCenter *notificationCenter= [NSNotificationCenter defaultCenter];
    [notificationCenter removeObserver:self name:AVCaptureDeviceSubjectAreaDidChangeNotification object:captureDevice];
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  移除所有通知
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)removeNotification{
    NSNotificationCenter *notificationCenter= [NSNotificationCenter defaultCenter];
    [notificationCenter removeObserver:self];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addNotificationToCaptureSession:(AVCaptureSession *)captureSession{
    NSNotificationCenter *notificationCenter= [NSNotificationCenter defaultCenter];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//会话出错
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[notificationCenter addObserver:self selector:@selector(sessionRuntimeError:) name:AVCaptureSessionRuntimeErrorNotification object:captureSession];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设备连接成功
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)deviceConnected:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"设备已连接..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设备连接断开
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)deviceDisconnected:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"设备已断开."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  捕获区域改变
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)areaChange:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"捕获区域改变..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  会话出错
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)sessionRuntimeError:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"会话发生错误."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 私有方法

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得指定位置的摄像头
 *
 *  @param position 摄像头位置
 *
 *  @return 摄像头设备
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(AVCaptureDevice *)getCameraDeviceWithPosition:(AVCaptureDevicePosition )position{
    NSArray *cameras= [AVCaptureDevice devicesWithMediaType:AVMediaTypeVideo];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">for </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(AVCaptureDevice *camera in cameras) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([camera position]==position) {
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">camera;
        }
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">nil;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  改变设备属性的统一操作方法
 *
 *  @param propertyChange 属性改变操作
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)changeDeviceProperty:(PropertyChangeBlock)propertyChange{
    AVCaptureDevice *captureDevice= [self.captureDeviceInput device];
    NSError *error;
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//注意改变设备属性前一定要首先调用lockForConfiguration:调用完之后使用unlockForConfiguration方法解锁
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice lockForConfiguration:&error]) {
        propertyChange(captureDevice);
        [captureDevice unlockForConfiguration];
    }</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">{
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"设置设备属性过程发生错误,错误信息:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
    }
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置闪光灯模式
 *
 *  @param flashMode 闪光灯模式
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setFlashMode:(AVCaptureFlashMode )flashMode{
    [self changeDeviceProperty:^(AVCaptureDevice *captureDevice) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isFlashModeSupported:flashMode]) {
            [captureDevice setFlashMode:flashMode];
        }
    }];
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置聚焦模式
 *
 *  @param focusMode 聚焦模式
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setFocusMode:(AVCaptureFocusMode )focusMode{
    [self changeDeviceProperty:^(AVCaptureDevice *captureDevice) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isFocusModeSupported:focusMode]) {
            [captureDevice setFocusMode:focusMode];
        }
    }];
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置曝光模式
 *
 *  @param exposureMode 曝光模式
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setExposureMode:(AVCaptureExposureMode)exposureMode{
    [self changeDeviceProperty:^(AVCaptureDevice *captureDevice) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isExposureModeSupported:exposureMode]) {
            [captureDevice setExposureMode:exposureMode];
        }
    }];
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置聚焦点
 *
 *  @param point 聚焦点
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)focusWithMode:(AVCaptureFocusMode)focusMode exposureMode:(AVCaptureExposureMode)exposureMode atPoint:(CGPoint)point{
    [self changeDeviceProperty:^(AVCaptureDevice *captureDevice) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isFocusModeSupported:focusMode]) {
            [captureDevice setFocusMode:AVCaptureFocusModeAutoFocus];
        }
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isFocusPointOfInterestSupported]) {
            [captureDevice setFocusPointOfInterest:point];
        }
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isExposureModeSupported:exposureMode]) {
            [captureDevice setExposureMode:AVCaptureExposureModeAutoExpose];
        }
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isExposurePointOfInterestSupported]) {
            [captureDevice setExposurePointOfInterest:point];
        }
    }];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  添加点按手势,点按时聚焦
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addGenstureRecognizer{
    UITapGestureRecognizer *tapGesture=[[UITapGestureRecognizer alloc]initWithTarget:self action:@selector(tapScreen:)];
    [self.viewContainer addGestureRecognizer:tapGesture];
}
-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)tapScreen:(UITapGestureRecognizer *)tapGesture{
    CGPoint point= [tapGesture locationInView:self.viewContainer];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//将UI坐标转化为摄像头坐标
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">CGPoint cameraPoint= [self.captureVideoPreviewLayer captureDevicePointOfInterestForPoint:point];
    [self setFocusCursorWithPoint:point];
    [self focusWithMode:AVCaptureFocusModeAutoFocus exposureMode:AVCaptureExposureModeAutoExpose atPoint:cameraPoint];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置闪光灯按钮状态
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setFlashModeButtonStatus{
    AVCaptureDevice *captureDevice=[self.captureDeviceInput device];
    AVCaptureFlashMode flashMode=captureDevice.flashMode;
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isFlashAvailable]){
        self.flashAutoButton.hidden=NO;
        self.flashOnButton.hidden=NO;
        self.flashOffButton.hidden=NO;
        self.flashAutoButton.enabled=YES;
        self.flashOnButton.enabled=YES;
        self.flashOffButton.enabled=YES;
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">switch </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(flashMode) {
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureFlashModeAuto:
                self.flashAutoButton.enabled=NO;
                </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureFlashModeOn:
                self.flashOnButton.enabled=NO;
                </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">case </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureFlashModeOff:
                self.flashOffButton.enabled=NO;
                </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">default</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">:
                </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">break</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
        }
    }</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">{
        self.flashAutoButton.hidden=YES;
        self.flashOnButton.hidden=YES;
        self.flashOffButton.hidden=YES;
    }
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置聚焦光标位置
 *
 *  @param point 光标位置
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setFocusCursorWithPoint:(CGPoint)point{
    self.focusCursor.center=point;
    self.focusCursor.transform=CGAffineTransformMakeScale(1.5, 1.5);
    self.focusCursor.alpha=1.0;
    [UIView animateWithDuration:1.0 animations:^{
        self.focusCursor.transform=CGAffineTransformIdentity;
    } completion:^(BOOL finished) {
        self.focusCursor.alpha=0;
        
    }];
}
@end</span>

运行效果:

视频录制

其实有了前面的拍照应用之后要在此基础上做视频录制功能并不复杂,程序只需要做如下修改:

  1. 添加一个音频输入到会话(使用[[AVCaptureDevice devicesWithMediaType:AVMediaTypeAudio] firstObject]获得输入设备,然后根据此输入设备创建一个设备输入对象),在拍照程序中已经添加了视频输入所以此时不需要添加视频输入。 
  2. 创建一个音乐播放文件输出对象AVCaptureMovieFileOutput取代原来的照片输出对象。 
  3. 将捕获到的视频数据写入到临时文件并在停止录制之后保存到相簿(通过AVCaptureMovieFileOutput的代理方法)。

相比拍照程序,程序的修改主要就是以上三点。当然为了让程序更加完善在下面的视频录制程序中加入了屏幕旋转视频、自动布局和后台保存任务等细节。下面是修改后的程序:

<span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//
//  ViewController.m
//  AVFoundationCamera
//
//  Created by Kenshin Cui on 14/04/05.
//  Copyright (c) 2014年 cmjstudio. All rights reserved.
//  视频录制

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"ViewController.h"
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><AVFoundation/AVFoundation.h>
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#import </span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;"><AssetsLibrary/AssetsLibrary.h>
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">typedef void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(^PropertyChangeBlock)(AVCaptureDevice *captureDevice);

@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">interface </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">ViewController ()<AVCaptureFileOutputRecordingDelegate></span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//视频文件输出代理

</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) AVCaptureSession *captureSession;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//负责输入和输出设备之间的数据传递
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) AVCaptureDeviceInput *captureDeviceInput;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//负责从AVCaptureDevice获得输入数据
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) AVCaptureMovieFileOutput *captureMovieFileOutput;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//视频输出流
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(strong,nonatomic) AVCaptureVideoPreviewLayer *captureVideoPreviewLayer;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//相机拍摄预览图层
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(assign,nonatomic) BOOL enableRotation;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//是否允许旋转(注意在视频录制过程中禁止屏幕旋转)
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(assign,nonatomic) CGRect *lastBounds;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//旋转的前大小
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(assign,nonatomic) UIBackgroundTaskIdentifier backgroundTaskIdentifier;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//后台任务标识
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIView *viewContainer;
@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIButton *takeButton;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//拍照按钮
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">property </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(weak, nonatomic) IBOutlet UIImageView *focusCursor; </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//聚焦光标


</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">@end

@implementation ViewController

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 控制器视图方法
- (</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidLoad {
    [super viewDidLoad];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewWillAppear:(BOOL)animated{
    [super viewWillAppear:animated];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//初始化会话
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureSession=[[AVCaptureSession alloc]init];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([_captureSession canSetSessionPreset:AVCaptureSessionPreset1280x720]) {</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//设置分辨率
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureSession.sessionPreset=AVCaptureSessionPreset1280x720;
    }
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//获得输入设备
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureDevice *captureDevice=[self getCameraDeviceWithPosition:AVCaptureDevicePositionBack];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//取得后置摄像头
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(!captureDevice) {
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"取得后置摄像头时出现问题."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    }
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//添加一个音频输入设备
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureDevice *audioCaptureDevice=[[AVCaptureDevice devicesWithMediaType:AVMediaTypeAudio] firstObject];
    
    
    NSError *error=nil;
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//根据输入设备初始化设备输入对象,用于获得输入数据
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureDeviceInput=[[AVCaptureDeviceInput alloc]initWithDevice:captureDevice error:&error];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(error) {
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"取得设备输入对象时出错,错误原因:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    }
    AVCaptureDeviceInput *audioCaptureDeviceInput=[[AVCaptureDeviceInput alloc]initWithDevice:audioCaptureDevice error:&error];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(error) {
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"取得设备输入对象时出错,错误原因:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">;
    }
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//初始化设备输出对象,用于获得输出数据
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureMovieFileOutput=[[AVCaptureMovieFileOutput alloc]init];
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//将设备输入添加到会话中
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([_captureSession canAddInput:_captureDeviceInput]) {
        [_captureSession addInput:_captureDeviceInput];
        [_captureSession addInput:audioCaptureDeviceInput];
        AVCaptureConnection *captureConnection=[_captureMovieFileOutput connectionWithMediaType:AVMediaTypeVideo];
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureConnection isVideoStabilizationSupported ]) {
            captureConnection.preferredVideoStabilizationMode=AVCaptureVideoStabilizationModeAuto;
        }
    }
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//将设备输出添加到会话中
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([_captureSession canAddOutput:_captureMovieFileOutput]) {
        [_captureSession addOutput:_captureMovieFileOutput];
    }
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//创建视频预览层,用于实时展示摄像头状态
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">_captureVideoPreviewLayer=[[AVCaptureVideoPreviewLayer alloc]initWithSession:self.captureSession];
    
    CALayer *layer=self.viewContainer.layer;
    layer.masksToBounds=YES;
    
    _captureVideoPreviewLayer.frame=layer.bounds;
    _captureVideoPreviewLayer.videoGravity=AVLayerVideoGravityResizeAspectFill;</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//填充模式
    //将视频预览层添加到界面中
    //[layer addSublayer:_captureVideoPreviewLayer];
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[layer insertSublayer:_captureVideoPreviewLayer below:self.focusCursor.layer];
    
    _enableRotation=YES;
    [self addNotificationToCaptureDevice:captureDevice];
    [self addGenstureRecognizer];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidAppear:(BOOL)animated{
    [super viewDidAppear:animated];
    [self.captureSession startRunning];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)viewDidDisappear:(BOOL)animated{
    [super viewDidDisappear:animated];
    [self.captureSession stopRunning];
}

-(BOOL)shouldAutorotate{
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">self.enableRotation;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">屏幕旋转时调整视频预览图层的方向
//-(void)willTransitionToTraitCollection:(UITraitCollection *)newCollection withTransitionCoordinator:(id<UIViewControllerTransitionCoordinator>)coordinator{
//    [super willTransitionToTraitCollection:newCollection withTransitionCoordinator:coordinator];
    NSLog(@"%i,%i",newCollection.verticalSizeClass,newCollection.horizontalSizeClass);
//    UIInterfaceOrientation orientation = [[UIApplication sharedApplication] statusBarOrientation];
//    NSLog(@"%i",orientation);
//    AVCaptureConnection *captureConnection=[self.captureVideoPreviewLayer connection];
//    captureConnection.videoOrientation=orientation;
//    
//}
//屏幕旋转时调整视频预览图层的方向
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)willRotateToInterfaceOrientation:(UIInterfaceOrientation)toInterfaceOrientation duration:(NSTimeInterval)duration{
    AVCaptureConnection *captureConnection=[self.captureVideoPreviewLayer connection];
    captureConnection.videoOrientation=(AVCaptureVideoOrientation)toInterfaceOrientation;
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//旋转后重新设置大小
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)didRotateFromInterfaceOrientation:(UIInterfaceOrientation)fromInterfaceOrientation{
    _captureVideoPreviewLayer.frame=self.viewContainer.bounds;
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)dealloc{
    [self removeNotification];
}
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - UI方法
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark 视频录制
- (IBAction)takeButtonClick:(UIButton *)sender {
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//根据设备输出获得连接
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureConnection *captureConnection=[self.captureMovieFileOutput connectionWithMediaType:AVMediaTypeVideo];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//根据连接取得设备输出的数据
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(![self.captureMovieFileOutput isRecording]) {
        self.enableRotation=NO;
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//如果支持多任务则则开始多任务
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([[UIDevice currentDevice] isMultitaskingSupported]) {
            self.backgroundTaskIdentifier=[[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:nil];
        }
        </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//预览图层和视频方向保持一致
        </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">captureConnection.videoOrientation=[self.captureVideoPreviewLayer connection].videoOrientation;
        NSString *outputFielPath=[NSTemporaryDirectory() stringByAppendingString:@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"myMovie.mov"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">];
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"save path is :%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,outputFielPath);
        NSURL *fileUrl=[NSURL fileURLWithPath:outputFielPath];
        [self.captureMovieFileOutput startRecordingToOutputFileURL:fileUrl recordingDelegate:self];
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">{
        [self.captureMovieFileOutput stopRecording];</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//停止录制
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">}
}
</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark 切换前后摄像头
- (IBAction)toggleButtonClick:(UIButton *)sender {
    AVCaptureDevice *currentDevice=[self.captureDeviceInput device];
    AVCaptureDevicePosition currentPosition=[currentDevice position];
    [self removeNotificationFromCaptureDevice:currentDevice];
    AVCaptureDevice *toChangeDevice;
    AVCaptureDevicePosition toChangePosition=AVCaptureDevicePositionFront;
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(currentPosition==AVCaptureDevicePositionUnspecified||currentPosition==AVCaptureDevicePositionFront) {
        toChangePosition=AVCaptureDevicePositionBack;
    }
    toChangeDevice=[self getCameraDeviceWithPosition:toChangePosition];
    [self addNotificationToCaptureDevice:toChangeDevice];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//获得要调整的设备输入对象
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">AVCaptureDeviceInput *toChangeDeviceInput=[[AVCaptureDeviceInput alloc]initWithDevice:toChangeDevice error:nil];
    
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//改变会话的配置前一定要先开启配置,配置完成后提交配置改变
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.captureSession beginConfiguration];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//移除原有输入对象
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.captureSession removeInput:self.captureDeviceInput];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//添加新的输入对象
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([self.captureSession canAddInput:toChangeDeviceInput]) {
        [self.captureSession addInput:toChangeDeviceInput];
        self.captureDeviceInput=toChangeDeviceInput;
    }
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//提交会话配置
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self.captureSession commitConfiguration];
    
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 视频输出代理
-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)captureOutput:(AVCaptureFileOutput *)captureOutput didStartRecordingToOutputFileAtURL:(NSURL *)fileURL fromConnections:(NSArray *)connections{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"开始录制..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}
-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)captureOutput:(AVCaptureFileOutput *)captureOutput didFinishRecordingToOutputFileAtURL:(NSURL *)outputFileURL fromConnections:(NSArray *)connections error:(NSError *)error{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"视频录制完成."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//视频录入完成之后在后台将视频存储到相簿
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">self.enableRotation=YES;
    UIBackgroundTaskIdentifier lastBackgroundTaskIdentifier=self.backgroundTaskIdentifier;
    self.backgroundTaskIdentifier=UIBackgroundTaskInvalid;
    ALAssetsLibrary *assetsLibrary=[[ALAssetsLibrary alloc]init];
    [assetsLibrary writeVideoAtPathToSavedPhotosAlbum:outputFileURL completionBlock:^(NSURL *assetURL, NSError *error) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(error) {
            NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"保存视频到相簿过程中发生错误,错误信息:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
        }
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(lastBackgroundTaskIdentifier!=UIBackgroundTaskInvalid) {
            [[UIApplication sharedApplication] endBackgroundTask:lastBackgroundTaskIdentifier];
        }
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"成功保存视频到相簿."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
    }];
    
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 通知
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  给输入设备添加通知
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addNotificationToCaptureDevice:(AVCaptureDevice *)captureDevice{
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//注意添加区域改变捕获通知必须首先设置设备允许捕获
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[self changeDeviceProperty:^(AVCaptureDevice *captureDevice) {
        captureDevice.subjectAreaChangeMonitoringEnabled=YES;
    }];
    NSNotificationCenter *notificationCenter= [NSNotificationCenter defaultCenter];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//捕获区域发生改变
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[notificationCenter addObserver:self selector:@selector(areaChange:) name:AVCaptureDeviceSubjectAreaDidChangeNotification object:captureDevice];
}
-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)removeNotificationFromCaptureDevice:(AVCaptureDevice *)captureDevice{
    NSNotificationCenter *notificationCenter= [NSNotificationCenter defaultCenter];
    [notificationCenter removeObserver:self name:AVCaptureDeviceSubjectAreaDidChangeNotification object:captureDevice];
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  移除所有通知
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)removeNotification{
    NSNotificationCenter *notificationCenter= [NSNotificationCenter defaultCenter];
    [notificationCenter removeObserver:self];
}

-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addNotificationToCaptureSession:(AVCaptureSession *)captureSession{
    NSNotificationCenter *notificationCenter= [NSNotificationCenter defaultCenter];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//会话出错
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">[notificationCenter addObserver:self selector:@selector(sessionRuntimeError:) name:AVCaptureSessionRuntimeErrorNotification object:captureSession];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设备连接成功
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)deviceConnected:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"设备已连接..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设备连接断开
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)deviceDisconnected:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"设备已断开."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  捕获区域改变
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)areaChange:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"捕获区域改变..."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  会话出错
 *
 *  @param notification 通知对象
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)sessionRuntimeError:(NSNotification *)notification{
    NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"会话发生错误."</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">);
}

</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">#pragma </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">mark - 私有方法

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  取得指定位置的摄像头
 *
 *  @param position 摄像头位置
 *
 *  @return 摄像头设备
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(AVCaptureDevice *)getCameraDeviceWithPosition:(AVCaptureDevicePosition )position{
    NSArray *cameras= [AVCaptureDevice devicesWithMediaType:AVMediaTypeVideo];
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">for </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">(AVCaptureDevice *camera in cameras) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([camera position]==position) {
            </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">camera;
        }
    }
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">return </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">nil;
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  改变设备属性的统一操作方法
 *
 *  @param propertyChange 属性改变操作
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)changeDeviceProperty:(PropertyChangeBlock)propertyChange{
    AVCaptureDevice *captureDevice= [self.captureDeviceInput device];
    NSError *error;
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//注意改变设备属性前一定要首先调用lockForConfiguration:调用完之后使用unlockForConfiguration方法解锁
    </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice lockForConfiguration:&error]) {
        propertyChange(captureDevice);
        [captureDevice unlockForConfiguration];
    }</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">else</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">{
        NSLog(@</span><span style="background-color: white; color: rgb(163, 21, 21); background-position: initial initial; background-repeat: initial initial;">"设置设备属性过程发生错误,错误信息:%@"</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">,error.localizedDescription);
    }
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置闪光灯模式
 *
 *  @param flashMode 闪光灯模式
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setFlashMode:(AVCaptureFlashMode )flashMode{
    [self changeDeviceProperty:^(AVCaptureDevice *captureDevice) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isFlashModeSupported:flashMode]) {
            [captureDevice setFlashMode:flashMode];
        }
    }];
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置聚焦模式
 *
 *  @param focusMode 聚焦模式
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setFocusMode:(AVCaptureFocusMode )focusMode{
    [self changeDeviceProperty:^(AVCaptureDevice *captureDevice) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isFocusModeSupported:focusMode]) {
            [captureDevice setFocusMode:focusMode];
        }
    }];
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置曝光模式
 *
 *  @param exposureMode 曝光模式
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setExposureMode:(AVCaptureExposureMode)exposureMode{
    [self changeDeviceProperty:^(AVCaptureDevice *captureDevice) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isExposureModeSupported:exposureMode]) {
            [captureDevice setExposureMode:exposureMode];
        }
    }];
}
</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置聚焦点
 *
 *  @param point 聚焦点
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)focusWithMode:(AVCaptureFocusMode)focusMode exposureMode:(AVCaptureExposureMode)exposureMode atPoint:(CGPoint)point{
    [self changeDeviceProperty:^(AVCaptureDevice *captureDevice) {
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isFocusModeSupported:focusMode]) {
            [captureDevice setFocusMode:AVCaptureFocusModeAutoFocus];
        }
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isFocusPointOfInterestSupported]) {
            [captureDevice setFocusPointOfInterest:point];
        }
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isExposureModeSupported:exposureMode]) {
            [captureDevice setExposureMode:AVCaptureExposureModeAutoExpose];
        }
        </span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">if </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">([captureDevice isExposurePointOfInterestSupported]) {
            [captureDevice setExposurePointOfInterest:point];
        }
    }];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  添加点按手势,点按时聚焦
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)addGenstureRecognizer{
    UITapGestureRecognizer *tapGesture=[[UITapGestureRecognizer alloc]initWithTarget:self action:@selector(tapScreen:)];
    [self.viewContainer addGestureRecognizer:tapGesture];
}
-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)tapScreen:(UITapGestureRecognizer *)tapGesture{
    CGPoint point= [tapGesture locationInView:self.viewContainer];
    </span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">//将UI坐标转化为摄像头坐标
    </span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">CGPoint cameraPoint= [self.captureVideoPreviewLayer captureDevicePointOfInterestForPoint:point];
    [self setFocusCursorWithPoint:point];
    [self focusWithMode:AVCaptureFocusModeAutoFocus exposureMode:AVCaptureExposureModeAutoExpose atPoint:cameraPoint];
}

</span><span style="background-color: white; color: green; background-position: initial initial; background-repeat: initial initial;">/**
 *  设置聚焦光标位置
 *
 *  @param point 光标位置
 */
</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">-(</span><span style="background-color: white; color: blue; background-position: initial initial; background-repeat: initial initial;">void</span><span style="background-color: white; color: black; background-position: initial initial; background-repeat: initial initial;">)setFocusCursorWithPoint:(CGPoint)point{
    self.focusCursor.center=point;
    self.focusCursor.transform=CGAffineTransformMakeScale(1.5, 1.5);
    self.focusCursor.alpha=1.0;
    [UIView animateWithDuration:1.0 animations:^{
        self.focusCursor.transform=CGAffineTransformIdentity;
    } completion:^(BOOL finished) {
        self.focusCursor.alpha=0;
        
    }];
}
@end</span>

运行效果:

总结

前面用了大量的篇幅介绍了iOS中的音、视频播放和录制,有些地方用到了封装好的播放器、录音机直接使用,有些是直接调用系统服务自己组织封装,正如本篇开头所言,iOS对于多媒体支持相当灵活和完善,那么开放过程中如何选择呢,下面就以一个表格简单对比一下各个开发技术的优缺点。

summary

提示:从本文及以后的文章中可能慢慢使用storyboard或xib,原因如下:1.苹果官方目前主推storyboard;2.后面的文章中做屏幕适配牵扯到很多内容都是storyboard中进行(尽管纯代码也可以实现,但是纯代码对autolayout支持不太好)3.通过前面的一系列文章大家对于纯代码编程应该已经有一定的积累了(纯代码确实可以另初学者更加了解程序运行原理)。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值