02_JNI中Java代码调用C代码,Android中使用log库打印日志,javah命令的使用,Android.mk文件的编写,交叉编译

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 1  编写以下案例(下面的三个按钮都调用了底层的C语言):项目案例的代码结构如下:2 编写DataProvider的代码: package com.example.ndkpassdata;   public class DataProvider {         /**      * 计算x和y的加法  apktools      *


1  编写以下案例(下面的三个按钮都调用了底层的C语言)

项目案例的代码结构如下:

2 编写DataProvider的代码:

package com.example.ndkpassdata;

 

public class DataProvider {

   

    /**

     * 计算xy的加法  apktools

     *

     * @param x

     * @param y

     * @return

     */

    public native int add(int x,int y);

   

    /**

     * 给字符串后面拼接字符串   加密运算   web   url

     * @param s

     * @return

     */

    public native String sayHelloInC(String s);

   

    /**

     * C代码传递int数组   C代码给这个数组进行操作

     * 图形   声音的处理

     *

     * @param iNum

     * @return

     */

    public native int[] intMethod(int[] iNum);

}

3 MainActivity的代码如下:

package com.example.ndkpassdata;

 

import android.annotation.SuppressLint;

import android.app.Activity;

import android.os.Bundle;

import android.view.View;

import android.widget.Toast;

 

public class MainActivity extends Activity {

         DataProvider provider;

         static{

                   System.loadLibrary("hello");

         }

        

         @Override

         protected void onCreate(Bundle savedInstanceState) {

                   super.onCreate(savedInstanceState);

                   setContentView(R.layout.activity_main);

                   provider=new DataProvider();

         }

        

         @SuppressLint("ShowToast")

         public void click1(View view){

                   int result=provider.add(3, 5);

                   Toast.makeText(getApplicationContext(), "provider.add(3, 5) = " + result, 0).show();

         }

        

         @SuppressLint("ShowToast")

         public void click2(View view){

                   String str=provider.sayHelloInC("yll");

                   Toast.makeText(getApplicationContext(), str, 0).show();

         }

        

         public void click3(View view){

                   int[] arr=new int[]{1,2,3,4,5};

                   provider.intMethod(arr);

                   for(int i:arr){

                            System.out.println(i);

                   }

         }

}

4 通过DataProvider生成C语言的头文件

进入/cygdrive/e/workspace/Android/NdkPassData/src目录下:

查看DataProvider的全路径:

在项目中生成头文件

头文件的内容如下:

/* DO NOT EDIT THIS FILE - it is machine generated */

#include <jni.h>

/* Header for class com_example_ndkpassdata_DataProvider */

 

#ifndef _Included_com_example_ndkpassdata_DataProvider

#define _Included_com_example_ndkpassdata_DataProvider

#ifdef __cplusplus

extern "C" {

#endif

/*

 * Class:     com_example_ndkpassdata_DataProvider

 * Method:    add

 * Signature: (II)I

 */

JNIEXPORT jint JNICALL Java_com_example_ndkpassdata_DataProvider_add

  (JNIEnv *, jobject, jint, jint);

 

/*

 * Class:     com_example_ndkpassdata_DataProvider

 * Method:    sayHelloInC

 * Signature: (Ljava/lang/String;)Ljava/lang/String;

 */

JNIEXPORT jstring JNICALL Java_com_example_ndkpassdata_DataProvider_sayHelloInC

  (JNIEnv *, jobject, jstring);

 

/*

 * Class:     com_example_ndkpassdata_DataProvider

 * Method:    intMethod

 * Signature: ([I)[I

 */

JNIEXPORT jintArray JNICALL Java_com_example_ndkpassdata_DataProvider_intMethod

  (JNIEnv *, jobject, jintArray);

 

#ifdef __cplusplus

}

#endif

#endif

5 在项目下创建一个文件夹jni,将上面生成的头文件拷贝到这个文件夹下:

然后按照头文件,编写hello.c,hello.c的代码如下:

#include <stdio.h>

#include "com_example_ndkpassdata_DataProvider.h"

#include <android/log.h>

#include <string.h>

#define LOG_TAG "clog"

#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)

#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__)

 

/**

 * Java中的字符串转换成为C语言的字符

 */

char*   Jstring2CStr(JNIEnv*   env,   jstring   jstr)

