[Java C++] JNI开发

JNI(Java Native Interface)是 Java 提供的一种编程桥梁,它允许 Java 代码和本地(Native)代码进行交互。通过 JNI,Java 程序可以调用本地语言(如C、C++)编写的代码,并且本地代码也可以调用 Java 方法。

JNI 的主要作用包括:

  1. 调用本地库:Java 可以调用本地库中的函数,实现对底层系统的访问和控制。
  2. 提高性能:通过 JNI,可以使用本地语言编写高性能的代码,例如对计算密集型任务进行优化。
  3. 平台特定功能:JNI 可以用于实现与特定平台相关的功能,比如访问操作系统特定的功能或者硬件设备。

在 JNI 中,主要涉及到以下几个方面的内容:

  • Java 本地方法接口声明:在 Java 代码中声明 native 方法,并使用关键字 native 来标识这些方法。
  • 本地方法库实现:使用 C、C++ 或者其他本地语言编写与 Java 本地方法相对应的本地方法实现。
  • 编译和链接:将本地方法实现编译成动态链接库,并确保 Java 虚拟机能够加载和调用这些库中的方法。
  • 调用本地方法:在 Java 代码中通过 JNI 调用本地方法,实现 Java 与本地代码的交互。

大致步骤:

  1. 编写 C++ 代码:首先,你需要编写带有 JNI 方法的 C++ 代码。在 C++ 中,你可以使用 JNI 提供的函数来与 Java 交互。确保你的 C++ 代码中包含了 JNI 函数,并且实现了与 Java 交互所需的功能。

  2. 生成动态链接库(DLL 或 SO 文件):将你的 C++ 代码编译成动态链接库文件(在 Windows 下通常是 DLL 文件,在类 Unix 系统下是 SO 文件)。这个库文件将被 Java 加载并调用其中的函数。

  3. 创建 Java 类:在 Java 中声明 native 方法,并加载动态链接库。在 Java 类中,你需要使用关键字 native 声明方法,以便 JVM 知道这些方法的实现在本地库中。

  4. 生成头文件:使用 javah 工具生成与 Java 类对应的头文件,这个头文件包含了 JNI 接口函数的声明,以便在 C++ 代码中引用这些函数。

  5. 实现 JNI 方法:在 C++ 中实现 JNI 方法,这些方法与 Java 中的 native 方法相对应。在 JNI 方法中,你可以通过 JNI 函数和数据结构与 Java 代码进行交互。

  6. 编译和链接:将 Java 类编译成字节码文件,然后使用 JNI 提供的函数将 Java 代码与本地库链接起来。

  7. 运行 Java 代码:运行 Java 代码,在 Java 代码中调用 native 方法,触发 JNI 调用 C++ 代码的过程。

1.windows环境JNI开发

动态链接库与编译器要对应目标操作系统和架构,对于 64 位的 Windows 操作系统,你需要使用 64 位的编译工具链来生成相应的 64 位 DLL 文件。

  • 将C/C++实现的方法用native关键字声明

  • 用静态代码块进行动态链接库加载

public class JNIDemo {
    public static void main(String[] args) {
        System.out.println(add(100,200));
    }

    public static native int add(int a,int b);

    static {
        System.loadLibrary("JNIDemo");
    }
}
  • javac -h ./ JNIDemo.java生成头文件

  • 根据.h文件里的声明,创建.cpp文件实现对应的函数
#include "JNIDemo.h"
#include <iostream>
JNIEXPORT jint JNICALL Java_JNIDemo_add(JNIEnv *, jclass, jint a, jint b){
    std::cout << "a = "<< a << std::endl;
    std::cout << "b = "<< b << std::endl;
    std::cout << "CPP算法调用成功:" << std::endl;
    return a + b;
}
  • 生成动态库
g++ -o jnidemo.dll -fPIC -shared -I"D:\JAVA\jdk1.8.0\include\win32" -I"D:\JAVA\jdk1.8.0\include" JNIDemo.cpp

运行java文件

2.Android系统JNI开发

在 Android 开发中,直接使用 Java 语言引入 Windows 平台上的 DLL 库是行不通的。这是因为 Android 运行在基于 Linux 内核的移动设备上,并不支持 Windows 上的 DLL 文件。此外,Android 使用的是基于 Java 的 Dalvik 虚拟机(现在是 ART 虚拟机),而不是标准的 Java 虚拟机。

