当前位置: 首页 > news >正文

(十五)深入了解 AVFoundation - 编辑:音视频裁剪与拼接

引言

在 iOS 开发中,视频编辑一直是一个门槛不低但极具潜力的能力场景。无论是用户上传短视频前的裁剪处理,还是内容拼接生成精彩合集,都离不开对 AVFoundation 框架的深入理解。

本篇我们将聚焦 AVFoundation 的两个核心功能实现

  1. 如何裁剪视频中的某一段时间范围;
  2. 如何拼接多个视频(或音频)片段为一段完整内容。

而这一切,我们都会通过一个可运行的实战 Demo 来完成:你可以选择本地素材,设置裁剪时间,预览播放效果,并导出成成品视频。

一、裁剪指定时间段视频(实战功能 1)

裁剪视频是编辑场景中最常见的需求,比如保留精彩片段、生成短视频或裁掉片头片尾。在 AVFoundation 中,这项工作可以通过 AVAssetExportSession 来轻松实现。

1.1 实现原理

整体流程如下:

  1. 使用 AVAsset 加载原始视频;
  2. 设置导出时间范围(timeRange);
  3. 创建导出任务,生成目标视频。

只需几行代码即可完成基本裁剪操作。

1.2 示例代码

let asset = AVAsset(url: inputURL)guard let exportSession = AVAssetExportSession(asset: asset, presetName: .presetHighestQuality) else {print("❌ 无法创建导出会话")return
}let startTime = CMTime(seconds: 3, preferredTimescale: 600)
let duration = CMTime(seconds: 5, preferredTimescale: 600)
exportSession.timeRange = CMTimeRange(start: startTime, duration: duration)exportSession.outputURL = outputURL
exportSession.outputFileType = .mp4
exportSession.exportAsynchronously {switch exportSession.status {case .completed:print("✅ 导出成功")case .failed:print("❌ 导出失败: \(exportSession.error?.localizedDescription ?? "未知错误")")default:break}
}

这段代码会将原始视频中的第 3 秒开始,截取 5 秒长度(即第 3~8 秒)作为输出结果。

1.3 常见问题与注意事项

  1. 起始时间越界:startTime + duration 不能超过视频总时长,否则导出失败
  2. 导出前几帧黑屏:起始位置若非关键帧,可能出现黑屏,建议使用更高的导出质量或手动设置关键帧
  3. presetName:建议使用 .presetHighestQuality 保证画质,可根据需要替换为 .presetMediumQuality 等
  4. 输出类型:常用为 .mp4(即 .mov 对应 .mov 容器),需确保目标路径无文件冲突

二、拼接多个视频(实战功能 2)

除了裁剪,另一个常见场景是将多个片段拼接成一段完整视频,比如:

  • 用户选择多个素材后顺序合成;
  • 为视频添加片头片尾;
  • 制作合集、片段合成。

这类需求的核心就是「轨道拼接」,AVFoundation 提供了 AVMutableComposition 来完成这项任务。

2.1 实现原理

拼接的思路是:

  1. 创建一个空的 AVMutableComposition;
  2. 添加一个视频轨道(AVMutableCompositionTrack);
  3. 将每个片段依次插入轨道,设置其插入时间点;
  4. 最终通过 AVAssetExportSession 导出成品。
注意:如果同时拼接音频,也需添加对应的音频轨并分别插入。

2.2 示例代码

let composition = AVMutableComposition()
guard let videoTrack = composition.addMutableTrack(withMediaType: .video,preferredTrackID: kCMPersistentTrackID_Invalid) else {print("❌ 无法创建视频轨道")return
}var currentTime = CMTime.zerofor asset in assetList {guard let sourceTrack = asset.tracks(withMediaType: .video).first else { continue }let range = CMTimeRange(start: .zero, duration: asset.duration)try? videoTrack.insertTimeRange(range, of: sourceTrack, at: currentTime)currentTime = currentTime + asset.duration
}

导出方式与裁剪完全一致:

let exportSession = AVAssetExportSession(asset: composition, presetName: .presetHighestQuality)!
exportSession.outputURL = outputURL
exportSession.outputFileType = .mp4
exportSession.exportAsynchronously {print("✅ 拼接导出完成")
}

2.3 注意事项

  1. 分辨率不一致:不同尺寸拼接后可能出现跳变,建议统一画面尺寸(将在后续文章中介绍)
  2. 帧率不一致:可能导致导出失败或播放卡顿
  3. 音频处理:如果需要保留音频轨道,需同时添加 audioTrack 并插入对应内容
  4. 拼接顺序:由代码中 insertTimeRange 的顺序决定

结语:实战篇预告

本篇我们用最小的实现方式,介绍了视频裁剪与拼接的原理与代码实现。但如果要把这些功能“真正用在 App 中”,就需要:

  1. 支持用户选择素材、设置时间段;
  2. 能够实时预览播放与导出进度;
  3. 有清晰的项目结构,便于扩展更多编辑功能。

这些内容将在下一篇中展开 —— 我将带你一步步剖析一个完整 AVFoundation 视频编辑 Demo 的设计与实战实现。

相关文章:

  • Python 脚本,用于将 PDF 文件高质量地转换为 PNG 图像
  • 设计模式:单例模式多种方式的不同实现
  • http测试方法三
  • 【动手学深度学习】3.7. softmax回归的简洁实现
  • 答题考试系统小程序ThinkPHP+UniApp
  • 【科研绘图系列】R语言绘制论文组图系列(multiple plots)
  • ai智能题库小程序题库刷题系统框架设计
  • 在Kibana上新增Elasticsearch生命周期管理
  • 【Spark征服之路-2.8-Spark-Core编程(四)】
  • Qwen3-Embedding-Reranker本地部署教程:8B 参数登顶 MTEB 多语言榜首,100 + 语言跨模态检索无压力!
  • 奇异值分解
  • 深度学习:人工神经网络基础概念
  • 数据驱动SEO:8种自动化实践,精准提升排名与监控效能
  • ECharts:柱状图背景虚线
  • 16. 线性表的链式表示和实现(5)
  • git如何强制拉取远程分支覆盖本地分支
  • 数据仓库面试题合集⑤
  • 在IntelliJ IDEA中使用Maven配置Tomcat环境
  • 条件查询详细说明
  • 如何在 Android 上备份音乐:保护歌曲的 5 种方法
  • 最专业的外贸网站建设/徐州seo外包
  • 厚街网站建设公司/百度网站大全
  • 保定专业做网站的公司哪家好/如何宣传网站
  • 网站 建设初步/济南百度竞价代运营
  • 做收费课程网站/广州seo软件
  • wordpress小机巧/肇庆百度快照优化