{

     char*   rtn   =   NULL;

     jclass   clsstring   =   (*env)->FindClass(env,"java/lang/String");

     jstring   strencode   =   (*env)->NewStringUTF(env,"GB2312");

     jmethodID   mid   =   (*env)->GetMethodID(env,clsstring,   "getBytes",   "(Ljava/lang/String;)[B");

     jbyteArray   barr=   (jbyteArray)(*env)->CallObjectMethod(env,jstr,mid,strencode); // String .getByte("GB2312");

     jsize   alen   =   (*env)->GetArrayLength(env,barr);

     jbyte*   ba   =   (*env)->GetByteArrayElements(env,barr,JNI_FALSE);

     if(alen   >   0)

     {

      rtn   =   (char*)malloc(alen+1);         //"\0"

      memcpy(rtn,ba,alen);

      rtn[alen]=0;

     }

     (*env)->ReleaseByteArrayElements(env,barr,ba,0);  //

     return rtn;

}

 

JNIEXPORT jint JNICALL Java_com_example_ndkpassdata_DataProvider_add

  (JNIEnv * env, jobject jobject, jint x, jint y){

    // 想在logcat控制台上 打印日志

    LOGD("x=%d",x);

    LOGI("y=%d",y);

    // log.i(TAG,"sss");

    return x+y;

}

 

JNIEXPORT jstring JNICALL Java_com_example_ndkpassdata_DataProvider_sayHelloInC

  (JNIEnv * env, jobject jobject, jstring str){

 

    char* c="hello";

    // C语言中不能直接操作java中的字符串

    // java中的字符串转换成c语言中 char数组

    char* cstr=Jstring2CStr(env,str);

 

    strcat(cstr,c);

    LOGD("%s",cstr);

    return  (*env)->NewStringUTF(env,cstr);

}

 

JNIEXPORT jintArray JNICALL Java_com_example_ndkpassdata_DataProvider_intMethod

  (JNIEnv * env, jobject jobject, jintArray jarray){

    // jArray  遍历数组   jint*       (*GetIntArrayElements)(JNIEnv*, jintArray, jboolean*);

    // 数组的长度    jsize       (*GetArrayLength)(JNIEnv*, jarray);

    // 对数组中每个元素 +5

    int length=(*env)->GetArrayLength(env,jarray);

    int* array=(*env)->GetIntArrayElements(env,jarray,0);

    int i=0;

    for(;i<length;i++){

       *(array+i)+=5;

    }

    return jarray;

}

其中,上面的代码中使用了NDK中打印日志的库,Log共享库库的位置在:

6 编写Android.mk文件,内容如下:

LOCAL_PATH := $(call my-dir)

 

include $(CLEAR_VARS)

 

#测试写了lib前缀的情况,如果不加lib,生成.so的时候回自动加上lib前缀

LOCAL_MODULE    := libhello

LOCAL_SRC_FILES := hello.c

 

#这里使用C语言打印日志,要引用log的共享库

#共享库的位置在:\android-ndk-r9d\platforms\android-19\arch-arm\usr\lib

LOCAL_LDLIBS    += -llog

 

include $(BUILD_SHARED_LIBRARY)

7 使用ndk-build进行交叉编译:

F5刷新项目,发现项目下多了:

8  编写布局文件:

<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" >

 

    <TextView

        android:layout_width="wrap_content"

        android:layout_height="wrap_content"

        android:layout_centerHorizontal="true"

        android:layout_centerVertical="true"

        android:text="@string/hello_world" />

 

    <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="C语言实现第一个方法" />

 

    <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="C语言实现第二个方法" />

 

    <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="C语言实现第三个方法" />

 

</RelativeLayout>

9 编写Android清单文件:

<?xml version="1.0" encoding="utf-8"?>

<manifest xmlns:android="http://schemas.android.com/apk/res/android"

    package="com.example.ndkpassdata"

    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.ndkpassdata.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>

接下来运行应用,就可以看到如下:

同时注意控制台,可以打印出日志信息。

 

 

 

