0

0

JNI头文件的正确生成与使用:为何不能直接使用现有C头文件

心靈之曲

心靈之曲

发布时间:2025-09-22 12:13:13

|

913人浏览过

|

来源于php中文网

原创

JNI头文件的正确生成与使用:为何不能直接使用现有C头文件

JNI头文件并非通过手动修改现有C头文件来创建,而是由javac -h命令根据包含native方法的Java类自动生成。这种机制确保了Java与原生代码之间类型和函数签名的正确匹配,避免了手动适配现有C头文件时常见的错误和不兼容性。

JNI简介与核心概念

java native interface (jni) 是java平台的一个标准,它允许java代码与其他语言(特别是c/c++)编写的应用程序和库进行交互。通过jni,java虚拟机(jvm)能够调用原生方法,原生方法也能调用java方法,从而实现java应用程序对底层系统功能或现有原生库的访问。

JNI的核心在于定义了一套规范,包括数据类型映射、函数命名约定以及一套用于在原生代码中操作Java对象和调用Java方法的API。为了实现这种互操作性,JNI要求原生方法遵循特定的函数签名和命名规则。

JNI头文件的错误认知与正确生成方式

在JNI开发中,一个常见的误解是试图直接将现有的C语言头文件(如本例中提供的BITMAP.h)适配为JNI头文件。然而,这种做法是错误的,因为JNI头文件具有其特定的结构和宏定义,这些是手动修改难以正确实现和维护的。

例如,提供的BITMAP.h文件定义了一个BITMAP结构体和一系列操作该结构体的函数,如create、close、drawLn等。这些函数签名是标准的C语言风格,不包含任何JNI特有的宏(如JNIEXPORT, JNICALL)或类型(如JNIEnv*, jobject)。因此,这个文件不是一个有效的JNI头文件,不能直接用于JNI开发。

正确的JNI头文件生成流程如下:

  1. 定义Java native 方法: 首先,在Java类中声明一个或多个native方法。这些方法没有方法体,由native关键字修饰,表示它们的实现将由原生代码提供。

    // BitmapProcessor.java
    public class BitmapProcessor {
        static {
            // 加载包含原生实现的共享库
            System.loadLibrary("bitmapnative"); 
        }
    
        // 声明一个native方法,用于创建位图
        public native long nativeCreateBitmap(int width, int height); 
        // 声明一个native方法,用于绘制线条
        public native void nativeDrawLine(long bitmapPtr, int x1, int y1, int x2, int y2);
        // 声明一个native方法,用于关闭位图
        public native void nativeCloseBitmap(long bitmapPtr);
    
        // 假设这里还有其他Java业务逻辑
    }

    在上述示例中,我们定义了三个native方法。注意,long bitmapPtr在这里被用作一个占位符,用于在Java和C之间传递BITMAP结构体的内存地址(指针)。

  2. 编译Java类并生成JNI头文件: 使用javac命令的-h选项来编译包含native方法的Java类,并自动生成对应的C/C++头文件。

    QIMI奇觅
    QIMI奇觅

    美图推出的游戏行业广告AI制作与投放一体化平台

    下载
    # 假设BitmapProcessor.java在当前目录
    javac -h . BitmapProcessor.java 

    执行此命令后,javac会在当前目录(或指定的目录)下生成一个名为BitmapProcessor.h的头文件。如果BitmapProcessor在一个包中(例如com.example.jni),那么生成的头文件将位于com/example/jni/BitmapProcessor.h。

  3. 生成的JNI头文件示例: 自动生成的头文件将包含JNI特有的宏和类型,以及遵循JNI命名约定(Java_包名_类名_方法名)的函数声明。

    /* DO NOT EDIT THIS FILE - it is machine generated */
    #include 
    /* Header for class BitmapProcessor */
    
    #ifndef _Included_BitmapProcessor
    #define _Included_BitmapProcessor
    #ifdef __cplusplus
    extern "C" {
    #endif
    /*
     * Class:     BitmapProcessor
     * Method:    nativeCreateBitmap
     * Signature: (II)J
     */
    JNIEXPORT jlong JNICALL Java_BitmapProcessor_nativeCreateBitmap
      (JNIEnv *, jobject, jint, jint);
    
    /*
     * Class:     BitmapProcessor
     * Method:    nativeDrawLine
     * Signature: (JIIII)V
     */
    JNIEXPORT void JNICALL Java_BitmapProcessor_nativeDrawLine
      (JNIEnv *, jobject, jlong, jint, jint, jint, jint);
    
    /*
     * Class:     BitmapProcessor
     * Method:    nativeCloseBitmap
     * Signature: (J)V
     */
    JNIEXPORT void JNICALL Java_BitmapProcessor_nativeCloseBitmap
      (JNIEnv *, jobject, jlong);
    
    #ifdef __cplusplus
    }
    #endif
    #endif

    这个头文件才是JNI开发中需要的。它包含了JNIEnv*(指向JNI环境的指针)、jobject(Java对象引用)以及Java基本类型对应的JNI类型(如jint对应Java的int,jlong对应Java的long)。

