Java TCP服务端多线程接收RFID网络读卡器上传数据

news/2024/5/17 15:14:04 标签: java, TCP, Server, 多线程, 服务端

本示例使用设备介绍:WIFI/TCP/UDP/HTTP协议RFID液显网络读卡器可二次开发语音播报POE-淘宝网 (taobao.com)

java">import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Arrays;


public class TCPServer_Java {
    //监听端口
    private static final int PORT = 39169;

    public static void main(String[] args) throws IOException {
        ServerSocket serverSocket = null;
        Socket socket = null;
        try {
            //建立服务器的Socket,并设定一个监听的端口PORT
            serverSocket = new ServerSocket(PORT);
            //由于需要进行循环监听,因此获取消息的操作应放在一个while大循环中
            while(true){
                try {
                    //建立跟客户端的连接
                    socket = serverSocket.accept();
                    ServerThread thread = new ServerThread(socket);
                    thread.start();
                } catch (Exception e) {
                    System.out.println("建立与客户端的连接出现异常");
                    e.printStackTrace();
                }
                //ServerThread thread = new ServerThread(socket);
                //thread.start();
            }
        } catch (Exception e) {
            System.out.println("端口被占用");
            e.printStackTrace();
        }
        finally {
            serverSocket.close();
        }
    }
}

//服务端线程类
//继承Thread类的话,必须重写run方法,在run方法中定义需要执行的任务。
class ServerThread extends Thread {
    private Socket socket ;
    InputStream inputStream;
    OutputStream outputStream;

    public ServerThread(Socket socket){
        this.socket=socket;
    }

