IOS音视频(一)- AVFoundation核心类

AVFoundation核心类
做音视频开发是个很复杂的工作,需要我们理解很多有关素材的知识:声学和视觉相关的科学理论,数的程序开发技术和有AVFoundation框架而引出的其他框架的知识,比如:Core Media, Core Video, Core Image, Core Audio, Media Player 和 VideoToolbox 等等。
要做IOS音视频相关的开发,肯定要熟悉AVFoundation框架。学习框架最好的方式就是研究苹果官方文档:苹果官方AVFoundation框架介绍
AVFoundation 是 Objective-C 中创建及编辑视听媒体文件的几个框架之一,其提供了检查、创建、编辑或重新编码媒体文件的接口,也使得从设备获取的视频实时数据可操纵。但是,通常情况,简单的播放或者录像,直接使用 AVKit 框架或者 UIImagePickerController 类即可。另外,值得注意的是,在 AVFoundation 框架中使用的基本数据结构,如时间相关的或描述媒体数据的数据结构都声明在 CoreMedia 框架中。
AVFoundation 是 OSX 系统和 iOS 系统中用于处理基于时间的媒体数据的高级 objectivec 框架,其设计过程高度依赖多线程机制,充分利用了多核硬件优势,大量使用 Block 和 GCD 机制。AVFoundation 能与高层级框架无缝衔接,也能提供低层级框架的功能和性能。

  1. AVFoundation框架架构简介
    引用苹果官方文档介绍图如下:


    image.png

iOS系统上的AVFoundation框架架构


image.png

Mac OS X系统上的AVFoundation框架架构

Core Audio
Core Audio是OS X和IOS系统上处理所有音频 事件的框架。Core Audio是有多高框架整合在一起的总称,为音频和MIDI内容的录制,播放和处理提供相应的接口。Core Audio也提供高级的接口,比如通过Audio Queue Services框架所提供的那些接口,主要处理基本的音频播放和录音相关功能。同时还会提供相对低层级的接口,尤其是Audio Units接口,它们提供了针对音频信号进行完全控制的功能,并通过Audio Units让你能够构建一些复杂的音频处理模式,就像通过苹果公司的Logic Pro X和Avid's Pro Tolls工具所实现的功能一样。

Core Video
Core Video是OS X 和IOS系统上针对数字视频所提供的管道模式。Core Video为其相对的Core Media提供图片缓存和缓存支持,提供了一个能够对数字视频逐帧访问的接口。该框架通过像素格式之间的转换并管理同步事项时的复杂的工作得到了有效简化。

Core Media
Core Media 是AV Foundation所用到的低层级媒体管道的一部分。它提供针对音频样本和视频帧处理所需的低层级数据类型和接口。Core Media还提供了AV Foundation用的的基于CMTime数据类型的时间模型。CMTime及其相关数据类型一般在AV Foundation处理基于时间的操作时使用。

Core Animation
Core Animation时OS X和 iOS 提供的合成及动画相关框架。主要功能就是提供苹果平台所具有的美观,流畅的动画效果。提供了一个简单,声明行的编程模式,并已经封装了支持OpenGL 和OpenGL ES 功能的基于Object-C的各种类。使用Core Animation时,对于食品内容的播放和视频捕获这两个动作,AVFoundation 提供了硬件加速机制来对整个流程进行优化。AVFoundation 还可以利用Core Animation让开发者能够在视频编辑和播放过程中添加动画标题和图片效果。
苹果提供了AVFoundation框架,可以用来检测,编辑,创建,重新编码媒体文件,还可以实时获取设备的流媒体数据,实时操作这些被捕获的视频流数据。

苹果推荐我们尽可能的使用高度抽象的接口:
如果只是简单播放视频文件,使用AVKit框架即可。
如果只是想简单录制视频,使用UIKit框架里的UIImagePickerController既可以实现。
AVFoundation 框架包含视频相关的接口以及音频相关的接口,与音频相关的类有 AVAudioPlayer、AVAudioRecorder、AVAudioSession。

AVFoundation框架
AVFoundation 提供的核心功能如下所示
音频播放和记录——AVAudioPlayer 和 AVAudioRecorder
媒体文件检查——AVMetadataItem
视频播放——AVPlayer 和 AVPlayerItem
媒体捕捉——AVCaptureSession
媒体编辑
媒体处理——AVAssetReader、AVAssetWriter

image.png