如果想在 Android 应用中使用本地库,可以通过使用 Android NDK 来编写 C/C++ 代码,并将其编译为适用于 Android 平台的共享库(.so 文件)。然后,可以使用 JNI(Java Native Interface)来在 Java 代码中调用这些本地库。

在 Android 开发中,需要使用 Android NDK 提供的工具链(例如 ndk-build 或 CMake)来编译和构建你的 C/C++ 代码,生成适用于 Android 平台的 .so 文件。然后,在 Java 代码中使用 JNI 来加载和调用这些本地库。

1.新建JNI工程

2.下载JNI开发所需工具包

NDK:Native Development Kit(本地开发工具),一系列工具的集合,这套工具允许你在Android开发中使用C和C++代码。

CMake:跨平台编译工具。

LLDB:一种调试程序,ANDROID STUDIO使用它来调试原生代码。

3.切换到project选项,编写C++算法接口函数

  • 声明接口函数

  • alt+enter键,在native-lib.cpp中生成该接口函数的实现体:

该接口函数的实现与cpp并不完全相同,如jint表示整型变量,jintArray表示整形数组,因为接口函数是通过JNI编码实现的,JNI与java语言和C语言都能进行交互。

4.构建如下cmake工程

  • signalfeature.h代码如下
#ifndef MACHINELEARNING_SIGNALFEATURE_H
#define MACHINELEARNING_SIGNALFEATURE_H
#include <vector>
#include <cmath>
class SignalFeature {
public:
    static double calculateMean(const double* data, int length);//均值
};
#endif //MACHINELEARNING_SIGNALFEATURE_H
  • signalfeature.cpp
#include "../includes/signalfeature.h"

// 计算均值
double SignalFeature::calculateMean(const double* data, int length) {
    double sum = 0.0;
    for (int i = 0; i < length; i++) {
        sum += data[i];
    }
    return sum / length;
}
  •  内层cmake
cmake_minimum_required(VERSION 3.4.1)
ADD_LIBRARY(signalfeature SHARED signalfeature.cpp)
  • 外层cmake
# For more information about using CMake with Android Studio, read the
# documentation: https://d.android.com/studio/projects/add-native-code.html

# Sets the minimum version of CMake required to build the native library.

cmake_minimum_required(VERSION 3.4.1)

# Creates and names a library, sets it as either STATIC
# or SHARED, and provides the relative paths to its source code.
# You can define multiple libraries, and CMake builds them for you.
# Gradle automatically packages shared libraries with your APK.

add_subdirectory(thirdlib)    # 添加子目录
include_directories(includes) # 头文件搜索路径

add_library( # Sets the name of the library.
             native-lib

             # Sets the library as a shared library.
             SHARED

             # Provides a relative path to your source file(s).
             native-lib.cpp)

# Searches for a specified prebuilt library and stores the path as a
# variable. Because CMake includes system libraries in the search path by
# default, you only need to specify the name of the public NDK library
# you want to add. CMake verifies that the library exists before
# completing its build.

find_library( # Sets the name of the path variable.
              log-lib

              # Specifies the name of the NDK library that
              # you want CMake to locate.
              log )

# Specifies libraries CMake should link to your target library. You
# can link multiple libraries, such as libraries you define in this
# build script, prebuilt third-party libraries, or system libraries.

target_link_libraries( # Specifies the target library.
                       native-lib
                       signalfeature
                       # Links the target library to the log library
                       # included in the NDK.
                       ${log-lib} )
  • native-lib.cpp

extern "C"
JNIEXPORT jdouble JNICALL
Java_com_afison_machinelearning_MainActivity_calculateMean(JNIEnv *env, jobject thiz,
                                                           jdoubleArray data) {
    // TODO: implement calculateMean()
    jboolean isCopy1;
    //获取数组长度
    jsize len = env->GetArrayLength(data);
    double *srcData = env->GetDoubleArrayElements(data,&isCopy1);
    double res = SignalFeature::calculateMean(srcData,len);
    return res;
}

5.MainActivity中调用

@Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        // Example of a call to a native method
        TextView tv = findViewById(R.id.sample_text);
        double [] data = {1,2,3,4};
        double res = calculateMean(data);
        tv.setText(String.valueOf(res));
    }
//均值
    public native double calculateMean(double[] data);

6.结果展示

7.总结

这个例子展示了Android系统下,Java语言调用C++算法,当我们开发嵌入式软件时,选用的Android系统作界面展示,若需要使用算法对某些信号处理,就可以用这种方式调用机器学习算法。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/484421.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

如何用python编写记录你女友的生日呢?

如何用python编写记录你女友的生日呢&#xff1f; 我这边写一个简单的 Python 程序示例,可以用来记录生日.这个程序将用户输入的姓名和生日信息保存到一个字典中,并允许用户查找特定姓名对应的生日信息. def record_birthday():birthdays {}while True:print("1. 添加生…

IP地址、子网掩码、网关

这些概念的来源 很久以前&#xff0c;有两个计算机想要相互通信&#xff0c;于是它们在自己的设备上安装了一个网卡&#xff0c;并用网线连接&#xff1a; 这个时候&#xff0c;又来了一个计算机想要加入它们&#xff0c;于是这三个计算机互相通过网线连接&#xff1a; 随着想…

taro之Swiper的使用

图样&#xff1a; 往往我们需要轮播图去显示我们想要的图片之类的 这是工作的代码 <View classNametop-title><SwiperclassNamebanner-swiperinterval{3000}circularautoplay>{homeBannerList.map((item) > {return (<SwiperItem key{item.id}><View…

Linux之git

一、什么叫做版本控制 版本控制&#xff08;Revision control&#xff09;是一种在开发的过程中用于管理我们对文件、目录或工程等内容的修改历史&#xff0c;方便查看更改历史记录&#xff0c;备份以便恢复以前的版本的软件工程技术。简单来说就是用于管理多人协同开发项目的技…

MySQL临时表:临时存储数据的灵活利器

MySQL临时表&#xff1a;临时存储数据的灵活利器 MySQL临时表是处理数据时非常有用的工具&#xff0c;它提供了临时存储数据的能力&#xff0c;使得复杂查询、排序、聚合以及数据筛选变得更加高效和简单。在本文中&#xff0c;我们将深入探讨MySQL临时表的概念以及何时需要使用…

【算法刷题day1】Leetcode:704. 二分查找、27. 移除元素

文章目录 Leetcode 704. 二分查找解题思路代码总结 Leetcode 27. 移除元素解题思路代码总结 草稿图网站 java的Deque Leetcode 704. 二分查找 题目&#xff1a;704. 二分查找 解题思路 1.左闭右闭区间的搜索&#xff0c;循环条件为left < right。 2.左闭右开区间的搜索&…

C++一维数组练习oj(3)

为什么C的一维数组练习要出要做那么多的题目&#xff1f;因为我们是竞赛学生&#xff01;想要将每个知识点灵活运用的话就必须刷大量的题目来锻炼思维。 我使用的是jsswoj.com这个刷题网站&#xff0c;当然要钱... C一维数组练习oj(2)-CSDN博客这是上一次的题目讲解 这道题有…

Unity学习笔记 6.2D换帧动画

下载源码 UnityPackage 目录 1.导入图片 1.1. 图片的叠放顺序 2.图片切片 3.用动画控制器让马&#x1f40e;动起来 1.导入图片 直接拖拽进场景 检查 Texture Type&#xff08;纹理类型&#xff09;是否为 Sprite 创建2D精灵对象&#xff0c;拖拽图片到Sprite&#xff08…

【C++】关联式容器——map和set

1 关联式容器 STL中我们常用的部分容器&#xff0c;比如&#xff1a;vector、list、deque、forward_list(C11)等&#xff0c;这些容器统称为序列式容器&#xff0c;因为其底层为线性序列的数据结构&#xff0c;里面存储的是元素本身。 那什么是关联式容器呢&#xff1f;它与序…

蓝桥杯G431RBT6——定时器中使用led冲突以及led与lcd冲突等一系列问题

