百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 热门文章 > 正文

阿里云短视频 SDK For Android 快速接入

bigegpt 2024-08-28 12:15 8 浏览

摘要: 近期的一些创意短视频 App 风靡年轻群体,比较典型的例如抖音、MUSE 等,阿里云也适时地推出了简单易用的短视频 SDK,帮助开发者们以较低的成本快速引入功能完备的创意短视频功能。本文主要介绍如何快速接入阿里云短视频 SDK 的三个版本(基础版、标准版和专业版)。

前言

近期的一些创意短视频 App 风靡年轻群体,比较典型的例如抖音、MUSE 等,阿里云也适时地推出了简单易用的短视频 SDK,帮助开发者们以较低的成本快速引入功能完备的创意短视频功能。

本文主要介绍如何快速接入阿里云短视频 SDK 的三个版本(基础版、标准版和专业版)。帮助开发者以最快的速度了解接入的基本方式。

本文描述的阿里云短视频 SDK 版本基于 3.4.0,后续升级接口变动请参考 阿里云短视频 SDK 文档。

示例工程代码为 Kotlin,Java 接入类似。

正文

由于三个版本接入方式大同小异,本文将着重介绍基础版接入过程,标准版和专业版可以基于基础版方式接入,后续仅说明接入差异的地方。

基础版接入

1. 引入 aar 以及 so

目前 aar 平台版本最低要求 >= 4.3,先从SDK 下载页面下载相应版本的 SDK,解压之后,将 libs
文件夹下的 QuSdk-RC.aar拷到 Android 工程模块中的 libs文件夹下,将 jniLibs文件夹下的 armeabi-v7a文件夹也整体拷贝到 libs文件夹下。

拷贝完成之后目录的文件如下:

之后按照如下方式修改 Android 项目工程主模块下的 build.gradle文件:

Step1. 修改 jniLibs的源文件夹;

android {
 sourceSets.main {
 jniLibs.srcDir "libs"
 }
}

Step2. 将 libs文件夹加入仓库中;

repositories {
 flatDir {
 dirs 'libs'
 }
}

Step3. 增加 aar 所需依赖。

dependencies {
 implementation(name: 'QuSdk-RC', ext: 'aar')
 implementation 'com.android.support:appcompat-v7:24.2.1'
 implementation 'com.android.support:design:24.2.1'
 implementation 'com.google.code.findbugs:jsr305:3.0.0'
 implementation 'com.github.bumptech.glide:glide:3.7.0'
 implementation 'pub.devrel:easypermissions:0.2.1'
 implementation 'com.squareup.okhttp3:okhttp:3.2.0'
 implementation 'com.github.bumptech.glide:okhttp3-integration:1.4.0@aar'
 implementation 'com.squareup.okio:okio:1.12.0'
 implementation 'com.google.code.gson:gson:2.8.0'
}

如果此处遭遇 java.lang.NoSuchFieldError 错误,可以参考短视频安卓常见问题解决。

2. 初始化 SDK

请根据具体的项目情况选择合适的 SDK 初始化时机,Demo 工程在 Applicatioin 的 onCreate()方法中初始化。

package me.bogerchan.alishortvideodemoimport android.app.Applicationimport com.aliyun.common.httpfinal.QupaiHttpFinal/**
* Created by hb.chen on 2018/1/6.
*/class MyApplication : Application() { override fun onCreate() { super.onCreate()
 System.loadLibrary("QuCore-ThirdParty")
 System.loadLibrary("QuCore")
 QupaiHttpFinal.getInstance().initOkHttpFinal()
 }
}

3. 开始书写你的业务逻辑

经过上述过程,实际上已经接入完成,这时候你可以参考文档直接开始使用各种 API 了,附下示例代码。