AVFoundation 框架中最基本的类是 AVAsset ,它是一个或者多个媒体数据的集合,描述的是整个集合的属性,如标题、时长、大小等,并且没有特定的数据格式。集合的每一个媒体数据都是统一的数据类型,称之为 track。简单的情况是一种数据是音频数据,一种是视频数据,而较复杂的情况是一种数据交织着音频和视频数据,并且 AVAsset 是可能有元数据的。
另外,需要明白的是在 AVFoundation 中,初始化了 asset 及 track 后,并不意味着资源已经可用,因为若资源本身并不携带自身信息时,那么系统需要自己计算相关信息,这个过程会阻塞线程,所以应该使用异步方式进行获取资源信息后的操作。
AVFoundation 中可以使用 compositions 将多个媒体数据(video/audio tracks)合成为一个 asset ,这个过程中,可以添加或移除 tracks ,调整它们的顺序,或者设置音频的音量和变化坡度,视频容量等属性。这些媒体数据的集合保存在内存中,直到使用 export session 将它导出到本地文件中。另外,还可以使用 asset writer 创建 asset 。
使用 capture session 协调从设备(如相机、麦克风)输入的数据和输出目标(如视频文件)。可以为 session 设置多个输入和输出,即使它正在工作,还可以通过它停止数据的流动。另外,还可以使用 preview layer 将相机记录的影像实时展示给用户。
在 AVFoundation 中的回调处理并不保证回调任务在某个特定的线程或队列中执行,其遵循两个原则,UI 相关的操作在主线程中执行,其他回调需要为其指定调用的队列。

1.2 AVFoundation 之 Assets


image.png

Assets

参考苹果官方介绍

AVAsset 是AVFoundation框架里的一个核心类,主要提供了一种形式独立的基于时间的视听数据抽象,例如电影文件或视频流。


image

AVAsset类继承关系

AVAsset 包含需要一起呈现或处理的音轨集合,每个音轨都是统一的媒体类型,包括(但不限于)音频、视频、文本、封闭字幕和字幕。asset对象提供关于整个资源的信息,比如它的持续时间或标题,以及表示的提示,比如它的大小。AVAsset 也可以有元数据,由AVMetadataItem的实例表示。

跟踪由AVAssetTrack的实例表示,在一个典型的简单例子中,一个轨道表示音频组件,另一个表示视频组件;在一个复杂的组合中,音频和视频可能会有多个重叠的音轨。


image

AVAssetTrack结构

音轨具有许多属性,例如其类型(视频或音频)、可视和/或可听特征(视情况而定)、元数据和时间轴(以其父资产的形式表示)。磁道也有一组格式说明。数组包含CMFormatDescription对象(参见CMFormatDescriptionRef),每个对象描述曲目引用的媒体样本的格式。包含统一媒体的磁道(例如,所有使用相同设置编码的磁道)将提供一个计数为1的数组。

一个磁道本身可以被分割成段,用AVAssetTrackSegment的实例来表示。段是从源到资产跟踪时间线的时间映射。

CMTime是一个C结构,它将时间表示为一个有理数,具有一个分子(int64_t值)和一个分母(int32_t时间刻度)。从概念上讲,时间刻度指定了分子中每个单位所占的秒数。因此,如果时间刻度是4,每个单元代表四分之一秒;如果时间刻度是10,每个单元表示十分之一秒,以此类推。您经常使用600的时间刻度,因为这是几种常用帧率的倍数:24帧用于电影,30帧用于NTSC(用于北美和日本的电视),25帧用于PAL(用于欧洲的电视)。使用600的时间刻度,您可以精确地表示这些系统中的任意数量的帧。除了简单的时间值之外,CMTime结构还可以表示非数值:+∞、-∞和不定。它还可以指示时间是否在某个点被四舍五入,并保持一个历元数。

您可以使用CMTimeMake或CMTimeMakeWithSeconds等相关函数之一创建时间(该函数允许您使用浮点值创建时间并指定首选时间刻度)。有几个函数用于基于时间的算术和比较时间,如下例所示:

AVFoundation 提供了多种方法来创建 asset ,可以简单的重编码已经存在的 asset ,这个过程可以使用 export session 或者使用 asset reader 和 asset writer 。

若要生成视频的缩略图,可以使用 asset 初始化一个 AVAssetImageGenerator 实例对象,它会使用默认可用的视频 tracks 来生成图片。

创建 AVAsset 或其子类 AVURLAsset 时,需要提供资源的位置,方法如下:

上述方法的第二个参数是创建对象时的选择项,其中可能包含的选择项如下:

是否需要资源的准确时长,及访问资源各个准确的时间点

链接其他资源的约束

添加资源能够访问的 HTTP cookies

是否能够使用蜂窝网络