    public void run(){
        try {
            while (true){
                //接收客户端的消息,解析信息并回应读卡器
                //System.out.println(socket);
                byte[] bytes =new byte[1024];
                inputStream =socket.getInputStream();
                int GetDataLen=0;
                while ((GetDataLen=inputStream.read(bytes))!=-1){       //通过这个方法读取全部数据 及 长度,
                    break;
                }

                if(GetDataLen>0) {
                    String bytestr = "";
                    String DataStr = "";
                    String DispStr = "";
                    String CardNo16 = "";
                    String SerialNum = "";
                    long cardno10;
                    long cardnum;

                    for (int p = 0; p < GetDataLen; p++) {
                        bytestr = "00" + Integer.toHexString(bytes[p] & 0xff);
                        DataStr = DataStr + bytestr.substring(bytestr.length() - 2, bytestr.length()) + " ";
                    }
                    System.out.println("Received from " + socket.getRemoteSocketAddress() + " : " + DataStr);

                    switch (bytes[0]) {
                        case (byte)0xc1:
                        case (byte)0xcf:
                            if(bytes[0]==(byte)0xc1){
                                DispStr = "数据解析:接收到读取IC卡号信息,";
                            }else{DispStr = "数据解析:接收到IC卡离开读卡器,";}
                            DispStr = DispStr + "功能码:" + Integer.toHexString(bytes[0] & 0xff);
                            DispStr = DispStr + ",设备IP:" + Integer.toString(bytes[1] & 0xff) + "." + Integer.toString(bytes[2] & 0xff) + "." + Integer.toString(bytes[3] & 0xff) + "." + Integer.toString(bytes[4] & 0xff);
                            DispStr = DispStr + ",机号:" + Integer.toString((bytes[5] & 0xff) + ((bytes[6] & 0xff) * 256));
                            DispStr = DispStr + ",数据包序号:" + Integer.toString((bytes[7] & 0xff) + ((bytes[8] & 0xff) * 256));
                            int Cardlen = bytes[9];
                            DispStr = DispStr + ",卡号长度:" + Integer.toString(Cardlen);
                            CardNo16 = "";
                            for (int p = 10; p < 10 + Cardlen; p++) {
                                bytestr = "00" + Integer.toHexString(bytes[p] & 0xff);
                                CardNo16 = CardNo16 + bytestr.substring(bytestr.length() - 2, bytestr.length());
                            }
                            DispStr = DispStr + ",16进制卡号:" + CardNo16;
                            cardnum = bytes[10] & 0xff;
                            cardnum = cardnum + (bytes[11] & 0xff) * 256;
                            cardnum = cardnum + (bytes[12] & 0xff) * 65536;
                            cardnum = cardnum + (bytes[13] & 0xff) * 16777216;
                            cardno10 = 0;
                            for (int j = 28; j >= 0; j -= 4) {
                                cardno10 = cardno10 << 4 | (cardnum >>> j & 0xF);
                            }
                            DispStr = DispStr + ",转10进制卡号:" + String.format("%010d", cardno10);

                            SerialNum = "";
                            for (int p = 10 + Cardlen; p < GetDataLen; p++) {
                                bytestr = "00" + Integer.toHexString(bytes[p] & 0xff);
                                SerialNum = SerialNum + bytestr.substring(bytestr.length() - 2, bytestr.length());
                            }
                            DispStr = DispStr + ",唯一硬件序号:" + SerialNum;
                            System.out.println(DispStr + "\n");

                            //向客户端发送消息
                            byte[] RespByte=GetResponseData(2); //生成不同的回应数据包

                            outputStream = socket.getOutputStream();
                            outputStream.write(RespByte);
                            bytestr = "";
                            DataStr = "";
                            for (int p = 0; p < RespByte.length; p++) {
                                bytestr = "00" + Integer.toHexString(RespByte[p] & 0xff);
                                DataStr = DataStr + bytestr.substring(bytestr.length() - 2, bytestr.length()) + " ";
                            }
                            System.out.println("Send Data To "+socket.getRemoteSocketAddress()+" : "+DataStr+"\n");
                            break;

                        case (byte)0xd1:
                        case (byte)0xdf:
                            if(bytes[0]==(byte)0xd1){
                                DispStr="数据解析:接收到读取ID卡号信息,";
                            }else{DispStr="数据解析:接收到ID卡离开读卡器,";}
                            DispStr=DispStr+"功能码:"+ Integer.toHexString(bytes[0] & 0xff);
                            DispStr=DispStr+",设备IP:"+  Integer.toString(bytes[1] & 0xff) + "." + Integer.toString(bytes[2] & 0xff) + "." + Integer.toString(bytes[3] & 0xff) + "." + Integer.toString(bytes[4] & 0xff);
                            DispStr=DispStr+",机号:"+  Integer.toString((bytes[5] & 0xff) + ((bytes[6] & 0xff) * 256));
                            DispStr=DispStr+",数据包序号:"+ Integer.toString((bytes[7] & 0xff) + ((bytes[8] & 0xff) * 256));
                            CardNo16="";
                            for (int p = 9; p < 14; p++) {
                                bytestr = "00" + Integer.toHexString(bytes[p] & 0xff);
                                CardNo16 = CardNo16 + bytestr.substring(bytestr.length() - 2, bytestr.length());
                            }
                            DispStr=DispStr+",16进制卡号:"+CardNo16;

                            cardnum = bytes[9] & 0xff;
                            cardnum = cardnum + (bytes[10] & 0xff) * 256;
                            cardnum = cardnum + (bytes[11] & 0xff) * 65536;
                            cardnum = cardnum + (bytes[12] & 0xff) * 16777216;
                            cardno10 = 0;
                            for (int j = 28; j >= 0; j -= 4) {
                                cardno10 = cardno10 << 4 | (cardnum >>> j & 0xF);
                            }
                            DispStr=DispStr+",转10进制卡号:"+ String.format("%010d", cardno10);

                            SerialNum="";
                            for (int p = 14; p < GetDataLen; p++) {
                                bytestr = "00" + Integer.toHexString(bytes[p] & 0xff);
                                SerialNum = SerialNum + bytestr.substring(bytestr.length() - 2, bytestr.length());
                            }
                            DispStr=DispStr+",唯一硬件序号:"+ SerialNum;
                            System.out.println(DispStr+"\n");

                            //向客户端发送消息
                            RespByte=GetResponseData(2); //生成不同的回应数据包

                            outputStream = socket.getOutputStream();
                            outputStream.write(RespByte);
                            bytestr = "";
                            DataStr = "";
                            for (int p = 0; p < RespByte.length; p++) {
                                bytestr = "00" + Integer.toHexString(RespByte[p] & 0xff);
                                DataStr = DataStr + bytestr.substring(bytestr.length() - 2, bytestr.length()) + " ";
                            }
                            System.out.println("Send Data To "+socket.getRemoteSocketAddress()+" : "+DataStr+"\n");
                            break;

                        case (byte)0xf3:
                            DispStr="数据解析:接收到设备心跳包,";
                            DispStr=DispStr+"功能码:"+Integer.toHexString(bytes[0] & 0xff);
                            DispStr=DispStr+",设备IP:"+Integer.toString(bytes[1] & 0xff)+"."+Integer.toString(bytes[2] & 0xff)+"."+Integer.toString(bytes[3] & 0xff)+"."+Integer.toString(bytes[4] & 0xff);
                            DispStr=DispStr+",机器号:"+Integer.toString((bytes[5] & 0xff)+((bytes[6] & 0xff) *256));
                            DispStr=DispStr+",包序号:"+Integer.toString((bytes[7] & 0xff)+((bytes[8] & 0xff) *256));
                            DispStr=DispStr+",心跳码:"+Integer.toHexString(bytes[9] & 0xff);
                            DispStr=DispStr+",长  度:"+Integer.toHexString(bytes[10] & 0xff);
                            DispStr=DispStr+",继电器:"+Integer.toHexString(bytes[11] & 0xff);
                            DispStr=DispStr+",按键值:"+Integer.toHexString(bytes[12] & 0xff);
                            for(int p=13;p<17;p++){
                                bytestr="00"+Integer.toHexString(bytes[p] & 0xff);
                                SerialNum=SerialNum+ bytestr.substring(bytestr.length() -2,bytestr.length());
                            }
                            DispStr=DispStr+",随机码:"+SerialNum;
                            SerialNum="";
                            for(int p=17;p<GetDataLen;p++){
                                bytestr="00"+Integer.toHexString(bytes[p] & 0xff);
                                SerialNum=SerialNum+ bytestr.substring(bytestr.length() -2,bytestr.length());
                            }
                            DispStr=DispStr+",设备序列号:"+SerialNum;
                            System.out.println(DispStr+"\n");
                            break;
                    }


                }
            }
        } catch (Exception e) {
            System.out.println("客户端主动断开连接了");
            e.printStackTrace();
        }

        //操作结束,关闭socket
        try{
            socket.close();
        }catch(IOException e){
            System.out.println("关闭连接出现异常");
            e.printStackTrace();
        }
    }