package me.bogerchan.alishortvideodemoimport android.Manifestimport android.app.Activityimport android.content.Intentimport android.content.pm.PackageManagerimport android.os.Bundleimport android.support.v4.app.ActivityCompatimport android.support.v7.app.AppCompatActivityimport android.widget.Toastimport com.aliyun.demo.recorder.AliyunVideoRecorderimport com.aliyun.struct.common.VideoQualityimport com.aliyun.struct.snap.AliyunSnapVideoParamimport me.bogerchan.alishortvideodemo.basic.Rclass MainActivity : AppCompatActivity() {
 companion object {
 val REQUEST_CODE_RECORD_VIDEO = 1
 val REQUEST_CODE_FOR_PERMISSION = 2
 } override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState)
 setContentView(R.layout.activity_main)
 findViewById(R.id.btn_start_record).setOnClickListener {
 startRecordActivity()
 }
 ActivityCompat.requestPermissions(this,
 arrayOf(Manifest.permission.WRITE_EXTERNAL_STORAGE,
 Manifest.permission.READ_EXTERNAL_STORAGE,
 Manifest.permission.CAMERA,
 Manifest.permission.RECORD_AUDIO),
 REQUEST_CODE_FOR_PERMISSION)
 } private fun startRecordActivity() {
 val recordParam = AliyunSnapVideoParam.Builder()
 .setResolutionMode(AliyunSnapVideoParam.RESOLUTION_720P)
 .setRatioMode(AliyunSnapVideoParam.RATIO_MODE_9_16)
 .setRecordMode(AliyunSnapVideoParam.RECORD_MODE_AUTO)
 .setNeedClip(true)
 .setMaxDuration(10000)
 .setMinDuration(2000)
 .setVideQuality(VideoQuality.HD)
 .setSortMode(AliyunSnapVideoParam.SORT_MODE_MERGE)
 .build()
 AliyunVideoRecorder.startRecordForResult(this, REQUEST_CODE_RECORD_VIDEO, recordParam)
 } override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) { super.onActivityResult(requestCode, resultCode, data)
 when (requestCode) {
 REQUEST_CODE_RECORD_VIDEO -> { if (resultCode == Activity.RESULT_OK && data != null) {
 val type = data.getIntExtra(AliyunVideoRecorder.RESULT_TYPE, 0) if (type == AliyunVideoRecorder.RESULT_TYPE_CROP) {
 Toast.makeText(this, "类型为裁剪", Toast.LENGTH_SHORT).show()
 } else if (type == AliyunVideoRecorder.RESULT_TYPE_RECORD) {
 Toast.makeText(this, "文件路径为 " + data.getStringExtra(AliyunVideoRecorder.OUTPUT_PATH), Toast.LENGTH_SHORT).show()
 }
 } else if (resultCode == Activity.RESULT_CANCELED) {
 Toast.makeText(this, "用户取消录制", Toast.LENGTH_SHORT).show()
 }
 }
 }
 } override fun onRequestPermissionsResult(requestCode: Int, permissions: Array<out String>, grantResults: IntArray) { super.onRequestPermissionsResult(requestCode, permissions, grantResults)
 when (requestCode) {
 REQUEST_CODE_FOR_PERMISSION -> {
 grantResults.forEach { if (it == PackageManager.PERMISSION_DENIED) {
 Toast.makeText(this, "没有权限,不玩了", Toast.LENGTH_SHORT).show()
 finish() return@forEach
 }
 }
 }
 }
 }
}

标准版接入

1. 引入 aar 以及 so

标准版相较于基础版,在引入 so 文件时候多了几个文件,同时 aar文件名有所变动。最终拷贝结果如下:

build.gradle文件修改与基础版接入一样,只是需要将接入 aar
文件名替换成标准版对应的名字。

2. 初始化 SDK

相较于基础版,需要加载的 so 增多了几个,其中部分 so 文件作为可选功能根据实际情况决定是否加载,具体可以参阅阿里云短视频 SDK 文档。接入后的 Application 文件参考:

package me.bogerchan.alishortvideodemoimport android.app.Applicationimport com.aliyun.common.httpfinal.QupaiHttpFinal/**
* Created by hb.chen on 2018/1/6.
*/class MyApplication : Application() { override fun onCreate() { super.onCreate()
 System.loadLibrary("aliresample")
 System.loadLibrary("live-openh264")
 System.loadLibrary("QuCore-ThirdParty")
 System.loadLibrary("QuCore")
 QupaiHttpFinal.getInstance().initOkHttpFinal()
 }
}

3. 开始书写你的业务逻辑

经过上述过程,实际上已经接入完成,这时候你可以参考文档直接开始使用各种 API 了,附下示例代码。

