大家都知道用C++编写的程序如果用于windows使用则编译为xxx.dll文件,如果是Linux使用则编译为libxxx.so文件。下面将java调用dll/so文件的方法粘出来方便下次使用。此处使用的jna的方式

jna可以从官方下载,也可以从maven里面去引用,下面是引用代码:

<dependency>
<groupId>net.java.dev.jna</groupId>
<artifactId>jna</artifactId>
<version>4.2.2</version>
</dependency>

先把C++的一段代码粘出来参考。

#pragma once
#include <stdint.h>
#ifdef __linux
#define _APICALL
#define DLL_PUBLIC __attribute__ ((visibility ("default")))
#else
#define DLL_PUBLIC
#define _APICALL __stdcall
#endif // __linux static const int8_t CRYPTTYPE_SM2 = ; // 非对称加密,SM2官方推荐模式(256bit加密强度)
static const int8_t CRYPTTYPE_SM4CBC = ; // 对称分组加密,SM4 CBC模式(128bit加密强度)
enum //错误码
{
GMC_ERR_OK = , //成功
GMC_ERR_CryptoTypeNotSupport = , //不支持该密码系统
GMC_ERR_PubKeyLost = , //公钥未设置
GMC_ERR_PriKeyLost = , //私钥未设置
GMC_ERR_OperationNotSupportThisCryptoType = , //操作不支持目前的密码系统类型。
GMC_ERR_SMALLMEMSIZE = , //相关内存长度过小
GMC_ERR_BADALLOCATE = , //内存分配失败
GMC_ERR_KEY_LENGTH = , //KEY长度不符合
GMC_ERR_KEY_FORMAT = , //KEY格式不符合
GMC_ERR_PARAM = , //参数错误
GMC_ERR_OPENSSL_INTERERR = , //openssl 内部错误
GMC_ERR_CIPHER = , //CIPHER相关错误
GMC_ERR_CIPHERTYPE_NOTSUPPORT = , //不支持的cipher类型
GMC_ERR_CIPHER_NOTINIT = , //未初始化
GMC_ERR_CIPHER_NOTSETKEY = , //没有设置KEY
GMC_ERR_SM2 = , //sm2相关错误码
GMC_ERR_SM2_NOTINIT =, //未初始化
GMC_ERR_SM2_NOTSETKEY = , //未设置 公钥或者私钥
};
extern "C"
{
/*
函数:GMC_New_CTX
功能:初始化获取一个指定类型的密码体系。
参数:
type:密码系统类型
返回:成功返回上下文指针,失败返回NULL。如果返回失败这表明不支持type指定的类型。
Note:在相关使用完成后需要调用GMC_Delete_CTX函数进行释放资源。
*/
DLL_PUBLIC void* _APICALL GMC_New_CTX(int8_t type);
/*
函数:GMC_Delete_CTX
功能:释放指定的密码体系上下文
参数:
pCtx:指定密码体系上下文
返回:无
*/
DLL_PUBLIC void _APICALL GMC_Delete_CTX(void *pCtx);
/*
函数:GMC_set_Key
功能:用于对称加密设置密钥
参数:
pKey:密钥(16进制数据的字符串化形式)
返回指:成功返回0,失败见错误码
*/
DLL_PUBLIC long _APICALL GMC_set_Key(void *pCtx,const char *pszKey);
/*
函数:GMC_ECKEY_set_PublicKey
功能:椭圆曲线加密系统公钥设置
参数:
pszXKey:X坐标(16进制数据的字符串化形式)
pszYKey:Y坐标(16进制数据的字符串化形式)
返回:成功返回0,失败见错误码
*/
DLL_PUBLIC long _APICALL GMC_ECKEY_set_PublicKey(void *pCtx,const char *pszXKey, const char *pszYKey);
/*
函数:GMC_ECKEY_set_PrivateKey
功能:椭圆曲线加解密系统私钥设置
参数:
pszXKey:私钥(16进制数据的字符串化形式)
返回:成功返回0,失败见错误码
*/
DLL_PUBLIC long _APICALL GMC_ECKEY_set_PrivateKey(void *pCtx,const char *pszKey);
/*
函数:GMC_Encrypt
功能:加密指定数据
参数:
pData:被加密数据地址
cbData:被加密数据长度
pEncrypttedData[out]:用于保存加密数据的地址(可以与pData内存重叠)
pcbEncryptted[int][out]:in:pEncrypttedData内存的大小,out:加密后数据的长度
返回:见错误码
note:参数pEncrypttedData可以为NULL,如果为NULl则pcbEncryptted参数返回需要长度。
*/
DLL_PUBLIC long _APICALL GMC_Encrypt(void *pCtx,const unsigned char *pData, uint32_t cbData, unsigned char *pEncrypttedData, uint32_t *pcbEncryptted);
/*
函数:GMC_Decrypt
功能:解密指定数据
参数:
pData:被解密数据地址
cbData:被解密数据长度
pDecrypttedData[out]:用于保存解密数据的地址(可以与pData内存重叠)
pcbDecryptted[int][out]:in:pDecrypttedData内存的大小,out:解密后数据的长度
返回:见错误码
note:参数pDecrypttedData可以为NULL,如果为NULl则pcbDecryptted参数返回需要长度。
*/
DLL_PUBLIC long _APICALL GMC_Decrypt(void *pCtx,const unsigned char *pData, uint32_t cbData, unsigned char *pDecrypttedData, uint32_t *pcbDecryptted);
/*
函数:GMC_GetLastErrMsg
功能:获取最后一次错误消息
参数:无
返回值:错误消息字串,返回值指针值必定不为NULL
*/
DLL_PUBLIC const char * _APICALL GMC_GetLastErrMsg();
}