相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
目录
打赏
0
0
0
0
58
分享
相关文章
Android历史版本与APK文件结构
通过以上内容,您可以全面了解Android的历史版本及其主要特性,同时掌握APK文件的结构和各部分的作用。这些知识对于理解Android应用的开发和发布过程非常重要,也有助于在实际开发中进行高效的应用管理和优化。希望这些内容对您的学习和工作有所帮助。
121 83
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
29 7
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
MySQL日志详解——日志分类、二进制日志bin log、回滚日志undo log、重做日志redo log
MySQL日志详解——日志分类、二进制日志bin log、回滚日志undo log、重做日志redo log、原理、写入过程;binlog与redolog区别、update语句的执行流程、两阶段提交、主从复制、三种日志的使用场景;查询日志、慢查询日志、错误日志等其他几类日志
MySQL日志详解——日志分类、二进制日志bin log、回滚日志undo log、重做日志redo log
java连接redis和基础操作命令
通过以上内容,您可以掌握在Java中连接Redis以及进行基础操作的基本方法,进而在实际项目中灵活应用。
84 30
MySQL事务日志-Undo Log工作原理分析
事务的持久性是交由Redo Log来保证,原子性则是交由Undo Log来保证。如果事务中的SQL执行到一半出现错误,需要把前面已经执行过的SQL撤销以达到原子性的目的,这个过程也叫做"回滚",所以Undo Log也叫回滚日志。
MySQL事务日志-Undo Log工作原理分析
什么是Apache日志?为什么Apache日志分析很重要?
Apache是全球广泛使用的Web服务器软件,支持超过30%的活跃网站。它通过接收和处理HTTP请求,与后端服务器通信,返回响应并记录日志,确保网页请求的快速准确处理。Apache日志分为访问日志和错误日志,对提升用户体验、保障安全及优化性能至关重要。EventLog Analyzer等工具可有效管理和分析这些日志,增强Web服务的安全性和可靠性。
什么是事件日志管理系统?事件日志管理系统有哪些用处?
事件日志管理系统是IT安全的重要工具,用于集中收集、分析和解释来自组织IT基础设施各组件的事件日志,如防火墙、路由器、交换机等,帮助提升网络安全、实现主动威胁检测和促进合规性。系统支持多种日志类型,包括Windows事件日志、Syslog日志和应用程序日志,通过实时监测、告警及可视化分析,为企业提供强大的安全保障。然而,实施过程中也面临数据量大、日志管理和分析复杂等挑战。EventLog Analyzer作为一款高效工具,不仅提供实时监测与告警、可视化分析和报告功能,还支持多种合规性报告,帮助企业克服挑战,提升网络安全水平。
115 2
什么是日志管理,如何进行日志管理?
日志管理是对IT系统生成的日志数据进行收集、存储、分析和处理的实践,对维护系统健康、确保安全及获取运营智能至关重要。本文介绍了日志管理的基本概念、常见挑战、工具的主要功能及选择解决方案的方法,强调了定义管理目标、日志收集与分析、警报和报告、持续改进等关键步骤,以及如何应对数据量大、安全问题、警报疲劳等挑战,最终实现日志数据的有效管理和利用。
348 0
java Runtime.exec()执行shell/cmd命令:常见的几种陷阱与一种完善实现
java Runtime.exec()执行shell/cmd命令:常见的几种陷阱与一种完善实现
86 1
【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板
本文介绍了Java日志框架的基本概念和使用方法,重点讨论了SLF4J、Log4j、Logback和Log4j2之间的关系及其性能对比。SLF4J作为一个日志抽象层,允许开发者使用统一的日志接口,而Log4j、Logback和Log4j2则是具体的日志实现框架。Log4j2在性能上优于Logback,推荐在新项目中使用。文章还详细说明了如何在Spring Boot项目中配置Log4j2和Logback,以及如何使用Lombok简化日志记录。最后,提供了一些日志配置的最佳实践,包括滚动日志、统一日志格式和提高日志性能的方法。
954 31
【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板

热门文章

最新文章

  • 1
    如何修复 Android 和 Windows 不支持视频编解码器的问题?
    56
  • 2
    Android历史版本与APK文件结构
    121
  • 3
    【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
    38
  • 4
    【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
    22
  • 5
    当flutter react native 等混开框架-并且用vscode-idea等编译器无法打包apk,打包安卓不成功怎么办-直接用android studio如何打包安卓apk -重要-优雅草卓伊凡
    5
  • 6
    APP-国内主流安卓商店-应用市场-鸿蒙商店上架之必备前提·全国公安安全信息评估报告如何申请-需要安全评估报告的资料是哪些-优雅草卓伊凡全程操作
    11
  • 7
    【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
    10
  • 8
    【03】仿站技术之python技术,看完学会再也不用去购买收费工具了-修改整体页面做好安卓下载发给客户-并且开始提交网站公安备案-作为APP下载落地页文娱产品一定要备案-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
    34
  • 9
    Android经典面试题之Kotlin中Lambda表达式和匿名函数的区别
    1
  • 10
    【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
    29
  • AI助理

    你好,我是AI助理

    可以解答问题、推荐解决方案等