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 }

目录
相关文章
|
6天前
|
Java
Java——抽象类和接口
抽象类是一种不能被实例化的类,至少包含一个抽象方法(无实现体的方法),常用于定义一组相关类的共同特征,并强制子类实现特定方法。抽象方法不能被 `static` 或 `final` 修饰,且必须被重写。 接口则是一个完全抽象的类,用于规范类的行为。接口使用 `interface` 关键字定义,不能实例化,并且类与接口之间是实现关系。 内部类是在一个类内定义的类,分为成员内部类、静态内部类、局部内部类和匿名内部类。成员内部类可被修饰符修饰,静态内部类只能访问外部类的静态成员,局部内部类定义在方法内,匿名内部类则隐藏了名字,直接通过 `new` 关键字定义并实现接口或继承类。
12 5
Java——抽象类和接口
|
6天前
|
Java
Java——接口的使用实例
Comparable接口用于自定义类的对象比较。通过实现此接口并重写`compareTo`方法,可以定义自定义类型的比较规则。 接下来介绍了Comparator接口,它提供了一种更灵活的比较方式。通过实现Comparator接口并重写`compare`方法,可以根据不同属性定义不同的比较规则。例如,定义一个`BrandComparator`类来比较汽车的品牌。 最后,介绍了Cloneable接口,用于实现对象的克隆。实现该接口并重写`clone`方法后,可以创建对象的浅拷贝或深拷贝。浅拷贝仅复制对象本身,深拷贝则会递归复制所有成员变量。
13 4
Java——接口的使用实例
|
12天前
|
Java 数据库连接 数据库
Java服务提供接口(SPI)的设计与应用剖析
Java SPI提供了一种优雅的服务扩展和动态加载机制,使得Java应用程序可以轻松地扩展功能和替换组件。通过合理的设计与应用,SPI可以大大增强Java应用的灵活性和可扩展性。
44 18
|
9天前
|
Java
Java的方法详解
Java的方法是类中的重要组成部分,用于定义类的行为。方法可以接收参数、执行操作并返回结果。其基本语法包括返回类型、方法名、参数列表和方法体。方法支持重载,即同名但参数不同的多个方法;静态方法则直接通过类名调用,无需实例化。此外,Java还支持可变参数,允许方法接收不定数量的参数。通过访问修饰符如`public`、`protected`、`private`,可以控制方法的可见性。方法是实现类功能的基本单元,增强了程序的灵活性和复用性。
|
9天前
|
Java 开发者
Java的接口详解
Java接口是编程中的一种重要特性,用于定义方法签名而不提供具体实现,作为类之间的契约,使不同类能以统一方式交互。接口使用`interface`关键字定义,可包含方法声明和常量。类通过`implements`关键字实现接口,并可同时实现多个接口,解决多重继承问题。接口中的方法默认为抽象方法,变量默认为`public static final`。Java 8引入了默认方法和静态方法,增强接口功能。接口广泛应用于回调机制和多态性实现,有助于构建更灵活和可维护的代码结构。
|
16天前
|
存储 Java 程序员
优化Java多线程应用:是创建Thread对象直接调用start()方法?还是用个变量调用?
这篇文章探讨了Java中两种创建和启动线程的方法,并分析了它们的区别。作者建议直接调用 `Thread` 对象的 `start()` 方法,而非保持强引用,以避免内存泄漏、简化线程生命周期管理,并减少不必要的线程控制。文章详细解释了这种方法在使用 `ThreadLocal` 时的优势,并提供了代码示例。作者洛小豆,文章来源于稀土掘金。
|
21天前
|
Java
用JAVA架建List集合为树形结构的代码方法
这段代码定义了一个表示树形结构的 `Node` 类和一个用于构建树形结构的 `TreeController`。`Node` 类包含基本属性如 `id`、`pid`、`name` 和 `type`,以及子节点列表 `children`。`TreeController` 包含初始化节点列表并将其转换为树形结构的方法。通过过滤和分组操作实现树形结构的构建。详情可见:[代码示例链接1](http://www.zidongmutanji.com/zsjx/43551.html),[代码效果参考链接2](https://www.257342.com/sitemap/post.html)。
28 5
|
19天前
|
Java
盘点java8 stream中隐藏的函数式接口
`shigen`是一位坚持更新文章的博客作者,记录成长历程,分享认知见解,留住感动瞬间。本文介绍了函数式接口的概念及其在Java中的应用,包括`Comparator`、`Runnable`、`Callable`等常见接口,并详细讲解了`Function`、`Predicate`、`Consumer`、`Supplier`和`Comparator`等函数式接口的使用方法及应用场景,展示了如何利用这些接口简化代码并提高编程效率。**个人IP:shigen**,与shigen一起,每天进步一点点!
29 0
盘点java8 stream中隐藏的函数式接口
|
20天前
|
Java 编译器 开发者
Java中的Lambda表达式与函数式接口
【8月更文挑战第31天】本文将深入探讨Java 8中引入的Lambda表达式和函数式接口,它们如何改变我们编写代码的方式。通过简化集合操作、事件处理等示例,我们将看到这些特性如何提升代码可读性、减少冗余,并提高开发效率。准备好一起探索这个让Java编程更加简洁强大的新世界吧!
|
23天前
|
Java
在Java多线程领域,精通Lock接口是成为高手的关键。
在Java多线程领域,精通Lock接口是成为高手的关键。相较于传统的`synchronized`,Lock接口自Java 5.0起提供了更灵活的线程同步机制,包括可中断等待、超时等待及公平锁选择等高级功能。本文通过实战演练介绍Lock接口的核心实现——ReentrantLock,并演示如何使用Condition进行精确线程控制,帮助你掌握这一武林秘籍,成为Java多线程领域的盟主。示例代码展示了ReentrantLock的基本用法及Condition在生产者-消费者模式中的应用,助你提升程序效率和稳定性。
18 2