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

 

 

目录
相关文章
|
3月前
|
Java
在 Java 中捕获和处理自定义异常的代码示例
本文提供了一个 Java 代码示例,展示了如何捕获和处理自定义异常。通过创建自定义异常类并使用 try-catch 语句,可以更灵活地处理程序中的错误情况。
117 1
|
15天前
|
JavaScript NoSQL Java
接替此文【下篇-服务端+后台管理】优雅草蜻蜓z系统JAVA版暗影版为例-【蜻蜓z系列通用】-2025年全新项目整合搭建方式-这是独立吃透代码以后首次改变-独立PC版本vue版搭建教程-优雅草卓伊凡
接替此文【下篇-服务端+后台管理】优雅草蜻蜓z系统JAVA版暗影版为例-【蜻蜓z系列通用】-2025年全新项目整合搭建方式-这是独立吃透代码以后首次改变-独立PC版本vue版搭建教程-优雅草卓伊凡
160 96
接替此文【下篇-服务端+后台管理】优雅草蜻蜓z系统JAVA版暗影版为例-【蜻蜓z系列通用】-2025年全新项目整合搭建方式-这是独立吃透代码以后首次改变-独立PC版本vue版搭建教程-优雅草卓伊凡
|
1月前
|
缓存 前端开发 Android开发
【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
89 12
【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
|
1月前
|
SQL Java 数据库连接
如何在 Java 代码中使用 JSqlParser 解析复杂的 SQL 语句?
大家好,我是 V 哥。JSqlParser 是一个用于解析 SQL 语句的 Java 库,可将 SQL 解析为 Java 对象树,支持多种 SQL 类型(如 `SELECT`、`INSERT` 等)。它适用于 SQL 分析、修改、生成和验证等场景。通过 Maven 或 Gradle 安装后,可以方便地在 Java 代码中使用。
325 11
|
1月前
|
测试技术 Android开发 开发者
【03】优雅草央千澈详解关于APP签名以及分发-上架完整流程-第三篇安卓APP上架华为商店后面的步骤-华为应用商店相对比较麻烦一些-华为商店安卓上架
【03】优雅草央千澈详解关于APP签名以及分发-上架完整流程-第三篇安卓APP上架华为商店后面的步骤-华为应用商店相对比较麻烦一些-华为商店安卓上架
57 16
|
1月前
|
JSON Java 数据挖掘
利用 Java 代码获取淘宝关键字 API 接口
在数字化商业时代,精准把握市场动态与消费者需求是企业成功的关键。淘宝作为中国最大的电商平台之一,其海量数据中蕴含丰富的商业洞察。本文介绍如何通过Java代码高效、合规地获取淘宝关键字API接口数据,帮助商家优化产品布局、制定营销策略。主要内容包括: 1. **淘宝关键字API的价值**:洞察用户需求、优化产品标题与详情、制定营销策略。 2. **获取API接口的步骤**:注册账号、申请权限、搭建Java开发环境、编写调用代码、解析响应数据。 3. **注意事项**:遵守法律法规与平台规则,处理API调用限制。 通过这些步骤,商家可以在激烈的市场竞争中脱颖而出。
|
3月前
|
Java
在Java中实现接口的具体代码示例
可以根据具体的需求,创建更多的类来实现这个接口,以满足不同形状的计算需求。希望这个示例对你理解在 Java 中如何实现接口有所帮助。
119 38
|
2月前
|
安全 Java 编译器
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
80 3
|
3月前
|
Java
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
77 24
|
2月前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
79 2

热门文章

最新文章

  • 1
    【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
  • 2
    Android历史版本与APK文件结构
  • 3
    【01】噩梦终结flutter配安卓android鸿蒙harmonyOS 以及next调试环境配鸿蒙和ios真机调试环境-flutter项目安卓环境配置-gradle-agp-ndkVersion模拟器运行真机测试环境-本地环境搭建-如何快速搭建android本地运行环境-优雅草卓伊凡-很多人在这步就被难倒了
  • 4
    当flutter react native 等混开框架-并且用vscode-idea等编译器无法打包apk,打包安卓不成功怎么办-直接用android studio如何打包安卓apk -重要-优雅草卓伊凡
  • 5
    APP-国内主流安卓商店-应用市场-鸿蒙商店上架之必备前提·全国公安安全信息评估报告如何申请-需要安全评估报告的资料是哪些-优雅草卓伊凡全程操作
  • 6
    【03】仿站技术之python技术,看完学会再也不用去购买收费工具了-修改整体页面做好安卓下载发给客户-并且开始提交网站公安备案-作为APP下载落地页文娱产品一定要备案-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
  • 7
    【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
  • 8
    【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
  • 9
    escrcpy:【技术党必看】Android开发,Escrcpy 让你无线投屏新体验!图形界面掌控 Android,30-120fps 超流畅!🔥
  • 10
    Cellebrite UFED 4PC 7.71 (Windows) - Android 和 iOS 移动设备取证软件