    //根据Respcode返回不同的回应数据包
    static byte[] GetResponseData(int Respcode)  throws Exception {
        byte[] RespByte= new byte[300];
        String DispStr="本次刷卡成功,感谢您的使用,再见!                   ";  //满屏显示34个字符,不足后面加空格
        byte[] DispBuf;
        switch(Respcode){
            case 0:            //驱动读卡器蜂鸣响声
                RespByte[0] = (byte) 0x96;  //蜂鸣响声的指令码
                RespByte[1] = (byte) 0x00;  //机号低位
                RespByte[2] = (byte) 0x00;  //机号高,0000表示任意机号
                RespByte[3] = (byte) 0x01;  //蜂鸣响声代码,取值范围0-12
                RespByte = Arrays.copyOf(RespByte, 4);
                break;

            case 1:            //驱动读卡器显示文字+蜂鸣响声
                DispBuf= DispStr.getBytes("gb2312"); //显示文字转字节数组
                RespByte[0]=(byte)0x5a;                          //指令码
                RespByte[1]=(byte)0x00;                          //机号低
                RespByte[2]=(byte)0x00;                          //机号高,如果高低位都为0表示任意机号
                RespByte[3]=(byte)0x02;                          //蜂鸣声代码
                RespByte[4]=(byte)0x14;                          //显示时长
                for(int i=0;i<34;i++){                           //显示文字
                    RespByte[i+5]=DispBuf[i];
                }
                RespByte = Arrays.copyOf(RespByte, 39);
                break;

            case 2:         //驱动读卡器显示文字+蜂鸣响声+继电器开关+TTS语音
                DispBuf= DispStr.getBytes("gb2312"); //显示文字转字节数组

                String SpeakStr="";                              //将要播报的TTS语音生成字节数组,最大不能超过126个字节
                SpeakStr="[v1]";                                //定义语音大小,取值范围v0 到 v16,可加在任何地方
                SpeakStr=SpeakStr+"欢迎您使用我们的网络读卡器,谢谢!";      //要播报的中文语音
                byte[] SpeakBuf= SpeakStr.getBytes("gb2312");
                int    SpeakLen=SpeakBuf.length;                //语音长度

                int SendLen=11+34+SpeakLen+4;                   //总计发送数据长度

                RespByte[0]=(byte)0x5C;                          //指令码
                RespByte[1]=(byte)0x00;                          //机号低
                RespByte[2]=(byte)0x00;                          //机号高,如果高低位都为0表示任意机号
                RespByte[3]=(byte)0x01;                          //蜂鸣声代码
                RespByte[4]=(byte)0xF0;                          //继电器代码 F0表示全部继电器 F1表示1号继电器 F2表示2号继电器

                RespByte[5]=(byte)0x20;                          //继电器开启时长 低位
                RespByte[6]=(byte)0x00;                          //继电器开启时长 高位

                RespByte[7]=(byte)0x14;                          //显示保留时间,单位为秒,为0xFF时表示永久显示
                RespByte[8]=(byte)0x00;                          //在显示屏中的哪个位置开始
                RespByte[9]=(byte)0x22;                          //显示字符串长度 为全屏显示为 34 个字符
                RespByte[10]=(byte)SpeakLen;                     //语音长度

                for(int i=0;i<34;i++){                           //显示的文字信息
                    RespByte[i+11]=DispBuf[i];
                }

                for(int i=0;i<SpeakLen;i++){                     //TTS语音播报信息
                    RespByte[i+45]=SpeakBuf[i];
                }

                RespByte[10+34+SpeakLen+1]=(byte)0x55;           //固定的抗干扰后缀
                RespByte[10+34+SpeakLen+2]=(byte)0xaa;
                RespByte[10+34+SpeakLen+3]=(byte)0x66;
                RespByte[10+34+SpeakLen+4]=(byte)0x99;
                RespByte = Arrays.copyOf(RespByte, 10+34+SpeakLen+5);
                break;

            case 3:            //驱动读卡器开启继电器开关
                RespByte[0]=(byte)0x78;                          //指令码
                RespByte[1]=(byte)0x00;                          //机号低
                RespByte[2]=(byte)0x00;                          //机号高,如果高低位都为0表示任意机号
                RespByte[3]=(byte)0xF0;                          //继电器代码 F0表示全部继电器、F1表示1号继电器 、F2表示2号继电器......
                RespByte[4]=(byte)0x2C;                          //继电器开启延时低位
                RespByte[5]=(byte)0x01;                          //继电器开启延时高位,FFFF表示继电器一直开启
                RespByte = Arrays.copyOf(RespByte, 6);
                break;

            default:
        }
        return  RespByte;
    }
}

 