package me.bogerchan.alishortvideodemoimport android.Manifestimport android.content.pm.PackageManagerimport android.opengl.GLSurfaceViewimport android.os.Bundleimport android.support.v4.app.ActivityCompatimport android.support.v7.app.AppCompatActivityimport android.widget.Toastimport com.aliyun.recorder.AliyunRecorderCreatorimport com.aliyun.struct.recorder.CameraTypeimport com.aliyun.struct.recorder.MediaInfoimport me.bogerchan.alishortvideodemo.std.Rclass MainActivity : AppCompatActivity() {
 companion object {
 val REQUEST_CODE_FOR_PERMISSION = 1
 } private val mRecorder by lazy {
 AliyunRecorderCreator.getRecorderInstance(this)
 } private var mCameraType = CameraType.FRONT
 override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState)
 setContentView(R.layout.activity_main)
 ActivityCompat.requestPermissions(this,
 arrayOf(Manifest.permission.WRITE_EXTERNAL_STORAGE,
 Manifest.permission.READ_EXTERNAL_STORAGE,
 Manifest.permission.CAMERA,
 Manifest.permission.RECORD_AUDIO),
 REQUEST_CODE_FOR_PERMISSION)
 initAliyunRecorder()
 findViewById(R.id.btn_start_record).setOnClickListener {
 Toast.makeText(this, "开始录制片段", Toast.LENGTH_SHORT).show()
 mRecorder.startRecording()
 }
 findViewById(R.id.btn_stop_record).setOnClickListener {
 Toast.makeText(this, "停止录制片段", Toast.LENGTH_SHORT).show()
 mRecorder.stopRecording()
 }
 findViewById(R.id.btn_finish_record).setOnClickListener {
 Toast.makeText(this, "结束录制", Toast.LENGTH_SHORT).show()
 mRecorder.finishRecording()
 }
 findViewById(R.id.btn_change_camera_type).setOnClickListener {
 Toast.makeText(this, "切换前后置", Toast.LENGTH_SHORT).show()
 mRecorder.switchCamera()
 }
 } override fun onStart() { super.onStart()
 mRecorder.startPreview()
 } override fun onPause() { super.onPause()
 mRecorder.stopPreview()
 } override fun onDestroy() { super.onDestroy()
 AliyunRecorderCreator.destroyRecorderInstance()
 } private fun initAliyunRecorder() {
 mRecorder.setDisplayView(findViewById(R.id.glsv_content) as GLSurfaceView)
 val mediaInfo = MediaInfo()
 mediaInfo.videoWidth = 800
 mediaInfo.videoHeight = 1200
 mediaInfo.isHWAutoSize = true
 mRecorder.setMediaInfo(mediaInfo)
 mRecorder.setCamera(mCameraType)
 mRecorder.setOutputPath(externalCacheDir.absolutePath + "/capture.mp4")
 } override fun onRequestPermissionsResult(requestCode: Int, permissions: Array<out String>, grantResults: IntArray) { super.onRequestPermissionsResult(requestCode, permissions, grantResults)
 when (requestCode) {
 REQUEST_CODE_FOR_PERMISSION -> {
 grantResults.forEach { if (it == PackageManager.PERMISSION_DENIED) {
 Toast.makeText(this, "没有权限,不玩了", Toast.LENGTH_SHORT).show()
 finish() return@forEach
 }
 }
 }
 }
 }
}

专业版接入

1. 引入 aar 以及 so

专业版相较于基础版,在引入 so 文件时候多了几个文件,同时 aar文件名有所变动。最终拷贝结果如下:

build.gradle 文件修改与基础版接入一样,只是需要将接入 aar文件名替换成专业版对应的名字。

2. 初始化 SDK

相较于基础版,需要加载的 so 增多了几个,其中部分 so 文件作为可选功能根据实际情况决定是否加载,具体可以参阅阿里云短视频 SDK 文档。接入后的 Application 文件参考:

package me.bogerchan.alishortvideodemoimport android.app.Applicationimport com.aliyun.common.httpfinal.QupaiHttpFinal/**
* Created by hb.chen on 2018/1/6.
*/class MyApplication : Application() { override fun onCreate() { super.onCreate()
 System.loadLibrary("live-openh264")
 System.loadLibrary("QuCore-ThirdParty")
 System.loadLibrary("QuCore")
 System.loadLibrary("FaceAREngine")
 System.loadLibrary("AliFaceAREngine")
 QupaiHttpFinal.getInstance().initOkHttpFinal()
 }
}

