概述:前段时间对读卡器进行开发,目的是开发一个工具类接口给其他同事用在项目上调用读卡器,于是呼写了一个Java调用读卡器,进行读卡,写卡的操作例子。由于使用的是厂商提供的dll库,所以选择了JNA进行开发。当然还有一种JNI方式,这种方式适合开发底层,比如封装SDK。业务开发建议使用JNA,方便使用。JNI开发比较繁琐,需要Java类,包对应C的类,包名称,以及编译等注意事项。
1、首先在pom里引入:JNA三方库
<dependency>
<groupId>com.sun.jna</groupId>
<artifactId>jna</artifactId>
<version>3.0.9</version>
</dependency>
2、接口类封装:用于存放dll库中可以调用的接口。
import com.sun.jna.Library;
/**
* @author liuxiaobao
* @date 2022/6/22 9:12
* 说明:RWOptionService 声明了对YOW-60X系列的读卡器,读和写的相关接口
*/
public interface RWOptionService extends Library {
int YW_GetReaderID(int ReaderID);
/**
* 函数: YW_USBHIDInitial
* 名称: 免驱USB端口初始化
* 参数: 无
* 返回值:>0为成功 ,其它失败
*/
int YW_USBHIDInitial();
/**
* 函数: YW_USBHIDFree
* 名称: 免驱USB端口释放
* 参数: 无
* 返回值:>0为成功,其它失败
*/
int YW_USBHIDFree();
/**
* 函数: YW_Buzzer
* 名称: 蜂鸣器操作函数
* 参数: ReaderID:读写器ID号,0为广播地址
* Time_ON:蜂鸣器响时间,单位0.1s
* Time_OFF:蜂鸣器不响时间,单位0.1s
* Cycle:蜂鸣器循环次数
* 返回值:>=0为成功,其它失败
*/
int YW_Buzzer(int ReaderID, int Time_ON, int Time_OFF, int Cycle);
/**
* 函数: YW_Led
* 名称: LED灯操作函数
* 参数: ReaderID:读写器ID号,0为广播地址
* LEDIndex:选择要操作的LED灯
* Time_ON: 灯亮的时间,单位0.1s
* Time_OFF:灯不亮时间,单位0.1s
* Cycle: 循环次数
* LedIndexOn:最后要亮的灯 0不亮 1 红灯 2绿灯
* 返回值:>=0为成功,其它失败
*/
int YW_Led(int ReaderID, int LEDIndex, int Time_ON, int Time_OFF, int Cycle, int LedIndexOn);
/**
* 函数: YW_AntennaStatus
* 名称: 开启天线,在所有卡操作之前必须开启天线
* 参数: ReaderID:读写器ID号,0为广播地址
* Status: true为开启天线, false为关闭天线
* 返回值:>=0为成功,其它失败
*/
int YW_AntennaStatus(int ReaderID, boolean Status);
/**
* 函数: YW_SearchCardMode
* 名称: 寻卡模式设置
* 参数: ReaderID:读写器ID号,0为广播地址
* Mode: 寻卡模式
* SEARCHMODE_14443A = Byte(Char('A'));
* SEARCHMODE_14443B = Byte(Char('B'));
* SEARCHMODE_15693 = Byte(Char('1'));
* SEARCHMODE_ST = Byte(Char('S'));
* SEARCHMODE_AT88RF020 = Byte(Char('R'));
* 详细参见说明书
* 返回值:>=0为成功,其它失败
*/
int YW_SearchCardMode(int ReaderID, int Mode);
/**
* 函数: YW_RequestCard
* 名称: 寻卡TypeA卡
* 参数: ReaderID:读写器ID号,0为广播地址
* RequestMode: 寻卡模式
* 所有卡 常数 REQUESTMODE_ALL=$52;
* 激活的卡 常数 REQUESTMODE_ACTIVE=$26;
* CardType:输出卡类型
* 0x4400 -> Ultralight/UltraLight C /MifarePlus(7Byte UID)
* 0x4200 -> MifarePlus(7Byte UID)
* 0x0400 ->Mifare Mini/Mifare 1K (S50) /MifarePlus(4Byte UID)
* 0x0200->Mifare_4K(S70)/ MifarePlus(4Byte UID)
* 0x4403 ->Mifare_DESFire
* 0x0800 ->Mifare_Pro
* 返回值:>=0为成功,其它失败
*/
int YW_RequestCard(int ReaderID, int RequestMode, short CardType[]);
/**
* 函数: YW_AntiCollide
* 名称: 访冲突操作
* 参数: ReaderID:读写器ID号,0为广播地址
* LenSNO: 输出卡号的长度
* SNO: 输出卡号
* 返回值:>=0为成功,其它失败
*/
int YW_AntiCollide(int ReaderID, byte LenSNO[], byte SNO[]);
/**
* 函数: YW_CardSelect
* 名称: 选卡
* 参数: ReaderID: 读写器ID号,0为广播地址
* LenSNO: 要选择卡卡号的长度
* SNO: 要选择卡卡号
* 返回值:>=0为成功,其它失败
*/
int YW_CardSelect(int ReaderID, byte LenSNO, byte SNO[]);
/**
* 函数: YW_DownLoadKey
* 名称: 下载秘钥到读写器中
* 参数: ReaderID: 读写器ID号,0为广播地址
* KeyIndex: 秘钥的序号,从0-31
* Key : 秘钥指针,6个字节
* 返回值:>=0为成功,其它失败
*/
int YW_DownLoadKey(int ReaderID, int KeyIndex, byte Key[]);
/**
* 函数: YW_KeyDown_Authorization
* 名称: 用读写器中的秘钥进行授权
* 参数: ReaderID: 读写器ID号,0为广播地址
* KeyMode: 秘钥选择Key A或者Key B
* 常数 PASSWORD_A = $60;
* 常数 PASSWORD_B = $61;
* BlockAddr: 要授权的块号
* KeyIndex: 秘钥的序号,从0-31
* 返回值:>=0为成功,其它失败
*/
int YW_KeyDown_Authorization(int ReaderID, int KeyMode, int BlockAddr, byte KeyIndex);
/**
* 函数: YW_KeyAuthorization
* 名称: M1卡授权
* 参数: ReaderID: 读写器ID号,0为广播地址
* KeyMode: 秘钥选择Key A或者Key B
* 常数 PASSWORD_A = $60;
* 常数 PASSWORD_B = $61;
* BlockAddr: 要授权的块
* Key: 秘钥字节指针,6字节
* 返回值:>=0为成功,其它失败
*/
int YW_KeyAuthorization(int ReaderID, int KeyMode, int BlockAddr, byte[] Key);
/**
* 函数: YW_ReadaBlock
* 名称: 读取M1卡一个块
* 参数: ReaderID: 读写器ID号,0为广播地址
* BlockAddr: 要读取的块号
* LenData: 要读取的字节数,最大为16
* Data: 数据指针
* 返回值:>=0为成功,其它失败
*/
int YW_ReadaBlock(int ReaderID, int BlockAddr, int LenData, byte pData[]);
/**
* 函数: YW_WriteaBlock
* 名称: 写入M1卡一个块
* 参数: ReaderID: 读写器ID号,0为广播地址
* BlockAddr: 要写入的块号
* LenData: 要读取的字节数,必须为16
* Data: 数据指针
* 返回值:>=0为成功,其它失败
*/
int YW_WriteaBlock(int ReaderID, int BlockAddr, int LenData, byte pData[]);
/**
* 函数: YW_Purse_Initial
* 名称: M1卡将某一块初始化钱包
* 参数: ReaderID: 读写器ID号,0为广播地址
* BlockAddr: 要初始化钱包的块号
* IniValue: 钱包初始化值
* 返回值:>=0为成功,其它失败
*/
int YW_Purse_Initial(int ReaderID, int BlockAddr, int IniMoney);
/**
* 函数: YW_Purse_Read
* 名称: 读取M1卡某个块的钱包值
* 参数: ReaderID: 读写器ID号,0为广播地址
* BlockAddr: 要初始化钱包的块号
* Value: 钱包的当前值
* 返回值:>=0为成功,其它失败
*/
int YW_Purse_Read(int ReaderID, int BlockAddr, int Money[]);
/**
* 函数: YW_Purse_Decrease
* 名称: 对钱包进行减值操作
* 参数: ReaderID: 读写器ID号,0为广播地址
* BlockAddr: 要初始化钱包的块号
* Decrement: 要减去的值
* 返回值:>=0为成功,其它失败
*/
int YW_Purse_Decrease(int ReaderID, int BlockAddr, int Decrement);
/**
* 函数: YW_Charge
* 名称: 对钱包进行加值操作
* 参数: ReaderID: 读写器ID号,0为广播地址
* BlockAddr: 要初始化钱包的块号
* Charge: 要增加的值
* 返回值:>=0为成功,其它失败
*/
int YW_Purse_Charge(int ReaderID, int BlockAddr, int Charge);
/**
* 函数: YW_CardHalt
* 名称: 对M1卡进行Halt操作
* 参数: ReaderID: 读写器ID号,0为广播地址
* 返回值:>=0为成功,其它失败
*/
int YW_CardHalt(int ReaderID);
/**
* 函数: YW_Restore
* 名称: 对钱包进行Restor操作
* 参数: ReaderID: 读写器ID号,0为广播地址
* BlockAddr: 钱包的块号
* 返回值:>=0为成功,其它失败
*/
int YW_Restore(int ReaderID, int BlockAddr);
/**
* 函数: YW_Transfer
* 名称: 对钱包进行Transfer操作
* 参数: ReaderID: 读写器ID号,0为广播地址
* BlockAddr: 钱包的块号
* 返回值:>=0为成功,其它失败
*/
int YW_Transfer(int ReaderID, int BlockAddr);
/**
* 函数: YW_AntiCollideAndSelect
* 名称: 对M1卡进行防碰撞并选卡
* 参数: ReaderID: 读写器ID号,0为广播地址
* MultiCardMode:对多张卡的处理方式
* 00 返回多卡错误
* 01 返回一张卡片
* CardMem:返回卡的内存代码
* SNOLen: 输出卡号的长度
* SNO:卡的序列号
* 返回值:>=0为成功,其它失败
*/
int YW_AntiCollideAndSelect(int ReaderID, byte ReturnMode, byte CardMem[], byte SNLen[], byte SN[]);
}
3、读写实现,dll库文件看你自己放在什么位置了,我放在了项目根目录下lib里了,这个需要根据你自己的现实的情况而定。
import com.sun.jna.Native;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.time.LocalTime;
import java.time.format.DateTimeFormatter;
import java.util.Calendar;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
/**
* @author liuxiaobao
* @date 2022/6/22 9:21
*/
public class RWUtils {
public static RWOptionService INSTANCE;//动态库声明
/**
* 说明:通过静态代码块实例化动态库,并且判断加载 x32 dll还是x64 dll 库
* */
static {
String bits = System.getProperty("sun.arch.data.model");// 获取jdk位数
System.out.println("jdk 位数:" + bits);
String ops = System.getProperty("os.name"); // 获取os名称
System.out.println("操作系统:" + ops);
if (ops.startsWith("win") || ops.startsWith("Win"))//windows
{
if ("32".equals(bits)) {
System.out.println("当前程序加载了 YW60X-32.dll");
System.load(System.getProperty("user.dir") + "\\lib\\YW60X-32.dll");
INSTANCE = (RWOptionService) Native.loadLibrary("YW60X-32", RWOptionService.class);
} else {
System.out.println("当前程序加载了 YW60X-64.dll");
System.load(System.getProperty("user.dir") + "\\lib\\YW60X-64.dll");
INSTANCE = (RWOptionService) Native.loadLibrary("YW60X-64", RWOptionService.class);
}
} else {//其他操作系统
System.out.println("当前仅支持Windows系统,暂不支持其他操作系统!");
}
}
/**
* 写卡方法 from liuxiaobao
*
* @param readerID 读卡器id
* @param requestmode 请求模式
* @param cardtype 卡类型
* @param lenSn 卡号长度
* @param cardNo 卡号
* @param key 秘钥
* @param blockData 块数据
* @param blockAddr 块
* @param buzzerAndLed 是否开启声音和led灯
* @return boolean (true:成功 false:失败 )
*/
public static boolean write(int readerID, int requestmode, short[] cardtype, byte[] lenSn, byte[] cardNo, byte[] key, byte[] blockData, int blockAddr, boolean buzzerAndLed) {
INSTANCE.YW_USBHIDInitial();
int rt = 0;
rt = INSTANCE.YW_RequestCard(readerID, requestmode, cardtype);
rt = INSTANCE.YW_AntiCollide(readerID, lenSn, cardNo);
rt = INSTANCE.YW_CardSelect(readerID, lenSn[0], cardNo);
rt = INSTANCE.YW_KeyAuthorization(readerID, 96, blockAddr, key);
System.out.println();
rt = INSTANCE.YW_WriteaBlock(readerID, blockAddr, 16, blockData);
if (buzzerAndLed) {
INSTANCE.YW_Led(readerID, 1, 1, 1, 3, 2);
INSTANCE.YW_Buzzer(readerID, 1, 1, 2);
}
INSTANCE.YW_USBHIDFree();
if (rt > 0) {
//写卡完成
System.out.println("写入成功,写入的数据为:" + byteArrayToInt(blockData, 0, true));
return true;
}
//写卡失败
return false;
}
/**
* 读卡方法 from liuxiaobao
*
* @param readerID 读卡器id
* @param requestmode 请求模式
* @param cardtype 卡类型
* @param lenSn 卡号长度
* @param cardNo 卡号
* @param key 秘钥
* @param blockData 块数据
* @param blockAddr 块
* @param buzzerAndLed 是否开启声音和led灯
* @return map (cardNo:十进制卡号 data:十进制数据 )
*/
public static Map reader(int readerID, int requestmode, short[] cardtype, byte[] lenSn, byte[] cardNo, byte[] key, byte[] blockData, int blockAddr, boolean buzzerAndLed) {
Map<String, Integer> resultMap = new HashMap<>();
INSTANCE.YW_USBHIDInitial();
int rt = 0;
rt = INSTANCE.YW_RequestCard(readerID, requestmode, cardtype);
rt = INSTANCE.YW_AntiCollide(readerID, lenSn, cardNo);
resultMap.put("cardNo", byteArrayToInt(cardNo, 0, true));
rt = INSTANCE.YW_CardSelect(readerID, lenSn[0], cardNo);
rt = INSTANCE.YW_KeyAuthorization(readerID, 96, blockAddr, key);
rt = INSTANCE.YW_ReadaBlock(readerID, blockAddr, 16, blockData);
System.out.println();
if (rt > 0) {
resultMap.put("data", byteArrayToInt(blockData, 0, true));
}
if (rt > 0 && buzzerAndLed) {
INSTANCE.YW_Led(readerID, 1, 1, 1, 3, 2);
INSTANCE.YW_Buzzer(readerID, 2, 2, 1);
}
INSTANCE.YW_USBHIDFree();
return resultMap;
}
public static void main(String args[]) {
/* int readerID = 1001;//读卡器id随意整型
int requestmode = 38;
short cardtype[] = new short[1];
byte lenSn[] = new byte[1];
byte cardNo[] = new byte[8];
byte Key[] = new byte[]{(byte) 0xFF, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF};
byte blockData[] = new byte[16];
int blockAddr = 5;
int eNo = 88188;*/
Calendar instance = Calendar.getInstance();
DateFormat dateFormat = new SimpleDateFormat("yyyyMMdd");
System.out.println(dateFormat.format(instance.getTime()));
/*
System.out.println(getTimeAdd());*/
/*
byte[] bytes = intToByteArray(eNo, true);
for (int i=0;i<bytes.length;i++) {
blockData[i]=bytes[i];
}
boolean flag = write(readerID, requestmode, cardtype, lenSn, cardNo, Key, blockData, blockAddr, true);
if(true){
int i = byteArrayToInt(blockData, 0, true);
System.out.println("eNoInt:"+i);
System.out.println("写入成功,写入的数据为:"+i);
}*/
/* while (true) {
Map reader = reader(readerID, requestmode, cardtype, lenSn, cardNo, Key, blockData, blockAddr, true);
System.out.println("cardNo:" + reader.get("cardNo"));
System.out.println("data:" + reader.get("data"));
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}*/
}
public static int getTimeAdd(){
LocalTime now = LocalTime.now();
DateTimeFormatter dtf = DateTimeFormatter.ofPattern("HHmm");
String timeStr = now.format(dtf);
if(!timeStr.isEmpty()){
String times[]= timeStr.split(":");
return Integer.parseInt(times[0])+Integer.parseInt(times[1]);
}
return -1;
}
/**
* 整型转byte数组 from liuxiaobao
*
* @param i 整数
* @param isBig true:大端模式 false:小端模式
* @return 数组
*/
public static byte[] intToByteArray(int i, boolean isBig) {
byte[] result = new byte[4];
// 由高位到低位
if (isBig) {
result[0] = (byte) ((i >> 24) & 0xFF);
result[1] = (byte) ((i >> 16) & 0xFF);
result[2] = (byte) ((i >> 8) & 0xFF);
result[3] = (byte) (i & 0xFF);
} else {
result[3] = (byte) ((i >> 24) & 0xFF);
result[2] = (byte) ((i >> 16) & 0xFF);
result[1] = (byte) ((i >> 8) & 0xFF);
result[0] = (byte) (i & 0xFF);
}
return result;
}
/**
* byte数组转整型 from liuxiaobao
*
* @param bytes byte数组
* @param start 起始位
* @param isBig true:大端模式 false:小端模式
* @return 整数
*/
public static int byteArrayToInt(byte[] bytes, int start, boolean isBig) {
int value = 0;
if (isBig) {
value += (bytes[start] & 0x000000FF) << 24;
value += (bytes[start + 1] & 0x000000FF) << 16;
value += (bytes[start + 2] & 0x000000FF) << 8;
value += (bytes[start + 3] & 0x000000FF);
} else {
value += (bytes[start] & 0x000000FF);
value += (bytes[start + 1] & 0x000000FF) << 8;
value += (bytes[start + 2] & 0x000000FF) << 16;
value += (bytes[start + 3] & 0x000000FF) << 24;
}
return value;
}
}