android111 java中调用c代码

简介:

MainActivity:

复制代码
package com.itheima.helloworld1;

import android.os.Bundle;
import android.app.Activity;
import android.view.View;
import android.widget.Toast;

public class MainActivity extends Activity {

    static{
        //加载打包完毕的so类库
        System.loadLibrary("hello");//hello是Abdroid.mk中指定的名字。
    }
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
    }

    /*##使用jni
    1. 在项目根目录下创建jni文件夹
    2. 在jni文件中创建一个c文件(src是写java代码的)
    3. 在java代码中,创建一个本地方法helloFromC
            public native String helloFromC();
    4. 在jni中定义函数实现这个方法,函数名必须为
            jstring Java_com_itheima_helloworld1_MainActivity_helloFromC(JNIEnv* env, jobject obj)
    5. 返回一个字符串,用c定义一个字符串
            char* cstr = "hello from c";
    6. 把c的字符串转换成java的字符串
            jstring jstr = (*env)->NewStringUTF(env, cstr);
            return jstr;
    7. 在jni中创建Android.mk文件
    8. 在c文件中添加<jni.h>头文件
    9. 在jni文件夹下执行ndk-build.cmd指令( 在磁盘jni文件夹下用cmd命令编译,编译完后在libs下的armeabi文件夹下有一个libhello.so文件,这就是编译后的c代码)
    10. java代码中加载so类库,调用本地方法*/
    
    public void click(View v){
        Toast.makeText(this, helloFromC(), 0).show();
    }
    
    //定义一个本地方法,方法体由c语言实现
    public native String helloFromC();
    
}
复制代码

Android.mk

复制代码
# mk是make的意思
 LOCAL_PATH := $(call my-dir)

    include $(CLEAR_VARS)
    #编译后生成的文件的类库叫什么名字
    LOCAL_MODULE    := hello
    #要编译的c文件
    LOCAL_SRC_FILES := Hello.c

    include $(BUILD_SHARED_LIBRARY)
复制代码

Hello.c

复制代码
#include <stdio.h>
#include <stdlib.h>
#include <jni.h>

//jstring表示string,Java_com_itheima_helloworld1_MainActivity_helloFromC是包名类名方法名
//JNIEnv* env, jobject obj是必须写的参数,env是java虚拟机的内存地址,java代码是运行在虚拟机的而C代码不是运行在虚拟机的,
//env本来就是一个结构体指针,此处* env是一个java运行环境的二级指针,obj是MainActivity对象,C中jobject表示对象。
jstring Java_com_itheima_helloworld1_MainActivity_helloFromC(JNIEnv* env, jobject obj){
    //c语言的字符串不是string,是字符数组,char* cstr或者char cstr[],数组的名字就是第0个元素的地址,
    char* cstr = "hello from c";
    //把C语言的字符串转换成java的字符串
    // jstring     (*NewStringUTF)(JNIEnv*, const char*);
//    jstring jstr = (*(*env)).NewStringUTF(env, cstr);
    jstring jstr = (*env)->NewStringUTF(env, cstr);
    return jstr;
}
复制代码

Application.mk中:APP_ABI := armeabi armeabi-v7a x86 //表示不同的架构

java代码很容易反编译,c代码反编译比较难。

java代码最后使用的是c代码编译后的类库,原C代码已经没用了。



本文转自农夫山泉别墅博客园博客,原文链接:http://www.cnblogs.com/yaowen/p/4986806.html,如需转载请自行联系原作者


相关文章
|
11天前
|
Java 测试技术 应用服务中间件
常见 Java 代码缺陷及规避方式(下)
常见 Java 代码缺陷及规避方式(下)
34 0
|
13天前
|
Java
Java中ReentrantLock释放锁代码解析
Java中ReentrantLock释放锁代码解析
25 8
|
16天前
|
前端开发 小程序 Java
uniapp上传图片 前端以及java后端代码实现
uniapp上传图片 前端以及java后端代码实现
30 0
|
18天前
|
设计模式 存储 Java
23种设计模式,享元模式的概念优缺点以及JAVA代码举例
【4月更文挑战第6天】享元模式(Flyweight Pattern)是一种结构型设计模式,旨在通过共享技术有效地支持大量细粒度对象的重用。这个模式在处理大量对象时非常有用,特别是当这些对象中的许多实例实际上可以共享相同的状态时,从而可以减少内存占用,提高程序效率
31 4
|
18天前
|
设计模式 Java 中间件
23种设计模式,适配器模式的概念优缺点以及JAVA代码举例
【4月更文挑战第6天】适配器模式(Adapter Pattern)是一种结构型设计模式,它的主要目标是让原本由于接口不匹配而不能一起工作的类可以一起工作。适配器模式主要有两种形式:类适配器和对象适配器。类适配器模式通过继承来实现适配,而对象适配器模式则通过组合来实现
30 4
|
19天前
|
存储 缓存 算法
优化 Java 后台代码的关键要点
【4月更文挑战第5天】本文探讨了优化 Java 后台代码的关键点,包括选用合适的数据结构与算法、减少不必要的对象创建、利用 Java 8 新特性、并发与多线程处理、数据库和缓存优化、代码分析与性能调优、避免阻塞调用、JVM 调优以及精简第三方库。通过这些方法,开发者可以提高系统性能、降低资源消耗,提升用户体验并减少运营成本。
|
20天前
|
Java 开发工具 流计算
flink最新master代码编译出现Java Runtime Environment 问题
在尝试编译Flink源码时遇到Java运行时环境致命错误:EXCEPTION_ACCESS_VIOLATION。问题出现在JVM.dll+0x88212。使用的是Java 11.0.28和Java HotSpot(TM) 64-Bit Server VM。系统为Windows客户端,没有生成核心dump文件。错误日志保存在hs_err_pid39364.log和replay_pid39364.log。要解决这个问题,建议检查JDK版本兼容性,更新JDK或参照错误报告文件提交Bug至http://bugreport.java.com/bugreport/crash.jsp。
|
20天前
|
移动开发 Java Android开发
构建高效Android应用:探究Kotlin与Java的性能差异
【4月更文挑战第3天】在移动开发领域,性能优化一直是开发者关注的焦点。随着Kotlin的兴起,其在Android开发中的地位逐渐上升,但关于其与Java在性能方面的对比,尚无明确共识。本文通过深入分析并结合实际测试数据,探讨了Kotlin与Java在Android平台上的性能表现,揭示了在不同场景下两者的差异及其对应用性能的潜在影响,为开发者在选择编程语言时提供参考依据。
|
21天前
|
Java
使用Java代码打印log日志
使用Java代码打印log日志
75 1
|
11天前
|
Java
代码的魔法师:Java反射工厂模式详解
代码的魔法师:Java反射工厂模式详解
24 0