3. 开始书写你的业务逻辑

经过上述过程,实际上已经接入完成,这时候你可以参考文档直接开始使用各种 API 了,附下示例代码。

package me.bogerchan.alishortvideodemoimport android.Manifestimport android.content.pm.PackageManagerimport android.opengl.GLSurfaceViewimport android.os.Bundleimport android.support.v4.app.ActivityCompatimport android.support.v7.app.AppCompatActivityimport android.widget.Toastimport com.aliyun.recorder.AliyunRecorderCreatorimport com.aliyun.struct.recorder.CameraTypeimport com.aliyun.struct.recorder.MediaInfoimport me.bogerchan.alishortvideodemo.pro.Rclass MainActivity : AppCompatActivity() {
 companion object {
 val REQUEST_CODE_FOR_PERMISSION = 1
 } private val mRecorder by lazy {
 AliyunRecorderCreator.getRecorderInstance(this)
 } private var mCameraType = CameraType.FRONT
 override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState)
 setContentView(R.layout.activity_main)
 ActivityCompat.requestPermissions(this,
 arrayOf(Manifest.permission.WRITE_EXTERNAL_STORAGE,
 Manifest.permission.READ_EXTERNAL_STORAGE,
 Manifest.permission.CAMERA,
 Manifest.permission.RECORD_AUDIO),
 REQUEST_CODE_FOR_PERMISSION)
 initAliyunRecorder()
 findViewById(R.id.btn_start_record).setOnClickListener {
 Toast.makeText(this, "开始录制片段", Toast.LENGTH_SHORT).show()
 mRecorder.startRecording()
 }
 findViewById(R.id.btn_stop_record).setOnClickListener {
 Toast.makeText(this, "停止录制片段", Toast.LENGTH_SHORT).show()
 mRecorder.stopRecording()
 }
 findViewById(R.id.btn_finish_record).setOnClickListener {
 Toast.makeText(this, "结束录制", Toast.LENGTH_SHORT).show()
 mRecorder.finishRecording()
 }
 findViewById(R.id.btn_change_camera_type).setOnClickListener {
 Toast.makeText(this, "切换前后置", Toast.LENGTH_SHORT).show()
 mRecorder.switchCamera()
 }
 } override fun onStart() { super.onStart()
 mRecorder.startPreview()
 } override fun onPause() { super.onPause()
 mRecorder.stopPreview()
 } override fun onDestroy() { super.onDestroy()
 AliyunRecorderCreator.destroyRecorderInstance()
 } private fun initAliyunRecorder() {
 mRecorder.setDisplayView(findViewById(R.id.glsv_content) as GLSurfaceView)
 val mediaInfo = MediaInfo()
 mediaInfo.videoWidth = 800
 mediaInfo.videoHeight = 1200
 mediaInfo.isHWAutoSize = true
 mRecorder.setMediaInfo(mediaInfo)
 mRecorder.setCamera(mCameraType)
 mRecorder.needFaceTrackInternal(true)
 mRecorder.setOutputPath(externalCacheDir.absolutePath + "/capture.mp4")
 } override fun onRequestPermissionsResult(requestCode: Int, permissions: Array<out String>, grantResults: IntArray) { super.onRequestPermissionsResult(requestCode, permissions, grantResults)
 when (requestCode) {
 REQUEST_CODE_FOR_PERMISSION -> {
 grantResults.forEach { if (it == PackageManager.PERMISSION_DENIED) {
 Toast.makeText(this, "没有权限,不玩了", Toast.LENGTH_SHORT).show()
 finish() return@forEach
 }
 }
 }
 }
 }
}

结语

至此已经介绍完了阿里云短视频 SDK 的接入方法,示例代码展示的仅仅只是阿里云视频 SDK 强大功能的冰山一角,开发者们可以通过相关的 SDK 文档获取更多的接口信息。如果集成过程中遇到问题,在联系客服之前不妨先看下 常见问题解决,说不定你的问题就在里面。

相关推荐

当Frida来“敲”门(frida是什么)