本文是解决 同时在 定时器中点灯 与 LCD屏幕显示 冲突异常的问题 我们大家都知道&#xff0c;G431RBT6开发板上led与lcd是冲突的&#xff0c;所以在lcd.c文件中的这三个函数中 void LCD_WriteReg(u8 LCD_Reg, u16 LCD_RegValue) void LCD_WriteRAM_Prepare(void) void LCD_Wr…

移动0【双指针】

移动零 cur每次走一步&#xff0c;dest走不走取决于cur有没有找到非0值&#xff0c;一旦找打非0值&#xff0c;交换。不是非0值&#xff0c;dest不动。》【非零&#xff0c;dest】【dest&#xff0c;0】 class Solution { public:void moveZeroes(vector<int>& num…

算法第三十二天-最长公共子序列

最长公共子序列 题目要求 解题思路 求这两个数组或者字符串的最长公共子序列问题&#xff0c;肯定要用到动态规划。 首先区分两个概念&#xff1a;子序列可以是不连续的&#xff1b;子数组&#xff08;子字符串&#xff09;是需要连续的&#xff1b;另外&#xff0c;动态规划…

制冷设备之转子式压缩机

滚动转子式压缩机又称活塞式压缩机&#xff0c;属于回转式压缩机。 转子压缩机结构 滚动转子式压缩机与往复活塞式压缩机相比&#xff0c;具有下列特点 1.零部件少&#xff0c;尺寸紧凑&#xff0c;结构简单&#xff0c;重量轻易损零件少&#xff0c;运行可靠&#xff1b; 2.…

C语言动态内存的管理

前言 本篇博客就来探讨一下动态内存&#xff0c;说到内存&#xff0c;我们以前开辟空间大小都是固定的&#xff0c;不能调整这个空间大小&#xff0c;于是就有动态内存&#xff0c;可以让我们自己选择开辟多少空间&#xff0c;更加方便&#xff0c;让我们一起来看看动态内存的有…

Vue3 上手笔记

1. Vue3简介 2020年9月18日&#xff0c;Vue.js发布版3.0版本&#xff0c;代号&#xff1a;One Piece&#xff08;n 经历了&#xff1a;4800次提交、40个RFC、600次PR、300贡献者 官方发版地址&#xff1a;Release v3.0.0 One Piece vuejs/core 截止2023年10月&#xff0c;最…

Linux的一些基本指令

​​​​​​​ 目录 前言&#xff1a; 1.以指令的形式登录 2.ls指令 语法&#xff1a; 功能&#xff1a; 常用选项&#xff1a; 3.pwd指令 4.cd指令 4.1 绝对路径与相对路径 4.2 cd .与cd ..&#xff08;注意cd后先空格&#xff0c;然后两个点是连一起的&#xff0…

Git bash获取ssh key

目录 1、获取密钥 2、查看密钥 3、在vs中向GitHub推送代码 4、重新向GitHub推送修改过的代码 1、获取密钥 指令&#xff1a;ssh-keygen -t rsa -C "邮箱地址" 连续按三次回车&#xff0c;直到出现类似以下界面&#xff1a; 2、查看密钥 路径&#xff1a;C:\U…

复旦EMBA参访娃哈哈:交流企业创新转型、家族企业管理之道

早在多年前&#xff0c;复旦EMBA同学曾参访娃哈哈集团&#xff0c;与宗庆后先生对话&#xff0c;就国内企业创新转型、家族企业管理之道、“企二代”的成长、企业社会责任等热点问题向其探讨交流。通过面对面的实地企业参访和行业领袖的深入交流&#xff0c;亲身触摸中国科创的…

车辆信息查询API:高效获取车牌号对应车辆的实时信息

随着汽车的普及和交通管理的加强&#xff0c;对于车辆信息的查询需求也越来越大。车辆信息查询API就是为了满足这一需求而开发的&#xff0c;它可以通过输入车牌号&#xff0c;快速获取车辆的相关信息&#xff0c;包括初始登记日期、上险日期、保险到期时间、车架号、品牌等。但…

判断隔离纸到钢壳边缘的距离,燕尾是否超标

方法如下: 方法1:通过找圆工具上的点求解隔离纸边缘点-钢壳边缘点的距离。 #region namespace imports using System; using System.Collections; using System.Drawing; using System.IO; using System.Windows.Forms; using Cognex.VisionPro; using Cognex.VisionPro.To…
最新文章