重庆分公司,新征程启航
为企业提供网站建设、域名注册、服务器等服务
首先下截JNative组件
融水ssl适用于网站、小程序/APP、API接口等需要进行数据传输应用场景,ssl证书未来市场广阔!成为创新互联的ssl证书销售渠道,可以享受市场价格4-6折优惠!如果有意向欢迎电话联系或者加微信:18980820575(备注:SSL证书合作)期待与您的合作!
jnative.sourceforge.net/ 到这里下载JNative开源项目,我下载的是1.3.2
解压JNative-st1:chsdate isrocdate="False" islunardate="False" day="30"
month="12" year="1899"1.3.2/st1:chsdate.zip
获得三个文件,分别是:JNativeCpp.dll,libJNativeCpp.so,JNative.jar 。
JNativeCpp.dll
Windows下用的,拷贝到windows / system32目录下;
libJNativeCpp.so
Linux下的,拷贝到系统目录下;
JNative.jar 这是一个扩展包,导入工程LIB中或将其拷贝到jdk\jre\lib\ext
下,系统会自动加载。
•使用说明
我的项目将使用JNative组件调用一个测试应用服务器状态的TestAppSvr.dll文件,Dll文件中包含一个TestConnect()方法,返回一个整形的结果(1或0)
首先配置好JNative组件的windows环境:
将Native要用到JNativeCpp.dll放在系统盘的\WINDOWS\system32下
将JNative.jar导入工程中,新建一个调用类:
java 代码
复制代码
代码如下:
package com.tvjody;
import
java.io.File;
import java.io.FileOutputStream;
import
java.io.IOException;
import java.io.InputStream;
import
org.xvolks.jnative.JNative;
import org.xvolks.jnative.Type;
import
org.xvolks.jnative.exceptions.NativeException;
public class
AppSvrTestConnect {
public AppSvrTestConnect() {
}
/**
* 测试应用服务器连接状态
*
* TestConnect
* @param ip 应用服务器IP
* @param port 端口
* @param
intrcpt 是否采用数据压缩方式 1 :true 0:false
* @return int 1 :成功 0:失败
* @throws NativeException
* @throws IllegalAccessException
*/
private static final int TestConnect(String ip, int port, int
intrcpt)throws NativeException, IllegalAccessException {
JNative n
= null;
try {
n = new
JNative("TestAppSvr.dll", "TestConnect");
n.setRetVal(Type.INT);
int i = 0;
n.setParameter(i++, Type.STRING, ip);
n.setParameter(i++,
Type.INT, "" + port);
n.setParameter(i++, Type.INT, "" +
intrcpt);
n.invoke();
return
Integer.parseInt(n.getRetVal());
} finally {
if
(n != null)
n.dispose();
}
}
/**
* 指定Dll文件路径,动态加载本地链接库,测试应用服务器连接状态
* setDllPath
* @param path Dll文件的路径,不包含DLL名称 例如:windows - d:\test\test\ unix -
root/test/test/
* @param ip 应用服务器IP
* @param port 端口
* @param intrcpt 是否采用数据压缩方式 1 :true 0:false
* @return int 1
:成功 0:失败
* @throws NativeException
* @throws
IllegalAccessException
*/
public static final int
TestConnectFromDllPath(String path,String ip, int port, int intrcpt) throws
NativeException, IllegalAccessException{
path +=
"TestAppSvr.dll";
System.load(path);
return
TestConnect(ip,port,intrcpt);
}
/**
*
Dll文件放在JRE\bin目录下面,ClassLoader就能通过System.loadLibrary()动态加载本地链接库
*
TestConnectFromDllPath
* @param ip 应用服务器IP
* @param port 端口
* @param intrcpt 是否采用数据压缩方式 1 :true 0:false
* @return int 1
:成功 0:失败
* @throws NativeException
* @throws
IllegalAccessException
*/
public static final int
TestConnectFromDllPath(String ip, int port, int intrcpt) throws NativeException,
IllegalAccessException{
System.loadLibrary("TestAppSvr");
return TestConnect(ip,port,intrcpt);
}
}
这个类实现了一个静态私有方法,用来调用Dll文件中的方法返回结果
private static final int TestConnect(String ip, int port, int intrcpt)
两个静态公共方法,分两种方式装载DLL文件
public static final int TestConnectFromDllPath(String path,String ip, int
port, int intrcpt) //通过DLL文件的路径
public static final int
TestConnectFromDllPath(String ip, int port, int intrcpt)
//通过ClassLoader
然后新建一个类,调用AppSvrTestConnect.java,实现方法一调用,我是将TestAppSvr.dll文件与Demo.java放在一个目录下
,所以得到Demo.java的路径后就可以得到TestAppSvr.dll的路径,调用AppSvrTestConnect.TestConnectFromDllPath()方法后就能返回正确的信息.方法二是已经将TestAppSvr.dll放在了Jre\bin目录下,在JVM的Classloader的时候会自动加载,然后通过System.loadLibrary("TestAppSvr")就可以装配DLL文件.
java 代码
复制代码
代码如下:
public class Demo {
public int
getInfo() throws NativeException, IllegalAccessException{
String path=getClass().getResource(File.separator).getPath();
path = path.substring(1,path.length());
System.out.println(path); //得到DLL文件的路径
String ip =
"192.168.0.48"; //服务器IP
int port = 221; //端口
int intrcpt = 1; //数据压缩方式传送,1为采用;0为不采用
//方法1 传入Dll文件的路径
//int info =
AppSvrTestConnect.TestConnectFromDllPath(path, ip, port, intrcpt);
//方法2 Dll文件已经放在JRE\bin目录下面
int info =
AppSvrTestConnect.TestConnectFromDllPath(ip, port, intrcpt);
//1为成功,0为失败
if (info == 1)
System.out.println("应用服务器可用。");
else
System.out.println("应用服务器不可用,请检查IP地址和端口是否正确。");
return info;
}
System.loadLibrary():装载Windows\System32下或jre\bin或Tomcat\bin目录下的本地链接库
System.load():根据具体的目录来加截本地链接库,必须是绝对路径
•备注
上面的示例工程,因为是例子,所以没有大多的设计,只是实现了装载DLL文件,调用DLL文件方法,返回信息.
注意JVM只允许一个默认的ClassLoader来load native library,同时并不提供专门的API来unload一个loaded
native library,所以在项目调试的时候,独立启动Web Server.
JAVA通过JNI调用本地方法,而本地方法是以库文件的形式存放的(在WINDOWS平台上是DLL文件形式,在UNIX机器上是SO文件形式)。通过调用本地的库文件的内部方法,使JAVA可以实现和本地机器的紧密联系,调用系统级的各接口方法。
简单介绍及应用如下:
一、JAVA中所需要做的工作
在JAVA程序中,首先需要在类中声明所调用的库名称,如下:
static {
System.loadLibrary(“goodluck”);
}
在这里,库的扩展名字可以不用写出来,究竟是DLL还是SO,由系统自己判断。
还需要对将要调用的方法做本地声明,关键字为native。并且只需要声明,而不需要具 体实现。如下:
public native static void set(int i);
public native static int get();
然后编译该JAVA程序文件,生成CLASS,再用JAVAH命令,JNI就会生成C/C++的头文件。
例如程序testdll.java,内容为:
public class testdll
{
static
{
System.loadLibrary("goodluck");
}
public native static int get();
public native static void set(int i);
public static void main(String[] args)
{
testdll test = new testdll();
test.set(10);
System.out.println(test.get());
}
}
用javac testdll.java编译它,会生成testdll.class。
再用javah testdll,则会在当前目录下生成testdll.h文件,这个文件需要被C/C++程序调用来生成所需的库文件。
以下这个例子实现的是通过调用C写的DLL,计算字符串长度。 一、 生成C的头文件 1. 编辑Main.java public class Main { public native static int getStrNum(byte str[], int strLen); }2. 生成头文件 按win + r打开“运行”窗口,输入“cmd”,打开DOS控制台窗口。进入上面Main.java所在的目录中,输入: javac Main.java javah Main 两条命令完成后会生成Main.h文件 二、 生成DLL 1. 新建空工程 在VS中新建工程:Win32 Console Application,取名“MakeDLL”,选择空工程。新建后修改工程属性: (1) General中,将“Configuration Type”改为“Dynamic Library (.dll)”; (2) C/C++的General中,将“Additional Include Directories”添加两项: “C:Program FilesJavajdk1.5.0_06include” “C:Program FilesJavajdk1.5.0_06includewin32” 根据你安装的JAVA路径而设置,这是因为在C/C++编译过程中要使用JAVA的jni.h等文件 2. 将上面生成的Main.h添加入工程 3. 新建一个新的文件main.cpp,代码如下: #include 'Main.h' #include 'string.h' JNIEXPORT jint JNICALL Java_Main_getStrNum (JNIEnv *env, jclass cls, jbyteArray str, jint strLen) { jbyte *buffer = (*env).GetByteArrayElements(str, 0); buffer[strLen] = ''; // 由于JAVA自动分配空间,需要传进来字符串长度参数 jint len = strlen((char*)buffer); return len; }4. 按F7编译链接,生成MakeDLL.dll文件 三、 JAVA调用DLL 在JAVA中使用上面生成的DLL文件: 1. 将MakeDLL.dll复制到与使用DLL的JAVA文件相同的目录下。 2. JAVA文件的源代码: public class Main { static { System.loadLibrary('MakeDLL'); } public native static int getStrNum(byte str[], int strLen); public static void main(String[] args) { Main instance = new Main(); String str = new String('haha'); int num = instance.getStrNum(str.getBytes(), str.length()); System.out.println(num); } }总结: 另外,JAVA与DLL函数的参数传递过程,像int类型的,可以直接从JNI函数参数列表中获取;而数组元素则要通过JNIEnv来获取: jbyte *buffer = (*env).GetByteArrayElements(str, 0); // str是参数列表中jbyteArray变量 同理可以获取char[]的元素。 函数返回时,可以直接用return语句返回。也可以设置参数列表中的变量指针来返回: (*env).SetByteArrayRegion(byteArray, 0, 128, byteArr); byteArray是参数列表中的jbyteArray变量,作为指针用来返回需要返回的信息;byteArr是jbyte*类型,里面存放的值是要返回的内容。 其它的信息可以从Google或Baidu中通过查找“jni”或“java调用DLL”来查阅到。 (T007)
这个是 ActiveX 的库吧,包装成一个组件了,所以你得找个 jacob 这样的操作 ActiveX 对象的库 API.