前面将的都是如何使用C/C++文件生成so动态库,那么在使用别人的so动态库的时候应该怎么做呢?这篇文章就是使用一个变声功能的动态库,完成对于以有so动态库的说明。
动态库来源
在互联网中,有着许许多多动态库,很多厂商也对外提供动态库供开发者调用,例如高德地图的动态库,做地图开发的时候还是很方便的
本文主要讲一个可以使声音改变的动态库,这个动态库主要用于游戏中,游戏引擎中有使用到
这就是fmod动态库,首先我们要去下载其动态库文件
官网地址
先要注册才能下载其文件,按照步骤来就好在其下载界面,有
FMOD Studio API
,这里可以选择版本下载,我写这篇博客的时候,最新版本是1.10.07
,在这里就不下载最新的了,我选择的是1.08.28
,也就是1.08
的最后一个版本。
添加到项目中
解压下载的文件,发现在api文件夹下有三个目录:
fsbank
,lowlevel
,studio
这里选择
lowlevel
,这是基于普遍使用选择的,也可以选择studio
,其功能更为强大,不过相对地也更需要运算性能在Android项目中新建libs目录,将
fmod.jar
拷贝至libs目录新建jni目录,将
armeabi
或armeabi-v7a
下的so文件拷贝至jni目录,将lowlevel
目录下的inc
头文件拷贝至jni文件夹,在这里先实现原声播放的功能,故在lowlevel
下的examples
下找到play_sound.cpp
源文件,将其放在jni目录下,打开文件得知,其依赖的common.h
头文件并不在inc
中,找到common.h
并拷贝至jni中,逐步寻找缺失的依赖文件,导入到jni中,整理完成后的jni文件目录如下:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17│ Android.mk
│ common.cpp
│ common.h
│ common_platform.cpp
│ common_platform.h
│ libfmod.so
│ libfmodL.so
│ play_sound.cpp
└─inc
fmod.h
fmod.hpp
fmod_codec.h
fmod_common.h
fmod_dsp.h
fmod_dsp_effects.h
fmod_errors.h
fmod_output.h
修改文件使其能够调用
在
lowlevel
目录下,有Java的调用示例,在这里直接使用这个MainActivity.java
进行修改调用阅读
MainActivity.java
源代码,发现其使用的是动态获取权限,为方便使用,直接在清单文件中生命其权限,将其动态申请注释掉,在动态库加载时候,发现加载了一些没有的动态库,将没有的动态库去掉,加上自己的动态库,注意到jni中的调用方法和现有包名不统一,修改之,并将清单文件中的启动活动包名也修改
1 | package org.fmod.example; |
使用
Android Tools
添加本地支持修改
Android.mk
文件,并记录动态库的文件名,将其加载至MainActivity.java
1 | LOCAL_PATH := $(call my-dir) |
编译项目
此时便可以编译项目了,编译时候会提示有些文件找不到,那是因为包含文件路径不对造成的,此时修改包含文件路径即可
文件包含错误解决以后,再次编译,发现很多函数找不到,此时是因为编译时候那些函数的实现没有编译到项目,修改
Android.mk
文件,加入依赖实现
1 | LOCAL_PATH := $(call my-dir) |
另外,由于用到了STL库,需要在配置里说明,在jni下新建
Application.mk
文件,写入以下配置1
APP_STL := gnustl_static
至此,项目修改完毕,便可以生成apk了。运行界面如下:
仿QQ变声效果实现
在大致了解fmod以后,就可以做一些基于fmod的项目了,正好QQ有一个变声的功能,这里就使用fmod去实现
采集素材
直接将QQ安装包解压,就可以得到图片素材,将其加入到素材中
编辑界面
编写界面,这里直接采用他人布局好的文件1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent" >
<LinearLayout
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:orientation="vertical"
android:background="#FFF"
android:paddingBottom="20dp" >
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:layout_marginTop="20dp">
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="vertical">
<ImageView
android:id="@+id/btn_normal"
style="@style/AudioImgStyle"
android:src="@drawable/record"
android:onClick="mFix"/>
<TextView
style="@style/AudioTextStyle"
android:text="原声"/>
</LinearLayout>
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="vertical">
<ImageView
android:id="@+id/btn_luoli"
style="@style/AudioImgStyle"
android:src="@drawable/luoli"
android:onClick="mFix"/>
<TextView
style="@style/AudioTextStyle"
android:text="萝莉"/>
</LinearLayout>
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="vertical">
<ImageView
android:id="@+id/btn_dashu"
style="@style/AudioImgStyle"
android:src="@drawable/dashu"
android:onClick="mFix"/>
<TextView
style="@style/AudioTextStyle"
android:text="大叔"/>
</LinearLayout>
</LinearLayout>
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:layout_marginTop="20dp">
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="vertical">
<ImageView
android:id="@+id/btn_jingsong"
style="@style/AudioImgStyle"
android:src="@drawable/jingsong"
android:onClick="mFix"/>
<TextView
style="@style/AudioTextStyle"
android:text="惊悚"/>
</LinearLayout>
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="vertical">
<ImageView
android:id="@+id/btn_gaoguai"
style="@style/AudioImgStyle"
android:src="@drawable/gaoguai"
android:onClick="mFix"/>
<TextView
style="@style/AudioTextStyle"
android:text="搞怪"/>
</LinearLayout>
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="vertical">
<ImageView
android:id="@+id/btn_kongling"
style="@style/AudioImgStyle"
android:src="@drawable/kongling"
android:onClick="mFix"/>
<TextView
style="@style/AudioTextStyle"
android:text="空灵"/>
</LinearLayout>
</LinearLayout>
</LinearLayout>
</RelativeLayout>
style文件1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28<resources>
<style name="AppBaseTheme" parent="android:Theme.Light">
<!--
Theme customizations available in newer API levels can go in
res/values-vXX/styles.xml, while customizations related to
backward-compatibility can go here.
-->
</style>
<!-- Application theme. -->
<style name="AppTheme" parent="AppBaseTheme">
<!-- All customizations that are NOT specific to a particular API-level can go here. -->
</style>
<style name="AudioImgStyle">
<item name="android:layout_width">wrap_content</item>
<item name="android:layout_height">wrap_content</item>
<item name="android:layout_marginLeft">25dp</item>
<item name="android:layout_marginRight">25dp</item>
</style>
<style name="AudioTextStyle">
<item name="android:layout_width">wrap_content</item>
<item name="android:layout_height">wrap_content</item>
<item name="android:layout_marginTop">5dp</item>
<item name="android:layout_gravity">center_horizontal</item>
</style>
</resources>
编辑java代码
主活动1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58package org.fmod.example;
import java.io.File;
import org.fmod.FMOD;
import android.app.Activity;
import android.os.Bundle;
import android.os.Environment;
import android.util.Log;
import android.view.View;
public class QQActivity extends Activity {
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
FMOD.init(this);
setContentView(R.layout.activity_main);
}
public void mFix(View v) {
String path = Environment.getExternalStorageDirectory().getAbsolutePath() + File.separatorChar + "test.wav";
Log.e("java cj5785", path);
switch (v.getId()) {
case R.id.btn_normal:
Log.d("cj5785", "normal");
QQVoice.fix(path, QQVoice.MODE_NORMAL);
break;
case R.id.btn_luoli:
Log.d("cj5785", "luoli");
QQVoice.fix(path, QQVoice.MODE_LUOLI);
break;
case R.id.btn_dashu:
Log.d("cj5785", "dashu");
QQVoice.fix(path, QQVoice.MODE_DASHU);
break;
case R.id.btn_jingsong:
Log.d("cj5785", "jingsong");
QQVoice.fix(path, QQVoice.MODE_JINGSONG);
break;
case R.id.btn_gaoguai:
Log.d("cj5785", "gaoguai");
QQVoice.fix(path, QQVoice.MODE_GAOGUAI);
break;
case R.id.btn_kongling:
Log.d("cj5785", "kongling");
QQVoice.fix(path, QQVoice.MODE_KONGLING);
break;
}
}
protected void onDestroy() {
super.onDestroy();
FMOD.close();
}
}
工具类1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19package org.fmod.example;
public class QQVoice {
//音效的类型
public static final int MODE_NORMAL = 0;
public static final int MODE_LUOLI = 1;
public static final int MODE_DASHU = 2;
public static final int MODE_JINGSONG = 3;
public static final int MODE_GAOGUAI = 4;
public static final int MODE_KONGLING = 5;
public native static void fix(String path, int type);
static {
System.loadLibrary("fmod");
System.loadLibrary("fmodL");
System.loadLibrary("QQVoice");
}
}
native方法实现
1 |
|
mk文件修改
1 | LOCAL_PATH := $(call my-dir) |
jni目录结构
1 | │ Android.mk |