然后就就介绍java的调用方法。什么引用jna就不说了。

一、创建一个接口并继承于com.sun.jna.Library(这里只实现了C++的部分方法)

public interface gmcrypto extends Library {
gmcrypto INSTANCE = (gmcrypto) Native.loadLibrary("gmcrypto", gmcrypto.class); /**
* 初始化获取一个指定类型的密码体系。
* @param type
*/
IntByReference GMC_New_CTX(long type); /**
* 获取上次的错误信息
* @return
*/
String GMC_GetLastErrMsg(); /**
* 加密,注意加密前先获取长度
* @param intByReference
* @param pData
* @param cbData
* @param pEncrypttedData
* @param pcbEncryptted
* @return
*/
long GMC_Encrypt(IntByReference intByReference, byte[] pData, int cbData, byte[] pEncrypttedData, IntByReference pcbEncryptted); /**
* 释放资源上下文
* @param intByReference
*/
void GMC_Delete_CTX(IntByReference intByReference);
}

二、在普通主方法中就可以调用了

public static void main2(String[] args) {
IntByReference gm = gmcrypto.INSTANCE.GMC_New_CTX(1);
if (gm == null) {
System.err.println("指定密码体系:失败");
} else {
/****************解密******************/
byte[] inByte = Base64.getDecoder().decode("MHYCIFnXBM5gF7OF4VYVmPh+exzQi9ik8dZBAFYs0hKrr8WRAiEArIjXQyOR1vdraQcdv9kG9/NGwVCEJ/UKIGw6gKcrTc0EIABJ7041HF7OLvzcSLvPVWDz3zjKWxOBu91someJ7D1+BA2tOEJsGFtH5rYu2Sxn");
long res = gmcrypto.INSTANCE.GMC_ECKEY_set_PrivateKey(gm, "C51F66571D5C472E383939D3C8944599D50452F4D8909B0C989C68888C0A1509");
System.out.println("设置私钥结果:" + res); IntByReference total = new IntByReference(1);
res = gmcrypto.INSTANCE.GMC_Decrypt(gm, inByte, inByte.length, null, total);
System.out.println(res + "\t" + total.getValue()); byte[] outByte = new byte[total.getValue()];
res = gmcrypto.INSTANCE.GMC_Decrypt(gm, inByte, inByte.length, outByte, total);
byte[] temp=new byte[total.getValue()];
System.arraycopy(outByte, 0, temp, 0, temp.length);
outByte=temp; System.out.println("解密结果:" + res + " 长度:" + outByte.length);
System.out.println(new String(outByte)); System.out.println("最近错误:" + gmcrypto.INSTANCE.GMC_GetLastErrMsg());
gmcrypto.INSTANCE.GMC_Delete_CTX(gm);
} }

重要说明:

1、gmcrypto就代表需要引用的dll/so文件的名称,由于java是跨平台,所以不加后缀,文件名为gmcrypto.dll/libgmcrypto.so

2、java中可以用IntByReference对象代表指针

3、windows需要把dll文件放到C:\Windows\System32路径。Linux需要把so文件放到/lib64。当然要根据情况区分32位和64位的情况

