【JAVA_其他】JNI的简单使用

本文介绍如何使用Java通过JNI调用C++编写的DLL,包括创建Java接口、生成头文件、C++实现及编译DLL,最后在Java中加载并测试DLL。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在实际的工作中遇到了一个问题,需要使用JAVA调用C++编写的dll,其实类似的需求有很多,这是并不是一种牵强,奇怪的用法。JAVA可以通过JNI来使用C++的dll,本文为一个简单的操作流程。

通过JNI来调用dll,是需要让C++代码和JAVA代码互相依赖的,并不是说,随便拿一个dll,JAVA引用就行。

假设现在存在一个需求:通过JNI调用由C++编写的hello.dll,实现JAVA传入任意名name,dll让name说“hello world!”。

1首先我们需要新建一个java文件,命名为SimpleJni.java。在类内自定义一个函数SayHello。

public class SimpleJni

{

    public native void SayHello(String name);

}

2执行下面俩个命令得到头文件SimpleJni.h,注意脚本所在路径。

javac SimpleJni.java

javah SimpleJni

3至此,有了头文件之后,我们就可以开始用C++来开发dll给java调用了。打开vs,新建项目Hello。右边两个任选其一。

4配置项目:本例不需要依赖其他的东西,设置为空项目即可,配置如下图。

5将第2步得到的SimpleJni.h添加至项目中。

右键项目添加现有项。

6打开SimpleJni.h文件,可以看见一些报错的信息。

7右键项目,属性,选择VC++目录的包含目录,在后面加上如下两个路径(在你java的安装目录下可以找到),记得两条记录用英文的;号隔开。本人java jdk的安装路径为E:\Program Files\Java\jdk1.8.0_191\。

E:\Program Files\Java\jdk1.8.0_191\include;

E:\Program Files\Java\jdk1.8.0_191\include\win32

8头文件报错消除之后,新建hello.cpp文件,复制SimpleJni.h中的方法至hello.cpp中。

9其实,我们的目的就是用C++实现上述复制的那个方法,在第8步的基础上,引入头文件,去掉方法后的;号(容易看漏),给参数加上名称,最后一个jstring就是咱们java需要传入的name参数,hello.cpp代码如下。

#include "SimpleJni.h"


JNIEXPORT void JNICALL Java_SimpleJni_SayHello

(JNIEnv *env, jobject thiz, jstring name)

{

    const char* temp = env->GetStringUTFChars(name, false);

    printf("%s say : hello world!", temp);

    //释放资源

    env->ReleaseStringUTFChars(name, temp);

}

10因为本人的系统和jdk都是64位的,所以编译的dll应当是64位的,并且最终调用该dll的jdk环境也应该为64位的。做如下配置,生成可用的hello.dll。

右键解决方案属性

        右键项目属性

11当我们重新配置完成后,发现头文件又报错了,这并没有什么关系,重新指定一下第7步中那两条路径即可,但是指定的位置不一样,右键项目,属性。如下图。

12至此,所有配置也就完成了,右键项目,生成。生成成功之后,在项目路径中可以找到Hello.dll文件。

13测试,拷贝Hello.dll,我们回到SimpleJni.java中,添加主函数进行测试,代码如下。

public class SimpleJni

{

    public native void SayHello(String name);

    public static void main(String[] args)

    {

         System.loadLibrary("Hello");

         SimpleJni SJ=new SimpleJni();

         SJ.SayHello("AngryCaveman");

    }

}

14执行结果如下图

 

### Java JNI 示例:`MainActivity` 中 `stringFromJNI` 方法实现及调用过程 在 Android 应用开发中,Java Native Interface (JNI) 提供了一种机制让 Java 和 C/C++ 代码能够相互协作。为了展示如何利用 JNI 实现 Java 和本地代码之间的交互,下面将介绍一个具体的例子——在 `MainActivity.java` 文件里定义了一个名为 `stringFromJNI()` 的函数来获取来自原生库的消息。 #### 创建 Java 类中的声明部分 首先,在项目的主活动中声明两个静态变量用于加载共享库以及对外提供接口的方法: ```java public class MainActivity extends AppCompatActivity { static { System.loadLibrary("native-lib"); } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); TextView tv = findViewById(R.id.sample_text); tv.setText(stringFromJNI()); } // 声明要由本地代码实现的公共成员方法 public native String stringFromJNI(); } ``` 这段代码展示了如何在一个 Activity 内部引入外部编写的动态链接库,并通过关键字 `native` 来标记那些将在其他地方被实际编码执行的功能[^1]。 #### 编写对应的 C 或者 C++ 函数 接着需要编写相应的 C 或者 C++ 源文件(通常命名为 `.cpp`),其中包含了上述提到的 `stringFromJNI` 函数的具体逻辑。这里给出的是基于 C++ 版本的例子: ```cpp #include <jni.h> #include <string> extern "C" JNIEXPORT jstring JNICALL Java_com_example_myapplication_MainActivity_stringFromJNI(JNIEnv *env, jobject /* this */) { std::string hello = "Hello from C++"; return env->NewStringUTF(hello.c_str()); } ``` 此段程序实现了当应用程序请求访问该字符串资源时所应返回的内容。注意这里的命名约定遵循了特定模式以便于 JVM 正确识别它作为目标对象上的指定实例方法的一部分[^2]。 #### 构建与调试 完成以上两步之后就可以按照常规流程打包 APK 并安装至设备上测试效果了。如果一切顺利的话,则会在界面上看到预期的文字输出:“Hello from C++”。 #### 关联问题处理 对于更复杂的需求场景下可能还需要考虑更多因素,比如异常捕获、线程同步等问题;另外值得注意的一点是在某些情况下直接传递简单数据结构并不能完全解决问题,这时就需要借助 JNI 接口进一步操作 Java 对象或数组等高级特性[^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值