0x1渗透测试瓶颈目前,碰到越来越多的大客户都会将核心资产业务集中在统一的APP上,或者对自己比较重要的APP,如自己的主业务,办公APP进行加壳,流量加密,投入了很多精力在移动端的防护上。而现在挖...

服务端性能测试实战3-性能测试脚本开发

前言在前面的两篇文章中,我们分别介绍了性能测试的理论知识以及性能测试计划制定,本篇文章将重点介绍性能测试脚本开发。脚本开发将分为两个阶段:阶段一:了解各个接口的入参、出参,使用Python代码模拟前端...

Springboot整合Apache Ftpserver拓展功能及业务讲解(三)

今日分享每天分享技术实战干货,技术在于积累和收藏,希望可以帮助到您,同时也希望获得您的支持和关注。架构开源地址:https://gitee.com/msxyspringboot整合Ftpserver参...

Linux和Windows下:Python Crypto模块安装方式区别

一、Linux环境下:fromCrypto.SignatureimportPKCS1_v1_5如果导包报错:ImportError:Nomodulenamed'Crypt...

Python 3 加密简介(python des加密解密)

Python3的标准库中是没多少用来解决加密的,不过却有用于处理哈希的库。在这里我们会对其进行一个简单的介绍,但重点会放在两个第三方的软件包:PyCrypto和cryptography上,我...

怎样从零开始编译一个魔兽世界开源服务端Windows

第二章:编译和安装我是艾西,上期我们讲述到编译一个魔兽世界开源服务端环境准备,那么今天跟大家聊聊怎么编译和安装我们直接进入正题(上一章没有看到的小伙伴可以点我主页查看)编译服务端:在D盘新建一个文件夹...

附1-Conda部署安装及基本使用(conda安装教程)

Windows环境安装安装介质下载下载地址:https://www.anaconda.com/products/individual安装Anaconda安装时,选择自定义安装,选择自定义安装路径:配置...

如何配置全世界最小的 MySQL 服务器

配置全世界最小的MySQL服务器——如何在一块IntelEdison为控制板上安装一个MySQL服务器。介绍在我最近的一篇博文中,物联网,消息以及MySQL,我展示了如果Partic...

如何使用Github Action来自动化编译PolarDB-PG数据库

随着PolarDB在国产数据库领域荣膺桂冠并持续获得广泛认可,越来越多的学生和技术爱好者开始关注并涉足这款由阿里巴巴集团倾力打造且性能卓越的关系型云原生数据库。有很多同学想要上手尝试,却卡在了编译数据...

面向NDK开发者的Android 7.0变更(ndk android.mk)

订阅Google官方微信公众号:谷歌开发者。与谷歌一起创造未来!受Android平台其他改进的影响,为了方便加载本机代码,AndroidM和N中的动态链接器对编写整洁且跨平台兼容的本机...

信创改造--人大金仓(Kingbase)数据库安装、备份恢复的问题纪要

问题一:在安装KingbaseES时,安装用户对于安装路径需有“读”、“写”、“执行”的权限。在Linux系统中,需要以非root用户执行安装程序,且该用户要有标准的home目录,您可...

OpenSSH 安全漏洞,修补操作一手掌握

1.漏洞概述近日,国家信息安全漏洞库(CNNVD)收到关于OpenSSH安全漏洞(CNNVD-202407-017、CVE-2024-6387)情况的报送。攻击者可以利用该漏洞在无需认证的情况下,通...

Linux:lsof命令详解(linux lsof命令详解)

介绍欢迎来到这篇博客。在这篇博客中,我们将学习Unix/Linux系统上的lsof命令行工具。命令行工具是您使用CLI(命令行界面)而不是GUI(图形用户界面)运行的程序或工具。lsoflsof代表&...

幻隐说固态第一期:固态硬盘接口类别

前排声明所有信息来源于网络收集,如有错误请评论区指出更正。废话不多说,目前固态硬盘接口按速度由慢到快分有这几类:SATA、mSATA、SATAExpress、PCI-E、m.2、u.2。下面我们来...

新品轰炸 影驰SSD多款产品登Computex

分享泡泡网SSD固态硬盘频道6月6日台北电脑展作为全球第二、亚洲最大的3C/IT产业链专业展,吸引了众多IT厂商和全球各地媒体的热烈关注,全球存储新势力—影驰,也积极参与其中,为广大玩家朋友带来了...