Java 调用底层接口的几种方法

简介:

Java 调用底层接口

Java 调用底层接口要通过动态链接库进行,在windows下是dll文件,linux是so文件

Java调用动态库所需要关心的问题:

  •     如何装载文件,以及如何定位所要使用的方法;
  •      数据类型是如何对应的;
  •     如何给使用的方法传递参数;
  •     如何获取返回的值。

目前调用底层接口用的比较多的技术包括jni、jna、jnative、Nativecall等

JNI 封装本地接口

JAVA可以通过JNI接口访问本地的动态连接库,从而扩展JAVA的功能。使用JAVA JNI接口主要包括以下步骤:

²  编写JAVA代码,注明要访问的本地动态连接库和本地方法;

²  编译JAVA代码得到.class文件;

²  使用javah -jni 生成该类对应的C语言.h文件;

²  使用C/C++实现(3)生成的.h文件中声明的各函数;

²  编译C/C++实现代码生成动态连接库。
本文使用一个简单的helloWorld示例演示JNI的使用。

编写JAVA代码

 1 public class helloWorld
 2 
 3 {
 4 
 5     public native void SayHello(String name);
 6 
 7  
 8 
 9     static
10 
11     {
12 
13         System.loadLibrary("jniHelloworld");
14 
15     }
16 
17  
18 
19     public static void main(String [] argv)
20 
21     {
22 
23         helloWorld hello = new helloWorld();
24 
25         hello.SayHello("world ");
26 
27     }
28 }

编译JAVA代码

javac helloWorld.java


生成实现函数头文件


javah -classpath . helloWorld

得到的helloWorld.h文件内容如下:

1 /* DO NOT EDIT THIS FILE - it is machine generated */
 2 
 3 #include <jni.h>
 4 
 5  /* Header for class helloWorld */
 6 
 7  
 8 
 9 #ifndef _Included_helloWorld
10 
11  #define _Included_helloWorld
12 
13 #ifdef __cplusplus
14 
15  extern "C" {
16 
17  #endif
18 
19  /*
20 
21  * Class:     helloWorld
22 
23  * Method:    SayHello
24 
25  * Signature: (Ljava/lang/String;)V
26 
27  */
28 
29 JNIEXPORT void JNICALL Java_helloWorld_SayHello
30 
31   (JNIEnv *, jobject, jstring);
32 
33  
34 
35 #ifdef __cplusplus
36 
37 }
38 
39  #endif
40 
41  #endif

在VS中创建工程并实现该函数

1 #include "helloWorld.h"
 2 
 3 #include <stdio.h>
 4 
 5 #include <string.h>
 6 
 7  void JNICALL Java_helloWorld_SayHello(JNIEnv * env, jobject obj, jstring str)
 8 
 9 {
10 
11      jboolean  b  = true;
12 
13      char s[80];
14 
15      memset(s, 0, sizeof(s));
16 
17      strcpy_s(s ,(char*)env->GetStringUTFChars(str, &b));
18 
19      printf("Hello, %s", s);
20 
21      env->ReleaseStringUTFChars(str , NULL);
22 
23 }

 这是JNI的关键:通过env我们可以使用JAVA提供的一组函数操作与转换函数传递的参数。


编译VC项目得到动态连接库 helloWorld.dll。

把工程输出文件的位置设置成helloWorld类所在的目录,编译之前要把jdk的include目录加到工程属性中

然后在命令行中执行

Java helloWorld 会输出helloWorld

JNA封装本地接口

http://jna.java.net/#demos


1 package com.sun.jna.examples;
 2 
 3  
 4 
 5 import com.sun.jna.Library;
 6 
 7 import com.sun.jna.Native;
 8 
 9 import com.sun.jna.Platform;
10 
11  
12 
13  /** Simple example of JNA interface mapping and usage. */
14 
15  public class HelloWorld {
16 
17  
18 
19     // This is the standard, stable way of mapping, which supports extensive
20 
21     // customization and mapping of Java to native types.
22  
23     public interface CLibrary extends Library {
24 
25         CLibrary INSTANCE = (CLibrary)
26 
27             Native.loadLibrary((Platform.isWindows() ? "msvcrt" : "c"),
28 
29                                CLibrary.class);
30 
31  
32 
33         void printf(String format, Object... args);
34 
35     }
36 
37  
38 
39     public static void main(String[] args) {
40 
41         CLibrary.INSTANCE.printf("Hello, World\n");
42 
43         for (int i=0;i < args.length;i++) {
44 
45             CLibrary.INSTANCE.printf("Argument %d: %s\n", i, args[i]);
46 
47         }
48 
49     }
50 
51 }

