03_Android NDK中C语言调用Java代码,javah的使用,javap的使用以及生成签名,Android.mk的编写,C代码的编写
1 案例场景,通过C语言回调Java的代码,案例的最终界面:
2 案例的代码结构如下:
3 编写DataProvider的代码:
package com.example.ndkcallback; public class DataProvider { //C调用java空方法 public void helloFromJava(){ System.out.println("哈哈哈 我被调用了"); } //C调用java中的带两个int参数的方法 public int Add(int x,int y){ int result=x+y; System.out.println("result:"+result); return result; } //C调用java中参数为string的方法 public void printString(String s) { System.out.println(s); } public static void demo(){ System.out.println("哈哈哈,我是静态方法"); } public native void callMethod1(); public native void callMethod2(); public native void callMethod3(); public native void callMethod4(); public native void callMethod5(); } |
4 通过DataProvider获得头文件
接着带有header的结构如下:
5 编写MainActivity代码:
package com.example.ndkcallback; import android.os.Bundle; import android.support.v7.app.ActionBarActivity; import android.view.View; public class MainActivity extends ActionBarActivity { DataProvider dp; static { System.loadLibrary("hello"); } public void helloFromJava() { System.out.print("哈哈哈,我被调用了"); } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); dp = new DataProvider(); } public void click1(View view){ dp.callMethod1(); } public void click2(View view){ dp.callMethod2(); } public void click3(View view){ dp.callMethod3(); } public void click4(View view){ dp.callMethod4(); } public void click5(View view){ dp.callMethod5(); } } |
6 编写布局文件activity_main.xml:
<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" tools:context=".MainActivity" > <Button android:onClick="click1" android:id="@+id/button1" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentLeft="true" android:layout_alignParentTop="true" android:text="回调java中helloFromJava" /> <Button android:onClick="click2" android:id="@+id/button2" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentLeft="true" android:layout_below="@+id/button1" android:text="回调java中Add" /> <Button android:onClick="click3" android:id="@+id/button3" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentLeft="true" android:layout_below="@+id/button2" android:text="回调java中PrintString" /> <Button android:onClick="click4" android:id="@+id/button4" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentLeft="true" android:layout_below="@+id/button3" android:text="回调java中其它类的方法" /> <Button android:onClick="click5" android:id="@+id/button5" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentLeft="true" android:layout_below="@+id/button4" android:text="回调java中静态的方法" /> </RelativeLayout> |
7 接下来,通过javap命令获得DataProvider的方法签名,在cygwin上进入/NdkCallBack/bin/classes.
命令如下:
toto@toto-PC /cygdrive/e/workspace/Android/NdkCallBack/bin/classes
$ javap -scom.example.ndkcallback.DataProvider
上面红框圈的分别是执行命令和方法签名。这个签名在hello.c中会用到。
7 接下来编写hello.c
#include "com_example_ndkcallback_DataProvider.h" /** * 调用:DataProvider中的 public void helloFromJava(); */ JNIEXPORT void JNICALL Java_com_example_ndkcallback_DataProvider_callMethod1 (JNIEnv *env, jobject jobject) { /** * Class<?> forName = Class.forName("com.example.ndkcallback.DataProvider"); * Method declaredMethod = forName.getDeclaredMethod("helloFromJava", new Class[]{}); * declaredMethod.invoke(forName.newInstance(), new Object[]{}); */ //jclass (*FindClass)(JNIEnv*, const char*); jclass clazz = (*env)->FindClass(env,"com/example/ndkcallback/DataProvider"); //jmethodID (*GetMethodID)(JNIEnv*,jclass,const char*,const char*) //方法签名 参数和返回值 //GetMethodID中参数分别是env指针,class,方法名,方法签名 jmethodID methodId=(*env)->GetMethodID(env,clazz,"helloFromJava","()V"); //通过 void (*CallVoidMethod)(JNIEnv*, jobject, jmethodID, ...);来调用Java的代码 (*env)->CallVoidMethod(env,jobject,methodId); } /** * 调用:DataProvider中的 public int Add(int, int); */ JNIEXPORT void JNICALL Java_com_example_ndkcallback_DataProvider_callMethod2 (JNIEnv *env, jobject jobject) { jclass clazz=(*env)->FindClass(env,"com/example/ndkcallback/DataProvider"); //这里的方法签名中有几个I表示有几个参数 jmethodID methodId=(*env)->GetMethodID(env,clazz,"Add","(II)I"); // jint (*CallIntMethod)(JNIEnv*, jobject, jmethodID, ...); (*env)->CallIntMethod(env,jobject,methodId,3,5); } /** * 调用:DataProvider中的 public void printString(java.lang.String); */ JNIEXPORT void JNICALL Java_com_example_ndkcallback_DataProvider_callMethod3 (JNIEnv *env, jobject jobject) { // 参数 object 就是native方法所在的类 jclass clazz=(*env)->FindClass(env,"com/example/ndkcallback/DataProvider"); jmethodID methodId=(*env)->GetMethodID(env,clazz,"printString","(Ljava/lang/String;)V"); // jint (*CallIntMethod)(JNIEnv*, jobject, jmethodID, ...); jstring str=(*env)->NewStringUTF(env,"hello"); (*env)->CallVoidMethod(env,jobject,methodId,str); } JNIEXPORT void JNICALL Java_com_example_ndkcallback_DataProvider_callMethod4 (JNIEnv * env, jobject j){ jclass clazz=(*env)->FindClass(env,"com/example/ndkcallback/MainActivity"); // jmethodID (*GetMethodID)(JNIEnv*, jclass, const char*, const char*); // 方法签名 参数和返回值 jmethodID methodId=(*env)->GetMethodID(env,clazz,"helloFromJava","()V"); // void (*CallVoidMethod)(JNIEnv*, jobject, jmethodID, ...); // 需要创建DataProvider的 对象 // jobject (*AllocObject)(JNIEnv*, jclass); jobject obj=(*env)->AllocObject(env,clazz); // new MainActivity(); (*env)->CallVoidMethod(env,obj,methodId); } JNIEXPORT void JNICALL Java_com_example_ndkcallback_DataProvider_callMethod5 (JNIEnv * env, jobject j){ jclass clazz=(*env)->FindClass(env,"com/example/ndkcallback/DataProvider"); // jmethodID (*GetStaticMethodID)(JNIEnv*, jclass, const char*, const char*); jmethodID methodid=(*env)->GetStaticMethodID(env,clazz,"demo","()V"); //void (*CallStaticVoidMethod)(JNIEnv*, jclass, jmethodID, ...); (*env)->CallStaticVoidMethod(env,clazz,methodid); } |
8 编写Android.mk文件,内容如下:
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_MODULE := libhello LOCAL_SRC_FILES := Hello.c include $(BUILD_SHARED_LIBRARY) |
9 编写Android的清单文件,内容如下:
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.ndkcallback" android:versionCode="1" android:versionName="1.0" > <uses-sdk android:minSdkVersion="8" android:targetSdkVersion="19" /> <application android:allowBackup="true" android:icon="@drawable/ic_launcher" android:label="@string/app_name" android:theme="@style/AppTheme" > <activity android:name="com.example.ndkcallback.MainActivity" android:label="@string/app_name" > <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application> </manifest> |
10 交叉编译,生成.so文件
成功之后,查看Android控制台打印的结果
03_Android NDK中C语言调用Java代码,javah的使用,javap的使用以及生成签名,Android.mk的编写,C代码的编写相关推荐
- JNI基础 c语言调用java方法
利用c语言调用java无参的方法 java方法 com.example.jniparsedata.ParseData类中的方法 //打印 public void sayHelloFro ...
- c如何调用java_JNI学习------C语言调用Java (转)
最近项目中需要使用JNI,所以研究了一下,其中遇到过不少问题,总结一下,让遇到同样问题的人可以得到解决. 在C/C++中调用Java的方法一般分为五个步骤:初始化虚拟机.获取类.获取类的方法.创建类对 ...
- 记录一遍易语言调用java
易语言调用java 易语言有2中方式操作 第一种 如果我们没有安装jvm的情况, 需要通过易语言自带的东西,jvm.创建(,) 第一个参数为你自己的jar 或者.class ,第二个参数就是jre 包 ...
- f2812的c语言与标准c语言,F2812中C语言调用汇编函数
F2812中C语言调用汇编函数 参考资料: (1) SPRU514 ---- TMS320F28x Optimizing C/C++ Compiler User's Guide.pdf; (2) sp ...
- c 语言调用java_C语言调用java的方法
C语言调用java的方法 发布时间:2020-07-01 11:01:25 来源:亿速云 阅读:86 作者:Leah 本篇文章为大家展示了C语言调用java的方法,代码简明扼要并且容易理解,绝对能使你 ...
- 易语言操作java窗口,易语言调用JAVA源码
易语言调用JAVA源码 @510835147.版本 2 .支持库 Javalib .支持库 spec .程序集 窗口程序集1 .程序集变量 jvm, Java虚拟机 .程序集变量 java, Java ...
- android支付宝签名生成工具,Android支付宝支付的示例代码
上一篇,我们已经详细讲解了Android微信支付,今天接着为大家带来支付宝支付,支付宝支付相对微信支付要简单一些,吐槽一下,而且支付宝文档确实比微信的文档好了不少,下面开始讲解支付流程. 1.首先给出 ...
- jsb调用java_在JS代码中使用反射调用java代码注意事项(附webview使用方法)(转)...
本文是推荐使用过jsb.reflection的开发者进行阅读.关于jsb.reflection的说明请参照: 我们在代码编写过程中,通常会需要在js脚本中调用到java代码或者Objective-C的 ...
- rn在java中什么意思_[React Native Android 安利系列]RN中使用js调用java代码
欢迎大家收看react-native-android系列教程,跟着本系列教程学习,可以熟练掌握react-native-android的开发,你值得拥有: 书接上节,我们上节说道,如何控制原生andr ...
最新文章
- Android 金钱计算BigDecimal 的使用
- vue中使用transition标签底部导航闪烁问题
- opencv 人脸识别_Python学习:基于Opencv来快速实现人脸识别(完整版)
- robo光线机器人 java_Robocode教程6——用eclipse来写你的机器人!
- 【短信插件】短信如何对接74cms_v4.2.66_骑士人才系统
- 《ASP.NET 1.1入门经典—— VISUAL C# .NET 2003编程篇》学习笔记和心得 - 第十章
- oauth2.0协议流程_正确的工作流程:我应该使用哪个OAuth 2.0流程?
- 使用 IntraWeb (35) - TIWJQueryWidget
- 谷歌提出深度CNN模型NIMA:帮你挑选清晰且有美感的图片
- 小沙的杀球(贪心+模拟)
- 【CF-1285E】Delete a Segment(区间处理(并集)----思维)
- 华为od与中软外包哪个更好_华为外包,不是OD,OD也烂,呆了8个月。今天离职再见,…...
- “新产业50人论坛”之清华龙桂鲁教授:量子信息与创新发展
- 零售3.0时代,国民品牌都市丽人一次成功的变革
- eclipes的安装与使用
- “缺钱”的女人,掩饰不住以下三个特征,别不信
- 通过插画理解kubernetes基本概念
- js中 var a 和 a 的区别
- 机房服务器显示器切换,两台电脑主机,一台显示屏,怎么实现切换!需要什么转换器,大概多少...
- 为什么访问亚马逊的网站卡顿?
热门文章
- 二进制蚁群算法【源码实现】
- OpenCASCADE:Modeling Algorithms模块几何工具之来自约束的曲线和曲面
- wxWidgets:wxAuiManagerEvent类用法
- boost::spirit模块演示了 AST 的生成,然后将其转储为人类可读的格式
- boost::mp11::mp_map_update相关用法的测试程序
- boost::hana::keys用法的测试程序
- boost::log::static_type_dispatcher用法的测试程序
- boost::graph模块实现拉马努金图的周长和直径的测试程序
- GDCM:gdcm::EquipmentManufacturer的测试程序
- Boost::context模块callcc的分段的测试程序