创建并初始化一个 AVAsset 实例对象后,并不意味着该对象的所有属性都可以获取使用了,因为其中的一些属性需要额外的计算才能够得到,那么当获取这些属性时,可能会阻塞当前线程,所以需要异步获取这些属性。

AVAsset 与 AVAssetTrack 都遵循 AVAsynchronousKeyValueLoading 协议,这个协议中有以下两个方法:

通常,我们使用上述第二个方法异步加载想要的属性,而后在加载完成的回调 block 中使用第一个方法判断属性是否加载成功,然后访问想要的属性,执行自己的操作,如下代码:

1.3 AVFoundation 之 视频播放

image

AVFoundation Player1

image

AVFoundation Player2

1.3.1 AVPlayer

具体可以参考苹果官方介绍的Playback章节

要控制媒体资源的回放,可以使用AVPlayer对象。在回放期间,您可以使用AVPlayerItem实例来管理整个Asset的表示状态,使用AVPlayerItemTrack对象来管理单个曲目的表示状态。要显示视频,可以使用AVPlayerLayer对象。

player是一个控制器对象,您可以使用它来管理资产的回放,例如启动和停止回放,以及查找特定的时间。您使用AVPlayer实例来播放单个资产。您可以使用AVQueuePlayer对象按顺序播放许多项(AVQueuePlayer是AVPlayer的子类)。在OS X上,你可以选择使用AVKit框架的AVPlayerView类来回放视图中的内容。

播放器向您提供有关播放状态的信息,因此,如果需要,您可以将用户界面与播放器的状态同步。通常将播放器的输出定向到特定的核心动画层(AVPlayerLayer或AVSynchronizedLayer的实例)。有关层的更多信息,请参见Core Animation Programming Guide。

尽管最终你想要播放资产,但你不能直接提供资产给AVPlayer 对象。而是提供AVPlayerItem的一个实例。播放器项管理与其相关联的资产的表示状态。播放器项包含播放器项跟踪—avplayeritemtrack的实例—对应于资产中的跟踪。各对象之间的关系如图所示:


image

AVAsset对象关系

这种抽象意味着您可以同时使用不同的玩家来玩给定的资产,但是每个玩家呈现的方式不同。图2-2显示了一种可能性,即两个不同的玩家使用不同的设置来玩相同的资产。例如,使用项曲目,您可以在回放期间禁用特定的曲目(例如,您可能不想播放声音组件)。


image

图2-2

您可以使用现有资产初始化播放器项,也可以直接从URL初始化播放器项,以便在特定位置播放资源(AVPlayerItem将随后为资源创建和配置资产)。不过,与AVAsset一样,简单地初始化播放器项并不一定意味着它可以立即播放。您可以观察(使用键值观察)一个项目的状态属性来确定它是否准备好了,以及何时准备好了。

使用一个 AVPlayer 类实例可以管理一个 asset 资源,但是它的属性 currentItem 才是 asset 的实际管理者。currentItem 是 AVPlayerItem 类的实例,而它的属性 tracks 包含着的 实例对应着 asset 中的各个 track 。

那么,为了控制 asset 的播放,可以使用 AVPlayer 类,在播放的过程中,可以使用 AVPlayerItem 实例管理整个 asset 的状态,使用 对象管理 asset 中每个 track 的状态。另外,还可以使用 类来显示播放的内容。

所以,在创建 AVPlayer 实例对象时,除了可以直接传递资源文件的路径进行创建外,还可以传递 AVPlayerItem 的实例对象,如下方法:

创建后,并不是可以直接使用,还要对它的状态进行检查,只有 status 的值为 时,才能进行播放,所以这里需要使用 KVO 模式对该状态进行监控,以决定何时可以进行播放。

若要管理多个资源的播放,则应使用 AVPlayer 的子类 ,这个子类拥有的多个 AVPlayerItem 同各个资源相对应。

1.3.2 播放不同类型的资源

对于播放不同类型的资源,需要进行的准备工作有所不同,这主要取决于资源的来源。资源数据可能来自本地设备上文件的读取,也可能来自网络上数据流。

对于本地文件,可以使用文件地址创建 对象,而后使用该对象创建 对象,最后将这个 item 对象与 AVPlayer 对象相关联。之后,便是等待 status 的状态变为 ,便可以进行播放了。

对于网络数据的播放,不能使用地址创建 对象了,而是直接创建 对象,将其同 对象相关联,当 status 状态变为 后, 和 对象将由 item 对象创建。

1.3.3 播放控制

