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 }

目录
相关文章
|
27天前
|
Java
【Java】一个简单的接口例子(帮助理解接口+多态)
【Java】一个简单的接口例子(帮助理解接口+多态)
16 0
|
7天前
|
Java
Java中ReentrantLock中tryLock()方法加锁分析
Java中ReentrantLock中tryLock()方法加锁分析
9 0
|
1天前
|
设计模式 Java
Java接口与抽象类
Java接口与抽象类
13 0
|
6天前
|
Java Shell
Java 21颠覆传统:未命名类与实例Main方法的编码变革
Java 21颠覆传统:未命名类与实例Main方法的编码变革
10 0
|
6天前
|
安全 Java 编译器
接口之美,内部之妙:深入解析Java的接口与内部类
接口之美,内部之妙:深入解析Java的接口与内部类
25 0
接口之美,内部之妙:深入解析Java的接口与内部类
|
7天前
|
安全 Java
append在Java中是哪个类下的方法
append在Java中是哪个类下的方法
21 9
|
9天前
|
缓存 安全 Java
Java中函数式接口详解
Java 8引入函数式接口,支持函数式编程。这些接口有单一抽象方法,可与Lambda表达式结合,简化代码。常见函数式接口包括:`Function&lt;T, R&gt;`用于转换操作,`Predicate&lt;T&gt;`用于布尔判断,`Consumer&lt;T&gt;`用于消费输入,`Supplier&lt;T&gt;`用于无参生成结果。开发者也可自定义函数式接口。Lambda表达式使实现接口更简洁。注意异常处理和线程安全。函数式接口广泛应用于集合操作、并行编程和事件处理。提升代码可读性和效率,是现代Java开发的重要工具。
21 0
|
9天前
|
Java 关系型数据库 MySQL
大厂面试题详解:Java抽象类与接口的概念及区别
字节跳动大厂面试题详解:Java抽象类与接口的概念及区别
33 0
|
10天前
|
Java
Java中的多线程实现:使用Thread类与Runnable接口
【4月更文挑战第8天】本文将详细介绍Java中实现多线程的两种方法:使用Thread类和实现Runnable接口。我们将通过实例代码展示如何创建和管理线程,以及如何处理线程同步问题。最后,我们将比较这两种方法的优缺点,以帮助读者在实际开发中选择合适的多线程实现方式。
19 4
|
16天前
|
Java
Java通过反射获取类调用方法
Java通过反射获取类调用方法
18 0