实现JNI原生方法与集成现有C代码

有了生成的JNI头文件后,接下来就可以编写C/C++源文件来实现这些原生方法。在这个实现文件中,我们需要包含生成的JNI头文件,并根据其声明来实现函数。

对于本例中已有的BITMAP.h和.ec文件,正确的做法是:在JNI的C/C++实现文件中,包含由javac -h生成的JNI头文件,同时包含原有的BITMAP.h。然后,在JNI原生方法的实现中,调用BITMAP.h中声明的函数。

// bitmapnative.c (JNI原生实现文件)
#include "BitmapProcessor.h" // 包含由javac生成的JNI头文件
#include "BITMAP.h"          // 包含原有的C头文件
#include            // 用于示例输出

// 实现nativeCreateBitmap方法
JNIEXPORT jlong JNICALL Java_BitmapProcessor_nativeCreateBitmap
  (JNIEnv *env, jobject obj, jint width, jint height) {
    printf("C: Creating bitmap with width=%d, height=%d\n", width, height);
    struct BITMAP *pbmp = create(width, height); // 调用原有的C函数
    if (pbmp == NULL) {
        // 抛出Java异常,如果需要
        return 0; 
    }
    return (jlong)pbmp; // 将C指针转换为jlong返回给Java
}

// 实现nativeDrawLine方法
JNIEXPORT void JNICALL Java_BitmapProcessor_nativeDrawLine
  (JNIEnv *env, jobject obj, jlong bitmapPtr, jint x1, jint y1, jint x2, jint y2) {
    struct BITMAP *pbmp = (struct BITMAP *)bitmapPtr; // 将jlong转换回C指针
    if (pbmp != NULL) {
        printf("C: Drawing line on bitmap %p from (%d,%d) to (%d,%d)\n", pbmp, x1, y1, x2, y2);
        drawLn(pbmp, x1, y1, x2, y2); // 调用原有的C函数
    } else {
        fprintf(stderr, "C: Error: bitmapPtr is NULL in nativeDrawLine\n");
    }
}

// 实现nativeCloseBitmap方法
JNIEXPORT void JNICALL Java_BitmapProcessor_nativeCloseBitmap
  (JNIEnv *env, jobject obj, jlong bitmapPtr) {
    struct BITMAP *pbmp = (struct BITMAP *)bitmapPtr;
    if (pbmp != NULL) {
        printf("C: Closing bitmap %p\n", pbmp);
        close(pbmp); // 调用原有的C函数
    } else {
        fprintf(stderr, "C: Error: bitmapPtr is NULL in nativeCloseBitmap\n");
    }
}

将bitmapnative.c文件(以及编译后的.ec文件)编译成一个共享库(例如Windows上的.dll,Linux/macOS上的.so/.dylib),然后Java程序就可以通过System.loadLibrary("bitmapnative")来加载并调用这些原生方法了。