通过调用 player 的 play 、pause 、setRate: 方法,可以控制 item 的播放,这些方法都会改变 player 的属性 rate 的值,该值为 1 表示 item 按正常速率播放,为 0 表示 item 暂停播放,0~1 表示低速播放,大于 1 表示高速播放,小于 0 表示从后向前播放。

item 的属性 timeControlStatus 的值表示当前 item 的状态,有下面 3 个值:

暂停

播放

等待按指定速率播放状态,该状态是当 rate 的值设置为非 0 值时,而 item 因某些原因还无法播放的情况,而无法播放的原因,可依通过 item 的 reasonForWaitingToPlay 属性值查看。

item 的属性 actionAtItemEnd 的值表示当前 item 播放结束后的动作,有下面 3 个值:

只适用于 AVQueuePlayer 类,表示播放队列中的下一个 item

表示暂停

表示无操作,当前 item 的 currentTime 属性值仍然按 rate 的值改变

item 的 currentTime 属性值表示当前 item 的播放时间,可以调用下面的方法指定 item 从何处进行播放。

使用 管理多个 item 的播放,仍然可以通过调用 play 开始依次播放 item,调用 advanceToNextItem 方法播放下一个 item ,还可以通过下面的方法添加或移除 item 。

可以使用下面的方法监听播放时间的变化,需要强引用这两个方法返回的监听者。

用上面的方法每注册一个监听者,就需要对应的使用下面的方法进行注销,并且在注销之前,要确保没有 block 被执行。

当 item 播放结束后,再次调用 player 的方法 play 不会使 item 重新播放,要实现重播,可以注册一个 通知,当接收到这个通知时,可以调 seekToTime: 方法,传入 kCMTimeZero 参数,将 player 的播放时间重置。

1.3.4 自定义播放--音频

要在媒体资源播放的过程中实现音频的自定义播放,需要用 对不同的音频进行编辑。这个类的实例对象的属性 inputParameters 是音量描述对象的集合,每个对象都是对一个 audio track 的音量变化的描述,如下:

1.3.4.1 AVAudioMix

该类中有一个属性 inputParameters ,它是 实例对象的集合,每个实例都是对音频播放方式的描述。可见, 并不直接改变音频播放的方式,其只是存储了音频播放的方式。

1.3.4.2 AVMutableAudioMix

是 的子类,它的方法 audioMix 返回一个 inputParameters 属性为空的实例。

1.3.4.3 AVAudioMixInputParameters

这个类是音量变化的描述类,它同一个音频的 track 相关联,并设置音量随时间变化的算法,其获取音量变化的方法如下:

1.3.4.4 AVMutableAudioMixInputParameters

AVMutableAudioMixInputParameters 是 AVAudioMixInputParameters 的子类,它提供了直接设置某个时刻或时间段的音量的方法。

1.3.5 自定义播放--视频

同音频的自定义播放一样,要实现视频的自定义播放,仅仅将视频资源集合到一起是不够的,需要使用 类来定义不同的视频资源在不同的时间范围内的播放方式。

1.3.5.1 AVVideoComposition

AVVideoComposition 是 AVMutableVideoComposition 的父类,它的主要属性和方法如下:

1.3.5.2 AVMutableVideoComposition

AVMutableVideoComposition 是 AVVideoComposition 的可变子类,它继承父类的属性可以改变,并且新增了下面的创建方法。

1.3.5.3 AVVideoCompositionInstruction

在上述的两个类中,真正包含有视频播放方式信息的是 instructions 属性,这个集合中的对象都遵循 协议,若不使用自定义的类,那么可以使用 AVFoundation 框架中的 类。

该类的相关属性如下:

1.3.5.4 AVMutableVideoCompositionInstruction

AVMutableVideoCompositionInstruction 是 AVVideoCompositionInstruction 的子类,其继承的父类的属性可进行修改,并且提供了创建属性值为 nil 或无效的实例的方法。

1.3.5.5 AVVideoCompositionLayerInstruction

是对给定的视频资源的不同播放方式进行描述的类,通过下面的方法,可以获取仿射变化、透明度变化、裁剪区域变化的梯度信息。

1.3.5.6 AVMutableVideoCompositionLayerInstruction

AVMutableVideoCompositionLayerInstruction 是 AVVideoCompositionLayerInstruction 的子类,它可以改变 composition 中的 track 资源播放时的仿射变化、裁剪区域、透明度等信息。

相比于父类,该子类还提供了创建实例的方法:

该类的属性表示 instruction 所作用的 track 的 ID:

设置了 trackID 后,通过下面的方法,进行剃度信息的设置:

1.3.5.7 AVVideoCompositionCoreAnimationTool