JNA 可以直接调用底层接口,而不用对底层接口进行封装,像例子调用printf一样,底层接口可以通过这种方式直接把接口提供给java调用

Jnative

Jnative用法和jna类似,都是借助于开源项目实现对底层接口的调用,但是用法比jna简单一点,不需要在一个java接口中描述目标文件中的函数与结构,用法如下:

建立test_say.java文件,需要配置好JNative.jar包


1 import org.xvolks.jnative.JNative;
 2 
 3  import org.xvolks.jnative.Type;
 4 
 5  import org.xvolks.jnative.exceptions.NativeException;
 6 
 7 import org.xvolks.jnative.pointers.Pointer;
 8 
 9 import org.xvolks.jnative.pointers.memory.MemoryBlockFactory;
10 
11 import org.xvolks.jnative.pointers.memory.NativeMemoryBlock;
12 
13 import org.xvolks.jnative.util.Callback;
14 
15 //import org.xvolks.test.callbacks.linux.LinuxCallback;
16 
17  
18 
19 public class test_ helloWorld {
20 
21     private final static String LIB_NAME = " msvcrt ";    //自动判断.so 或者.dll
22 
23  
24 
25     public static void main(String[] args) throws NativeException, IllegalAccessException {
26 
27         try {
28 
29             JNative printf =new JNative(LIB_NAME," printf ");
30 
31            printf.setParameter(0,”hello world”);
32 
33             printf.invoke();
34 
35         }
36 
37         catch (Exception e)
38 
39         {
40 
41             e.printStackTrace();
42 
43         }
44 
45     }
46 
47 }

目录
相关文章
|
18天前
|
消息中间件 Java Kafka
在Java中实现分布式事务的常用框架和方法
总之,选择合适的分布式事务框架和方法需要综合考虑业务需求、性能、复杂度等因素。不同的框架和方法都有其特点和适用场景,需要根据具体情况进行评估和选择。同时,随着技术的不断发展,分布式事务的解决方案也在不断更新和完善,以更好地满足业务的需求。你还可以进一步深入研究和了解这些框架和方法,以便在实际应用中更好地实现分布式事务管理。
|
24天前
|
Java
java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
60 9
|
28天前
|
JSON Java Apache
非常实用的Http应用框架,杜绝Java Http 接口对接繁琐编程
UniHttp 是一个声明式的 HTTP 接口对接框架,帮助开发者快速对接第三方 HTTP 接口。通过 @HttpApi 注解定义接口,使用 @GetHttpInterface 和 @PostHttpInterface 等注解配置请求方法和参数。支持自定义代理逻辑、全局请求参数、错误处理和连接池配置,提高代码的内聚性和可读性。
112 3
|
17天前
|
安全 Java 开发者
Java中WAIT和NOTIFY方法必须在同步块中调用的原因
在Java多线程编程中,`wait()`和`notify()`方法是实现线程间协作的关键。这两个方法必须在同步块或同步方法中调用,这一要求背后有着深刻的原因。本文将深入探讨为什么`wait()`和`notify()`方法必须在同步块中调用,以及这一机制如何确保线程安全和避免死锁。
30 4
|
17天前
|
Java
深入探讨Java中的中断机制:INTERRUPTED和ISINTERRUPTED方法详解
在Java多线程编程中,中断机制是协调线程行为的重要手段。了解和正确使用中断机制对于编写高效、可靠的并发程序至关重要。本文将深入探讨Java中的`Thread.interrupted()`和`Thread.isInterrupted()`方法的区别及其应用场景。
21 4
|
14天前
|
Java 数据处理 数据安全/隐私保护
Java处理数据接口方法
Java处理数据接口方法
21 1
|
16天前
|
Java API
Java中内置的函数式接口
Java中内置的函数式接口
19 2
|
20天前
|
Java
在Java中如何实现接口?
实现接口是 Java 编程中的一个重要环节,它有助于提高代码的规范性、可扩展性和复用性。通过正确地实现接口,可以使代码更加灵活、易于维护和扩展。
43 3
|
19天前
|
Java
在Java中,接口之间可以继承吗?
接口继承是一种重要的机制,它允许一个接口从另一个或多个接口继承方法和常量。
57 1
|
19天前
|
Java 开发者
在 Java 中,一个类可以实现多个接口吗?
这是 Java 面向对象编程的一个重要特性,它提供了极大的灵活性和扩展性。
37 1