NDK
NDK介绍
定义:Native Development Kit,是 Android的一个工具开发包
NDK是属于 Android 的,与Java并无直接关系
作用:快速开发C、 C++的动态库,并自动将so和应用一起打包成 APK。
可通过 NDK在 Android中使用 JNI与本地代码(如C、C++)交互
步骤
1, 配置 Android NDK环境
A, Android 原生开发工具包:就是ndk工具包。
B, LLDB:一种调试程序,Android Studio 使用它来调试原生代码。
C, 下载这些组件:
D, 在打开的Android Studio项目中,从菜单栏选择 Tools > Android > SDK Manager。
E, 点击 SDK Tools 标签。
F, 选中 LLDB、CMake 和 NDK 旁的复选框
下载NDK
标题
创建 JNI 文件夹
直接在项目右键,选择 New - Folder
JNI Folder ,对话框直接点击 Finish 即可方便地在默认位置创建 jni 文件夹用于存放 c 源码。默认位置在 app/src/main/jni.
创建 Java 类
首先创建一个 Java 类用于调用 c 代码。
// An highlighted block
public class JniTest {
static {
System.loadLibrary("JniLib");
}
public static native String getString();
}
生成头文件 (.h) 命令行
最直接的方式就是通过命令行生成。
首先使用 javac 编译 java 文件。
小技巧。使用右键按住拖动文件夹到终端面板,可以快速进入对应目录。
进入到文件所在目录后执行 javac JniTest.java 编译。成功后会出现 JniTest.class 文件。
然后退回到包外目录执行 javah -jni 生成头文件。
注意路径不要写错了,最后也不需要加文件扩展名。成功后会生成一个 .h 文件,把它手动移到 jni 目录。之前编译出的 .class 文件可以删掉了。最终目录结构如下:
生成.h文件可以配置工具.。
先起个名字叫,这里叫做 javah.
Program: JDKPath JDKPathJDKPath\bin\javah.exe
Arguments: -classpath
. -jni -d ModuleFileDir ModuleFileDirModuleFileDir\src\main\jni FileClass FileClassFileClass
Working directory: ModuleFileDir ModuleFileDirModuleFileDir\src\main\Java
点击 OK 保存后就新建了一个工具。此时我们右击 JniTest.java,在菜单中选择 External Tools - javah 就可以快速生成头文件并放到
jni 目录。
编写 c 代码
在 jni 目录新建 一个 c 语言源码,这里叫做 JinLib.cpp. 然后实现头文件中所定义的函数,别忘引入头文件。这里简单地返回一个字符串:
// An highlighted block
// 头文件 可能不同
#include <cc_chenhe_ndkdemo_JniTest.h>
/*
-
Class:
cc_chenhe_ndkdemo_JniTest -
Method:
getString * Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_cc_chenhe_ndkdemo_JniTest_getString
(JNIEnv * env, jclass){
return (*env).NewStringUTF("Hello cpp");
}
创建 mk 文件
mk 文件用于告诉 ndk-build 该如何编译 c 源码,详情见官方指南。
在 jni 目录下创建 Android.mk:
// An highlighted block
LOCAL_PATH
:= $(call
my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE
:= JniLib
LOCAL_SRC_FILES
=: JniLib.cpp
include $(BUILD_SHARED_LIBRARY)
其中 LOCAL_SRC_FILES 列出了所有要编译的 c 源码文件。
然后创建 Application.mk:
// An highlighted block
APP_MODULES
:= JniLib
APP_ABI :=
all
gradle 配置:
在 module 的 build.gradle 里,android.defaultConfig 下加入下面配置:
// An highlighted block
ndk{
moduleName "JniLib"
//abiFilters
“armeabi-v7a”, “x86” //输出指定abi下的so库
}
sourceSets.main{
jni.srcDirs = []
jniLibs.srcDir "src/main/libs"
}
编译:
首先配置环境,在Path里设置路径,可以直接使用ndk-build。
在cmd里进入jni目录里,使用ndk-build编译.
运行:添加 Activity 的代码
// An highlighted block
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
TextView tv =
findViewById(R.id.text_view);
tv.setText(JniTest.getString());
}
}‘
java调用c实现计算机
// An highlighted block
package com.example.javaandc;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
import android.widget.Toast;
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
// Used to load the 'native-lib' library on application startup.
static {
System.loadLibrary("native-lib");
}
private TextView result;
private EditText numa;
private EditText numb;
private Button plus;
private Button subtract;
private Button ride;
private Button divide;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// Example of a call to a native method
initView();
}
private void initView() {
result = findViewById(R.id.main_result);
numa = findViewById(R.id.main_numa);
numb = findViewById(R.id.main_numb);
plus = findViewById(R.id.main_plus);
subtract = findViewById(R.id.main_subtract);
ride = findViewById(R.id.main_ride);
divide = findViewById(R.id.main_divide);
plus.setOnClickListener(this);
subtract.setOnClickListener(this);
ride.setOnClickListener(this);
divide.setOnClickListener(this);
}
/**
* A native method that is implemented by the 'native-lib' native library,
* which is packaged with this application.
*/
public native String stringFromJNI();
public native int add(int a,int b);
public native int subtract(int a,int b);
public native int ride(int a,int b);
public native int divide(int a,int b);
@Override
public void onClick(View v) {
String string1 = numa.getText().toString();
String string2 = numb.getText().toString();
int a = Integer.parseInt(string1);
int b = Integer.parseInt(string2);
switch (v.getId()){
case R.id.main_plus:
int add = add(a, b);
result.setText(add+"");
break;
case R.id.main_subtract:
int subtract = subtract(a, b);
result.setText(subtract+"");
break;
case R.id.main_ride:
int ride = ride(a, b);
result.setText(ride+"");
break;
case R.id.main_divide:
if(a == 0){
Toast.makeText(this, "零不能为除数", Toast.LENGTH_SHORT).show();
}
int divide = divide(a, b);
result.setText(divide+"");
break;
}
}
}
// An highlighted block
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".MainActivity">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
>
<EditText
android:id="@+id/main_numa"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"/>
<EditText
android:id="@+id/main_numb"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"/>
</LinearLayout>
<LinearLayout
android:layout_marginTop="20dp"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
>
<Button
android:id="@+id/main_plus"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="加"
/>
<Button
android:id="@+id/main_subtract"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="减"
/>
<Button
android:id="@+id/main_ride"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="乘"
/>
<Button
android:id="@+id/main_divide"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="除"
/>
</LinearLayout>
<TextView
android:layout_marginTop="20dp"
android:id="@+id/main_result"
android:gravity="center"
android:padding="10dp"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
</LinearLayout>