在自定义视频播放时,可能需要添加水印、标题或者其他的动画效果,需要使用该类。该类通常用来协调离线视频中图层与动画图层的组合(如使用 和 、 类导出视频文件或读取视频文件时),而若是在线实时的视频播放,应使用 类来同步视频的播放与动画的效果。

在使用该类时,注意动画在整个视频的时间线上均可以被修改,所以,动画的开始时间应该设置为 ,这个值其实比 0 大,属性值 removedOnCompletion 应该置为 NO,以防当动画执行结束后被移除,并且不应使用与任何的 UIView 相关联的图层。

作为视频组合的后期处理工具类,主要方法如下:

1.3.5.8 AVVideoCompositionValidationHandling

当我们经过编辑后得到一个视频资源 asset ,并且为该资源设置了自定义播放信息 video composition ,需要验证对于这个 asset 而言,video composition 是否有效,可以调用 的校验方法。

设置的代理对象要遵循协议 ,该对象在实现下面的协议方法时,若修改了传递的 composition 参数,上面的校验方法则会抛出异常。

该协议提供了以下回调方法,所有方法的返回值用来确定是否继续进行校验以获取更多的错误。

1.3.5.9 AVVideoCompositionValidationHandling

1.4 AVFoundation 之 视音频编辑

image

AVFoundation 编辑视频1


image

AVFoundation 编辑视2

详情可以参考苹果官方文档:Editing章节

AVFoundation使用组合从现有的媒体片段(通常是一个或多个视频和音频轨道)创建新资产。您可以使用可变组合来添加和删除轨迹,并调整它们的时间顺序。你也可以设置音轨的相对音量和倾斜;设置视频轨迹的不透明度和不透明度坡道。合成是存储在内存中的媒体片段的集合。当您使用导出会话导出一个组合时,它会折叠成一个文件。您还可以使用资产写入器从媒体(例如示例缓冲区或静态图像)创建资产。

AVFoundation 框架中提供了丰富的接口用于视听资源的编辑,其中的关键是 composition ,它将不同的 asset 相结合并形成一个新的 asset 。使用 AVMutableComposition 类可以增删 asset 来将指定的 asset 集合到一起。除此之外,若想将集合到一起的视听资源以自定义的方式进行播放,需要使用 和 类对其中的资源进行协调管理。最终要使用 类将编辑的内容保存到文件中。

AVFoundation框架提供了一组功能丰富的类,以方便编辑视听资产。AVFoundation编辑API的核心是复合。组合就是来自一个或多个不同媒体资产的音轨集合。AVMutableComposition类提供了一个接口,用于插入和删除轨迹,以及管理它们的时间顺序。图3-1显示了如何将现有资产组合拼凑成新资产。如果您想做的只是将多个资产按顺序合并到一个文件中,那么这就是您所需要的全部细节。如果你想在你的作曲中对音轨进行任何自定义音频或视频处理,你需要分别合并一个音频混合或一个视频合成。


image

将现有资产组合拼凑成新资产

使用AVMutableAudioMix类,您可以在组合中的音频轨道上执行自定义音频处理,如图3-2所示。目前,您可以为音轨指定最大音量或设置音量斜坡。


image

音频混合

您可以使用AVMutableVideoComposition类来直接编辑合成中的视频轨迹,如图3-3所示。对于单个视频合成,您可以为输出视频指定所需的渲染大小和比例以及帧持续时间。通过视频合成的指令(由AVMutableVideoCompositionInstruction类表示),您可以修改视频的背景颜色并应用层指令。这些层指令(由AVMutableVideoCompositionLayerInstruction类表示)可用于对组合中的视频轨道应用转换、转换坡道、不透明度和不透明度坡道。video composition类还允许您使用animationTool属性将核心动画框架的效果引入到视频中。

image

视频合成

要将组合与音频和视频组合组合在一起,可以使用AVAssetExportSession对象,如图3-4所示。使用组合初始化导出会话,然后分别将音频混合和视频组合分配给audioMix和videoComposition属性。


image

音视频组合

1.4.1 AVAssetExportSession

使用 类对视频进行裁剪及转码,即将一个 类实例修改后保存为另一个 类实例,最后保存到文件中。

在修改资源之前,为避免不兼容带来的错误,可以先调用下面的方法,检查预设置是否合理。

除了设置文件类型外,还可以设置文件的大小、时长、范围等属性,一切准备就绪后,调用方法:

进行文件的导出,导出结束后,会调用 handler 回调,在回调中应该检查 AVAssetExportSession 的 status 属性查看导出是否成功,若指定的文件保存地址在沙盒外,或在导出的过程中有电话打入都会导致文件保存失败,如下:

1.4.2 AVComposition

同 拥有多个 一样,作为子类的 也拥有多个 ,而 是 的子类。所以, 实例对象是多个 track 的集合,真正描述媒体属性的是 实例对象。而 又是媒体数据片段的集合,这些数据片段由 类进行描述。

该类的相关属性和方法如下:

值得注意的是 类中并没有提供初始化方法,一般我们使用它的子类 ,进行各种操作后,再生成 实例以供查询,如下:

1.4.3 AVMutableComposition

是 的子类,其包含的 tracks 则是 的子类 。

中提供了两个类方法用来获取一个空的 实例对象。

对整个 composition 中的 tracks 的修改方法如下:

从 composition 中获取 track 或向其中添加/移除 track 方法如下:

中也提供了过滤 的接口:

1.4.4 AVCompositionTrack

类同其父类 AVAssetTrack 一样是媒体资源的管理者,它实际是媒体资源数据的集合,它的属性 segments 是 类的实例对象集合,每个对象描述一个媒体数据片段。类 并不常用,通常使用的是它的子类 。

1.4.5 AVMutableCompositionTrack

AVMutableCompositionTrack 中提供的属性如下:

当我们获取了一个 AVMutableCompositionTrack 实例对象后,便可以通过以下方法对其进行添加或移除数据片段:

1.4.6 AVAssetTrackSegment

媒体资源 中的集合 管理着单条时间线上的媒体数据片段,而每个数据片段则由 类进行描述。

AVAssetTrackSegment 有两个属性:

timeMapping 描述的是数据片段在整个媒体文件中所处的时间范围.timeMapping 是一个结构体,拥有两个成员,对于编辑中的媒体数据片段,它们分别表示数据在源文件中的位置和目标文件中的位置.

empty 描述该数据片段是否为空,如果为空,其 timeMapping.source.start 为 kCMTimeInvalid

1.4.7 AVCompositionTrackSegment

在编辑媒体文件时,在描述数据时,使用的是 的子类 ,它的主要属性和方法如下:

1.5 AVFoundation 之 视音频媒体捕获

image

AVFoundation 媒体捕捉1

image

AVFoundation 媒体捕捉2

image

AVFoundation 媒体捕捉3

摄像机和麦克风的记录输入由捕获会话管理。捕获会话协调从输入设备到输出(如电影文件)的数据流。您可以为单个会话配置多个输入和输出,甚至在会话运行时也是如此。向会话发送消息以启动和停止数据流。此外,您可以使用预览层的实例向用户显示摄像机正在录制的内容。

更多关于视频捕获的详情可以参考苹果官方文档:Still and Video Media Capture章节

要管理来自摄像机或麦克风等设备的捕获,您需要组装对象来表示输入和输出,并使用AVCaptureSession实例来协调它们之间的数据流。你需要最低限度:

表示输入设备的AVCaptureDevice实例,如摄像机或麦克风

AVCaptureInput的一个具体子类的实例,用于配置来自输入设备的端口

AVCaptureOutput的一个具体子类的实例,用于管理电影文件或静态图像的输出

AVCaptureSession的一个实例,用于协调从输入到输出的数据流

要向用户显示摄像机记录内容的预览,可以使用AVCaptureVideoPreviewLayer的一个实例(CALayer的一个子类)。

您可以配置多个输入和输出,由单个会话进行协调,如图4-1所示:

image

配置多个输入和输出,由单个会话进行协调

对于许多应用程序,这是您需要的尽可能多的细节。但是,对于某些操作(例如,如果希望监视音频通道中的功率级别),需要考虑如何表示输入设备的各个端口,以及如何将这些端口连接到输出。

捕获会话中的捕获输入和捕获输出之间的连接由AVCaptureConnection对象表示。捕获输入(AVCaptureInput的实例)有一个或多个输入端口(AVCaptureInputPort的实例)。捕获输出(AVCaptureOutput的实例)可以接受来自一个或多个源的数据(例如,AVCaptureMovieFileOutput对象同时接受视频和音频数据)。

当您将输入或输出添加到会话时,会话将在所有兼容的捕获输入端口和捕获输出之间形成连接,如图4-2所示。捕获输入和捕获输出之间的连接由AVCaptureConnection对象表示。

image

AVCaptureConnection表示输入和输出之间的连接

您可以使用捕获连接来启用或禁用来自给定输入或到给定输出的数据流。您还可以使用连接来监视音频通道中的平均和峰值功率级别。

