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 }

目录
相关文章
|
13天前
|
Java
Java中ReentrantLock中tryLock()方法加锁分析
Java中ReentrantLock中tryLock()方法加锁分析
12 0
|
2天前
|
Java 关系型数据库 MySQL
Elasticsearch【问题记录 01】启动服务&停止服务的2类方法【及 java.nio.file.AccessDeniedException: xx/pid 问题解决】(含shell脚本文件)
【4月更文挑战第12天】Elasticsearch【问题记录 01】启动服务&停止服务的2类方法【及 java.nio.file.AccessDeniedException: xx/pid 问题解决】(含shell脚本文件)
23 3
|
4天前
|
存储 Java
Java动态转发代理IP的实现方法
Java动态转发代理IP的实现方法
20 11
|
4天前
|
Java 开发者
探索 Java 的函数式接口和 Lambda 表达式
【4月更文挑战第19天】Java 中的函数式接口和 Lambda 表达式提供了简洁、灵活的编程方式。函数式接口有且仅有一个抽象方法,用于与 Lambda(一种匿名函数语法)配合,简化代码并增强可读性。Lambda 表达式的优点在于其简洁性和灵活性,常用于事件处理、过滤和排序等场景。使用时注意兼容性和变量作用域,它们能提高代码效率和可维护性。
|
5天前
|
Java
Java接口中可以定义哪些方法?
【4月更文挑战第13天】
7 0
Java接口中可以定义哪些方法?
|
7天前
|
设计模式 Java
Java接口与抽象类
Java接口与抽象类
17 0
|
11天前
|
Java Shell
Java 21颠覆传统:未命名类与实例Main方法的编码变革
Java 21颠覆传统:未命名类与实例Main方法的编码变革
12 0
|
11天前
|
安全 Java 编译器
接口之美,内部之妙:深入解析Java的接口与内部类
接口之美,内部之妙:深入解析Java的接口与内部类
33 0
接口之美,内部之妙:深入解析Java的接口与内部类
|
13天前
|
Java
Java中关于ConditionObject的signal()方法的分析
Java中关于ConditionObject的signal()方法的分析
21 4
|
13天前
|
安全 Java
append在Java中是哪个类下的方法
append在Java中是哪个类下的方法
21 9