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 helloFromJav


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控制台打印的结果

 

 

目录
相关文章
|
1月前
|
移动开发 Java Android开发
构建高效Android应用:探究Kotlin与Java的性能差异
【4月更文挑战第3天】在移动开发领域,性能优化一直是开发者关注的焦点。随着Kotlin的兴起,其在Android开发中的地位逐渐上升,但关于其与Java在性能方面的对比,尚无明确共识。本文通过深入分析并结合实际测试数据,探讨了Kotlin与Java在Android平台上的性能表现,揭示了在不同场景下两者的差异及其对应用性能的潜在影响,为开发者在选择编程语言时提供参考依据。
|
2天前
|
Java API 开发工具
java与Android开发入门指南
java与Android开发入门指南
8 0
|
6天前
|
移动开发 Java Android开发
构建高效Android应用:探究Kotlin与Java的性能对比
【5月更文挑战第4天】在移动开发的世界中,性能一直是衡量应用质量的重要指标。随着Kotlin的兴起,许多Android开发者开始考虑是否应该从传统的Java迁移到Kotlin。本文通过深入分析两者在Android平台上的性能差异,帮助开发者理解Kotlin在实际项目中的表现,并提供选择编程语言时的参考依据。
20 5
|
9天前
|
Java 编译器 Android开发
构建高效Android应用:探究Kotlin与Java的性能差异
【5月更文挑战第1天】 在移动开发的世界中,性能优化始终是开发者关注的焦点。随着Kotlin的兴起,许多团队和开发者面临着一个选择:是坚持传统的Java语言,还是转向现代化、更加简洁的Kotlin?本文通过深入分析和对比Kotlin与Java在Android应用开发中的性能表现,揭示两者在编译效率、运行速度和内存消耗等方面的差异。我们将探讨如何根据项目需求和团队熟悉度,选择最适合的语言,以确保应用的高性能和流畅体验。
|
10天前
|
Java 编译器 Android开发
构建高效Android应用:探究Kotlin与Java的性能差异
【4月更文挑战第30天】在Android开发领域,Kotlin作为一种现代化的编程语言,因其简洁性和功能性受到了开发者的广泛欢迎。尽管与传统的Java相比,Kotlin提供了诸多便利,但关于其性能表现的讨论始终未息。本文将深入分析Kotlin和Java在Android平台上的性能差异,通过实际测试数据揭示两种语言在编译效率、运行速度以及内存占用方面的具体表现,并探讨如何利用Kotlin的优势来提升Android应用的整体性能。
|
11天前
|
传感器 算法 C语言
C语言在嵌入式系统开发中的优化策略与代码实现
C语言在嵌入式系统开发中的优化策略与代码实现
25 1
|
14天前
|
Java 测试技术 Android开发
构建高效Android应用:探究Kotlin与Java的性能对比
【4月更文挑战第26天】 在移动开发领域,性能优化一直是开发者追求的重要目标。随着Kotlin的兴起,其在Android平台上的应用逐渐增多,但关于Kotlin与Java在性能方面的对比,社区中仍存在诸多讨论。本文通过实际的性能测试,分析比较了使用Kotlin和Java编写的Android应用在多个维度上的运行效率,旨在为开发者提供一个明确的性能参考,帮助他们在选择编程语言时做出更加明智的决策。
|
15天前
|
存储 算法 C语言
C语言进阶:顺序表(数据结构基础) (以通讯录项目为代码练习)
C语言进阶:顺序表(数据结构基础) (以通讯录项目为代码练习)
|
15天前
|
编译器 Linux C语言
C语言:预处理详解(知识点和代码演示)
C语言:预处理详解(知识点和代码演示)
|
15天前
|
C语言
换硬币问题(C语言代码练习)
换硬币问题(C语言代码练习)