java调用dll/so文件的更多相关文章

  1. java调用dll或so动态库文件(c++/c)

    java调用dll或so动态库文件(c++/c) 博客分类:  工作 CC#C++JavaEclipse  java调用dll或so动态库文件(c++/c)开发平台:Eclipse3.3.1.1+CD ...

  2. Java 调用Dll

    Java 中怎么能调用到dll中的函数呢? 关键是java中生的本地函数名參数和dll中的本地函数名參数一模一样. 这个程序是java中调用dll中的求和函数. 一,java代码部分操作 1.新建pr ...

  3. 获取DLL的文件路径以及调用DLL的文件路径

    如何在DLL中,获取DLL本身的路径以及调用DLL的文件的路径呢?主要通过GetModuleFileName(HMODULEhModule,LPTSTR lpFilename,DWORD nSize) ...

  4. Java调用dll动态库

    最近项目里使用java调用dll动态库,因此研究了一下这方面的东西. 使用的工具包如下 <dependency> <groupId>net.java.dev.jna</g ...

  5. Java 调用 groovy 脚本文件,groovy 访问 MongoDB

    groovy 访问 MongoDB 示例: shell.groovy package db import com.gmongo.GMongoClient import com.mongodb.Basi ...

  6. Java调用.dll文件

    因为项目的需求,要在JAVA项目中调用Windows的Dll(动态链接库)文件,之前用Jni调用过C写的Dll文件,比较麻烦,这里不多说,网上也有很多这方面的文档.在网上找到一个开源的组件JNativ ...

  7. Java 调用 C++ (Java 调用 dll)康哥手把手教你

    摘要: 本文原创,转载请注明地址 http://www.cnblogs.com/baokang/p/4979243.html 因为要做点图形处理的项目,需要在Java中调用dll库,所以开发的第一步是 ...

  8. Java调用DLL有多种方式,常用的方式有JNative、JNA、JNI等。

    JNative方式调用dll JNative是一种能够使Java语言使调用DLL的一种技术,对JNI进行了封装,可能有些读者会有这样一个问题,JNative对JNI进行了封装,并且是一种跨语言的使用D ...

  9. (转)Java 调用 C++ (Java 调用 dll)

    转自: http://www.cnblogs.com/baokang/p/4979243.html 因为要做点图形处理的项目,需要在Java中调用dll库,所以开发的第一步是研究了一下Java Jni ...

随机推荐

  1. java bean属性拷贝工具类比对(性能+功能)

    业务系统中经常需要两个对象进行属性的拷贝,不能否认逐个的对象拷贝是最快速最安全的做法,但是当数据对象的属性字段数量超过程序员的容忍的程度比如通常超过5个属性的时候,代码因此变得臃肿不堪,使用一些方便的 ...

  2. 【题解】Luogu P2157 [SDOI2009]学校食堂

    原题传送门:P2157 [SDOI2009]学校食堂 一看题目就知道是状压dp 设f[i][j][k]表示第1到i-1个人都吃完了饭,第i个人以及后面的7个人是否打饭的状态为j,当前最后打饭的人的编号 ...

  3. opencv学习之路(8)、基本图像运算——加减与或

    一.图像加法 #include<opencv2/opencv.hpp> #include<iostream> using namespace cv; using namespa ...

  4. 动态规划之140 Word Break2

    这是一题不太明显的动态规划,主要考察的应该是深度优先搜索. static LinkedList<String> list = new LinkedList<String>(); ...

  5. 何为TLC、MLC、SLC?【转】

    本文转载自:https://blog.csdn.net/weixin_38233274/article/details/79310316 1.一块SSD由主控.DRAM缓存和NAND闪存三种芯片所组成 ...

  6. FFmpeg:视频转码、剪切、合并、播放速调整

    原文:https://fzheng.me/2016/01/08/ffmpeg/ FFmpeg:视频转码.剪切.合并.播放速调整 2016-01-08 前阵子帮导师处理项目 ppt,因为插入视频的格式问 ...

  7. 洛谷luogu2782

    P2782 友好城市 题目描述 有一条横贯东西的大河,河有笔直的南北两岸,岸上各有位置各不相同的N个城市.北岸的每个城市有且仅有一个友好城市在南岸,而且不同城市的友好城市不相同.每对友好城市都向政府申 ...

  8. POJ 2226 Muddy Fields(最小点覆盖)题解

    题意:一片r*c的地,有些地方是泥地,需要铺地板.这些地板宽1,长无限,但只能铺在泥地上不能压到其他地方,问你铺满所有泥地最少几块 思路:我们把一行中连续的泥地看成整体,并把所有横的整体里的点编成一个 ...

  9. LOJ#2427. 「POI2010」珍珠项链 Beads

    题目地址 题目链接 题解 不会算复杂度真是致命,暴力枚举k每次计算是n/2+n/3+n/4+...+1的,用调和级数算是\(O(nlogn)\)的... 如果写哈希表的话能够\(O(nlogn)\), ...

  10. Vue内置的Component标签用于动态切换组件

    html <div id="app"> <component :is="cut"></component> <butt ...