http://www.niftyadmin.cn/n/5158749.html

相关文章

2023-11-06 思考-数据库行业-日记

摘要: 2023-11-06 思考-数据库行业分析-记录 所切入点: 过去的计算机理论具体的数据库mysql/mdb对于对应问题的具体代码实现 参考innodb和mdb的事务部分的分别的具体实现理论的突破 列存的向量化所采取的具体做法 算法本身, 算法导论对细分具体领域的突破 事务安全性 redo/und…

③【操作表数据】MySQL添加数据、修改数据、删除数据

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ MySQL添加数据、修改数据、删除数据 &#x1f…

推荐一款网络拓扑自动扫描工具

topology-scanner Topology-Scanner是WeOps团队免费开放的一个网络拓扑自动扫描模块&#xff0c;可以自动发现网络设备的类型、网络设备之间的互联 资源下载地址&#xff1a;https://download.csdn.net/download/XMWS_IT/88510697 或 加 薇|信|号吗&#xff1a;xmws-IT 免…

卡尔曼滤波之二:Python实现

卡尔曼滤波之二&#xff1a;Python实现 1.背景描述2.构建卡尔曼滤波公式2.1 预测2.2 更新 3.代码实现3.1 输入值3.2 pykalman包实现3.3 不使用Python包实现3.4 效果可视化 参考文献 了解了卡尔曼滤波之一&#xff1a;基本概念&#xff0c;可以结合代码来理解下卡尔曼滤波的2个预…

Doris:MySQL数据同步到Doris的N种方式

目录 1.CSV文件方式 1.1 导出mysql数据 1.2 导入数据 2.JDBC 编码方式 3.JDBC Catalog 方式 3.1 上传mysql驱动包 3.2 创建mysql catalog 3.3. 插入数据 4.Binlog Load 方式 1.CSV文件方式 当mysql与doris服务之间无法通过网络互联时&#xff0c;可以通过将mysql数据导…

Rust和Pytho写一段采集公众号代码

首先&#xff0c;我们需要安装Rust和Python的requests库。Rust的requests库可以用来发送HTTP请求&#xff0c;而Python的requests库可以用来处理HTTP响应。 // 导入所需的库 use std::io; use std::env;// 使用rustc命令来编译我们的程序 fn main() {// 获取命令行参数let args…

五年制专转本的作文评分标准及拿分方法

第一档&#xff1a;差&#xff08;1-3分&#xff09; 总体要求&#xff1a;未能完成了试题规定任务&#xff0c;未能将信息传递给读者。 传达信息或所用语言太少&#xff0c;无法评价&#xff1b;内容与要求无关或者无法辨认。 第二档&#xff1a;较差&#xff08;英语一&am…

微信小程序(非个人)备案指南

一、小程序备案法律法规参考 1、《中华人民共和国反电信网络诈骗法》 第二十三条 设立移动互联网应用程序应当按照国家有关规定向电信主管部门办理许可或者备案手续。 2、《互联网信息服务管理办法》 第四条 国家对经营性互联网信息服务实行许可制度&#xff1b;对非经营性互…