通过麦克风、摄像机等设备,可以捕获外界的声音和影像。要处理设备捕获的数据,需要使用 AVCaptureDevice 类描述设备,使用 AVCaptureInput 配置数据从设备的输入,使用 类管理数据到文件的写入,而数据的输入到写出,需要使用 类进行协调。此外,可以使用 类显示相机正在拍摄的画面。

一个设备可以有多个输入,使用 类描述这些输入,用 类描述具体类型的输入与输出的关系,可以实现更精细的数据处理。

1.5.1 AVCaptureSession

`AVCaptureSession 是捕获视听数据的核心类,它协调数据的输入和输出。创建一个 AVCaptureSession 类的对象时,可以指定最终得到的视听数据的质量,当然这个质量与设备也有关系,通常在设置之前,可以调用方法判断 session 是否支持要设置的质量。

AVCaptureSession 类实例可设置的数据质量有 、 、 、 等。在进行设置之前,可以调用 中的方法进行校验。

设置好对象后,可调用下面的方法,添加、移除输入、输出。

开始执行 session 或者结束执行,调用下面的方法:

对于正在执行中的 session ,要对其进行改变,所作出的改变,应放在下面两个方法之间。

AVCaptureSession 开始执行、结束执行、执行过程中出错或被打断时,都会发出通知,通过注册下面的通知,可以获取我们感兴趣的信息。

通过 AVCaptureSessionErrorKey 可以获取出错的原因

开始 session

结束 session

通过 AVCaptureSessionInterruptionReasonKey 可以获取被打断的原因

打断结束,session 重新开始

1.5.2 AVCaptureDevice

AVCaptureDevice是用来描述设备属性的类,要捕获视听数据,需要获取相应的设备,使用该类获取有效的设备资源。这个设备资源列表是随时变动的,其在变动时,会发送 或 通知,以告知有设备连接或断开。

在获取设备之前,要先确定要获取的设备的类型 ,设备的位置 ,也可以通过要获取的媒体数据类型进行设备的选择。

获取设备后,可以保存它的唯一标识、模型标识、名称等信息,以待下次用来获取设备。

获取一个设备后,可以通过修改它的属性来满足自己的需要。

flashMode 闪光灯的模式(AVCaptureFlashModeOff 、AVCaptureFlashModeOn 、AVCaptureFlashModeAuto)

torchMode 手电筒的模式(AVCaptureTorchModeOff 、AVCaptureTorchModeOn 、AVCaptureTorchModeAuto)

torchLevel 手电筒的亮度(0~1)

focusMode 聚焦模式(AVCaptureFocusModeLocked 、AVCaptureFocusModeAutoFocus 、AVCaptureFocusModeContinuousAutoFocus)

exposureMode 曝光模式(AVCaptureExposureModeLocked 、AVCaptureExposureModeAutoExpose 、AVCaptureExposureModeContinuousAutoExposure 、AVCaptureExposureModeCustom)

whiteBalanceMode 白平衡模式(AVCaptureWhiteBalanceModeLocked 、AVCaptureWhiteBalanceModeAutoWhiteBalance 、AVCaptureWhiteBalanceModeContinuousAutoWhiteBalance)

在修改这些属性时,应先判断当前设备是否支持要设置的属性值,并且所有的属性修改都要放在下面两个方法之间,以保证属性能够被正确设置。

在调用硬件设备之前,应先判断应用是否拥有相应的权限,其权限分为以下几种:

AVAuthorizationStatusNotDetermined 未定义

AVAuthorizationStatusRestricted 无权限(因某些原因,系统拒绝权限)

AVAuthorizationStatusDenied 无权限(用户拒绝)

AVAuthorizationStatusAuthorized 有权限

1.5.3 AVCaptureDeviceInput

是 AVCaptureInput 的子类,使用一个 类实例创建该类的实例,其管理设备的输入。

在创建了实例对象后,将其添加到 session 中。

1.5.4 AVCaptureOutput

AVCaptureOutput 是一个抽象类,通常使用的是它的子类

AVCaptureMovieFileOutput 用来生成一个影视文件

AVCaptureVideoDataOutput 用来处理输入的视频的帧

AVCaptureAudioDataOutput 用来处理音频数据

AVCaptureStillImageOutput 用来获取图片

在创建了具体的子类后,将它添加到 session 中

1.5.5 AVCaptureFileOutput

是 的子类,是 、 的父类。这个类中定义了文件输出时的地址、时长、容量等属性。

1.5.6 AVCaptureFileOutputRecordingDelegate

是文件记录过程中需要用到的协议,它通常的作用是告知代理对象文件记录结束了。

1.5.7 AVCaptureFileOutputDelegate

这个协议只用于 Mac OS X 系统下,它给了客户端精准操控数据的机会。

1.5.8 AVCaptureAudioFileOutput

AVCaptureAudioFileOutput 是 AVCaptureFileOutput 的子类,该类用于将媒体数据记录为一个音频文件。

1.5.9 AVCaptureVideoDataOutput

是 的子类,该类可以用来处理捕获的每一个视频帧数据。创建一个该类的实例对象后,要调用下面的方法设置一个代理对象,及调用代理对象所实现的协议方法的队列。

指定的队列 sampleBufferCallbackQueue 必需是串行队列以保证传递的帧是按记录时间先后传递的。

1.5.10 AVCaptureVideoDataOutputSampleBufferDelegate

该协议用来处理接收的每一个帧数据,或者提示客户端有帧数据被丢弃。

1.5.11 AVCaptureVideoPreviewLayer

是 的子类,使用该类可以实现捕获视频的显示。使用一个 session 创建一个该类对象,而后将该类对象插入到图层树中,从而显示捕获的视频。

修改 的属性 videoGravity 值,可以选择显示捕获视频时的界面大小变化方式,它有以下可选值:

AVLayerVideoGravityResize 默认值,直接铺满屏幕,及时画面变形

AVLayerVideoGravityResizeAspect 保持画面的横纵比,不铺满屏幕,多余的空间显示黑色

AVLayerVideoGravityResizeAspectFill 保持画面的横纵比,铺满屏幕,多余的画面进行裁剪

1.5.12 AVCaptureAudioDataOutput

是 的子类,该类可以处理接收到的音频数据。同 类似,该类也提供了一个方法,用于设置代理对象,以及调用代理对象实现的协议方法时的队列。

1.5.13 AVCaptureAudioDataOutputSampleBufferDelegate

该协议提供了一个方法,用来实现对音频数据的接收处理。

1.5.14 AVAssetImageGenerator

使用 生成视频资源的缩略图,使用 对象创建 对象,可以使用类方法或实例方法,如下:

当然,在此之前,最好调用 AVAsset 中的方法 来判断 asset 中是否有可视媒体数据。如果有,那么再创建 对象,而后再调用下面的方法,来获取一张或多张图片。

1.6 AVFoundation 之媒体流输出

image

AVFoundation 视频音频输出

对媒体数据资源进行简单的转码或裁剪,使用 类便足够了,但是更深层次的修改媒体资源,便需要用到 类和 类。

只能与一个资源 asset 相关联,且不能用来读取实时数据,在开始读取数据之前,需要为 reader 添加 的实例对象。这个实例对象描述的是待读取的数据资源来源类型,通常使用 、 、 三种子类。

可以将来自多个数据源的数据以指定的格式写入到一个指定的文件中,且其只能对应一个文件。在写文件之前,需要用每一个 类实例对象来描述相应的数据源。每一个 实例对象接收的数据都应是 类型的变量。如果使用 类也可以直接将 类型的变量数据添加到 writer input 中。

与 结合起来使用,便可以对读取的数据进行相应的编辑修改,而后写入到一个文件中并保存。

1.6.1 AVAssetReader

使用该类读取媒体资源,其提供的初始化方法与一个 asset 相关联。

1.6.2 AVAssetReaderOutput

是用来描述待读取的数据的抽象类,读取资源时,应创建该类的对象,并添加到相应的 实例对象中去。

1.6.3 AVAssetReaderTrackOutput

是 的子类,它用来描述待读取的数据来自 asset track ,在读取前,还可以对数据的格式进行修改。

1.6.4 AVAssetReaderAudioMixOutput

是 的子类,它用来描述待读取的数据来自音频组合数据。创建该类实例对象提供的参数 audioTracks 集合中的每一个 asset track 都属于相应的 reader 中的 asset 实例对象,且类型为 AVMediaTypeAudio 。

参数 audioSettings 给出了音频数据的格式设置。

此外,该类的 audioMix 属性,描述了从多个 track 中读取的音频的音量变化情况:

1.6.5 AVAssetReaderVideoCompositionOutput

是 的子类,该类用来表示要读取的类是组合的视频数据。

同 类似,该类也提供了两个创建实例的方法,需要提供的参数的 videoTracks 集合中每一个 track 都是 与 reader 相关联的 asset 中的 track 。

该类的属性 videoComposition 同样描述了每个 track 的帧的显示方式。

1.7 AVFoundation 之媒体的时间和数据

image

AVFoundation 媒体的时间与数据

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容