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="回调javahelloFromJava" />

<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="回调javaAdd" />

<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="回调javaPrintString" />

<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代码的编写相关推荐

  1. JNI基础 c语言调用java方法

    利用c语言调用java无参的方法 java方法 com.example.jniparsedata.ParseData类中的方法     //打印     public void sayHelloFro ...

  2. c如何调用java_JNI学习------C语言调用Java (转)

    最近项目中需要使用JNI,所以研究了一下,其中遇到过不少问题,总结一下,让遇到同样问题的人可以得到解决. 在C/C++中调用Java的方法一般分为五个步骤:初始化虚拟机.获取类.获取类的方法.创建类对 ...

  3. 记录一遍易语言调用java

    易语言调用java 易语言有2中方式操作 第一种 如果我们没有安装jvm的情况, 需要通过易语言自带的东西,jvm.创建(,) 第一个参数为你自己的jar 或者.class ,第二个参数就是jre 包 ...

  4. f2812的c语言与标准c语言,F2812中C语言调用汇编函数

    F2812中C语言调用汇编函数 参考资料: (1) SPRU514 ---- TMS320F28x Optimizing C/C++ Compiler User's Guide.pdf; (2) sp ...

  5. c 语言调用java_C语言调用java的方法

    C语言调用java的方法 发布时间:2020-07-01 11:01:25 来源:亿速云 阅读:86 作者:Leah 本篇文章为大家展示了C语言调用java的方法,代码简明扼要并且容易理解,绝对能使你 ...

  6. 易语言操作java窗口,易语言调用JAVA源码

    易语言调用JAVA源码 @510835147.版本 2 .支持库 Javalib .支持库 spec .程序集 窗口程序集1 .程序集变量 jvm, Java虚拟机 .程序集变量 java, Java ...

  7. android支付宝签名生成工具,Android支付宝支付的示例代码

    上一篇,我们已经详细讲解了Android微信支付,今天接着为大家带来支付宝支付,支付宝支付相对微信支付要简单一些,吐槽一下,而且支付宝文档确实比微信的文档好了不少,下面开始讲解支付流程. 1.首先给出 ...

  8. jsb调用java_在JS代码中使用反射调用java代码注意事项(附webview使用方法)(转)...

    本文是推荐使用过jsb.reflection的开发者进行阅读.关于jsb.reflection的说明请参照: 我们在代码编写过程中,通常会需要在js脚本中调用到java代码或者Objective-C的 ...

  9. rn在java中什么意思_[React Native Android 安利系列]RN中使用js调用java代码

    欢迎大家收看react-native-android系列教程,跟着本系列教程学习,可以熟练掌握react-native-android的开发,你值得拥有: 书接上节,我们上节说道,如何控制原生andr ...

最新文章

  1. Android 金钱计算BigDecimal 的使用
  2. vue中使用transition标签底部导航闪烁问题
  3. opencv 人脸识别_Python学习:基于Opencv来快速实现人脸识别(完整版)
  4. robo光线机器人 java_Robocode教程6——用eclipse来写你的机器人!
  5. 【短信插件】短信如何对接74cms_v4.2.66_骑士人才系统
  6. 《ASP.NET 1.1入门经典—— VISUAL C# .NET 2003编程篇》学习笔记和心得 - 第十章
  7. oauth2.0协议流程_正确的工作流程:我应该使用哪个OAuth 2.0流程?
  8. 使用 IntraWeb (35) - TIWJQueryWidget
  9. 谷歌提出深度CNN模型NIMA:帮你挑选清晰且有美感的图片
  10. 小沙的杀球(贪心+模拟)
  11. 【CF-1285E】Delete a Segment(区间处理(并集)----思维)
  12. 华为od与中软外包哪个更好_华为外包,不是OD,OD也烂,呆了8个月。今天离职再见,…...
  13. “新产业50人论坛”之清华龙桂鲁教授:量子信息与创新发展
  14. 零售3.0时代,国民品牌都市丽人一次成功的变革
  15. eclipes的安装与使用
  16. “缺钱”的女人,掩饰不住以下三个特征,别不信
  17. 通过插画理解kubernetes基本概念
  18. js中 var a 和 a 的区别
  19. 机房服务器显示器切换,两台电脑主机,一台显示屏,怎么实现切换!需要什么转换器,大概多少...
  20. 为什么访问亚马逊的网站卡顿?

热门文章

  1. 二进制蚁群算法【源码实现】
  2. OpenCASCADE:Modeling Algorithms模块几何工具之来自约束的曲线和曲面
  3. wxWidgets:wxAuiManagerEvent类用法
  4. boost::spirit模块演示了 AST 的生成,然后将其转储为人类可读的格式
  5. boost::mp11::mp_map_update相关用法的测试程序
  6. boost::hana::keys用法的测试程序
  7. boost::log::static_type_dispatcher用法的测试程序
  8. boost::graph模块实现拉马努金图的周长和直径的测试程序
  9. GDCM:gdcm::EquipmentManufacturer的测试程序
  10. Boost::context模块callcc的分段的测试程序