总结与注意事项

  1. JNI头文件是生成的,不是手写的: 始终通过javac -h命令从Java native方法定义中生成JNI头文件。这是确保JNI接口正确性和兼容性的唯一官方途径。
  2. JNI作为适配层: JNI原生实现文件充当Java和现有C/C++代码之间的适配层。它接收来自Java的调用,将参数转换为C/C++类型,调用底层的C/C++函数,然后将结果(如果需要)转换回Java类型返回。
  3. 内存管理: 在JNI中,原生代码负责管理其自身分配的内存。如果原生代码创建了资源(如本例中的BITMAP结构体),并且其生命周期需要跨越Java方法调用,则通常会将指向这些资源的指针以jlong的形式返回给Java,由Java代码在适当的时机(例如通过另一个native方法)通知原生代码释放这些资源。
  4. 错误处理: 原生代码中发生的错误应适当地处理,例如通过JNI API向Java层抛出异常,或者返回错误码。
  5. 其他Java原生技术: 除了JNI,还有Java Native Access (JNA) 和 Java Bridging (BridJ) 等库,它们提供了更高级别的抽象,允许Java直接调用原生库中的函数,而无需编写JNI C/C++代码。这些技术在某些场景下可以简化开发,但JNI仍然是Java官方提供的标准原生接口。对于需要极致性能控制或与现有C/C++代码深度集成时,JNI是首选。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
C语言变量命名
C语言变量命名

c语言变量名规则是:1、变量名以英文字母开头;2、变量名中的字母是区分大小写的;3、变量名不能是关键字;4、变量名中不能包含空格、标点符号和类型说明符。php中文网还提供c语言变量的相关下载、相关课程等内容,供大家免费下载使用。

401

2023.06.20

c语言入门自学零基础
c语言入门自学零基础

C语言是当代人学习及生活中的必备基础知识,应用十分广泛,本专题为大家c语言入门自学零基础的相关文章,以及相关课程,感兴趣的朋友千万不要错过了。

620

2023.07.25

c语言运算符的优先级顺序
c语言运算符的优先级顺序

c语言运算符的优先级顺序是括号运算符 > 一元运算符 > 算术运算符 > 移位运算符 > 关系运算符 > 位运算符 > 逻辑运算符 > 赋值运算符 > 逗号运算符。本专题为大家提供c语言运算符相关的各种文章、以及下载和课程。

354

2023.08.02

c语言数据结构
c语言数据结构

数据结构是指将数据按照一定的方式组织和存储的方法。它是计算机科学中的重要概念,用来描述和解决实际问题中的数据组织和处理问题。数据结构可以分为线性结构和非线性结构。线性结构包括数组、链表、堆栈和队列等,而非线性结构包括树和图等。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

259

2023.08.09

c语言random函数用法
c语言random函数用法

c语言random函数用法:1、random.random,随机生成(0,1)之间的浮点数;2、random.randint,随机生成在范围之内的整数,两个参数分别表示上限和下限;3、random.randrange,在指定范围内,按指定基数递增的集合中获得一个随机数;4、random.choice,从序列中随机抽选一个数;5、random.shuffle,随机排序。

606

2023.09.05

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

531

2023.09.20

c语言get函数的用法
c语言get函数的用法

get函数是一个用于从输入流中获取字符的函数。可以从键盘、文件或其他输入设备中读取字符,并将其存储在指定的变量中。本文介绍了get函数的用法以及一些相关的注意事项。希望这篇文章能够帮助你更好地理解和使用get函数 。

646

2023.09.20

c数组初始化的方法
c数组初始化的方法

c语言数组初始化的方法有直接赋值法、不完全初始化法、省略数组长度法和二维数组初始化法。详细介绍:1、直接赋值法,这种方法可以直接将数组的值进行初始化;2、不完全初始化法,。这种方法可以在一定程度上节省内存空间;3、省略数组长度法,这种方法可以让编译器自动计算数组的长度;4、二维数组初始化法等等。

604

2023.09.22

clawdbot ai使用教程 保姆级clawdbot部署安装手册
clawdbot ai使用教程 保姆级clawdbot部署安装手册

Clawdbot是一个“有灵魂”的AI助手,可以帮用户清空收件箱、发送电子邮件、管理日历、办理航班值机等等,并且可以接入用户常用的任何聊天APP,所有的操作均可通过WhatsApp、Telegram等平台完成,用户只需通过对话,就能操控设备自动执行各类任务。

11

2026.01.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 8万人学习

Git 教程
Git 教程

共21课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号