zoukankan      html  css  js  c++  java
  • vs串口读写dll封装C++#(免费源码分享)

    一、首先。老规矩。配置好自己的vs环境和建立一个dll工程。

    我的环境:

    1. vs2013
    2. win10

    二、开始写代码

    废话不多说。因为自己之前搞这个的时候,很多都要积分才可以下载,我这里就把我的整个工程分享出来供大家使用。经过测试工程读写完全正常。我在之前有发了一篇博客记录了读写的截图。
    https://blog.csdn.net/weixin_43673603/article/details/107681314
    代码的输出输入为十六进制转换,你们可以根据自己的需要去重新写一个接收到的数据格式转换函数,就可以输出自己想要的格式。

    首先自己新建两个文件,一个com.c。一个com.h。名字可以自己其吧,只不过后面自己要改下头文件引用。

    com.c内容(这个串口的配置文件)

    #include "stdafx.h"
    #include <stdio.h>
    #include "Com.h"
    
    //
    // Construction/Destruction
    //
    CCom::CCom()
    {
    	hPort = INVALID_HANDLE_VALUE;
    
    	memset(LineBuf, 0, sizeof(LineBuf));
    	dataLen = 0;
    }
    
    CCom::~CCom()
    {
    	close();
    }
    
    /**
    奇偶校验 0-4 = no,odd,even,mark,space
    每字节一位停止位 0,1,2 = 1, 1.5, 2
    OpenPort( _T("com1:"), 4800, 8, 0, 1 );
    */
    BOOL CCom::open(LPTSTR lpszPortName,
    	DWORD baudRate,
    	BYTE byteSize,
    	BYTE parity,
    	BYTE stopBits)
    {
    	if (hPort != INVALID_HANDLE_VALUE)
    	{
    		//::MessageBox(NULL, "hPort != INVALID_HANDLE_VALUE", "错误", MB_OK);
    		return FALSE;
    	}
    
    	//打开串口
    	hPort = CreateFile(lpszPortName, GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, 0, NULL);
    
    	//如果打开端口出错, 返回FALSE
    	if (hPort == INVALID_HANDLE_VALUE)
    	{
    		//不能打开端口
    		DWORD errNum = GetLastError();
    		char buf[256];
    		sprintf(buf, "Unable to open port[%s] errCode[%d]!
    ", lpszPortName, errNum);
    		//::MessageBox(NULL, buf, "Err code", MB_OK);
    		return FALSE;
    	}
    
    	//指定端口监测的事件集
    	SetCommMask(hPort, EV_ERR | EV_RXCHAR | EV_BREAK | EV_CTS | EV_DSR | EV_ERR | EV_RING | EV_RLSD | EV_RXCHAR | EV_RXFLAG | EV_TXEMPTY);
    
    	//分配设备缓冲区
    	SetupComm(hPort, 512, 32);
    
    	//初始化缓冲区中的信息
    	PurgeComm(hPort, PURGE_TXCLEAR | PURGE_RXCLEAR);
    
    	//配置串行端口
    	if (!InitDCB(baudRate, byteSize, parity, stopBits))
    	{
    		//::MessageBox(NULL, "配置串行端口", "错误", MB_OK);
    		return FALSE;
    	}
    
    	//设置端口超时值	
    	if (!InitCommTimeouts())
    	{
    		//::MessageBox(NULL, "设置端口超时值", "错误", MB_OK);
    		return FALSE;
    	}
    
    	//设置端口上指定信号的状态
    	// SETDTR: 发送DTR (data-terminal-ready)信号
    	// SETRTS: 发送RTS (request-to-send)信号
    	EscapeCommFunction(hPort, SETDTR);
    	EscapeCommFunction(hPort, SETRTS);
    
    	//clear( hPort );
    	PurgeComm(hPort, PURGE_TXABORT | PURGE_RXABORT | PURGE_TXCLEAR | PURGE_RXCLEAR);
    
    	return TRUE;
    }
    
    DWORD CCom::write(TCHAR *buf, DWORD dwCharToWrite)
    {
    	if (hPort == INVALID_HANDLE_VALUE)
    		return 0;
    
    	BOOL  fWriteState;
    	DWORD dwBytesWritten;
    
    	//写入数据
    	fWriteState = WriteFile(hPort,
    		buf,
    		dwCharToWrite * sizeof(TCHAR),
    		&dwBytesWritten,
    		NULL);
    
    	if (!fWriteState)
    	{
    		//不能写数据
    		printf("Can't Write String to Com
    ");
    		dwBytesWritten = 0;
    	}
    
    	return dwBytesWritten;
    }
    
    
    BOOL CCom::InitDCB(DWORD baudRate,
    	BYTE byteSize,
    	BYTE parity,
    	BYTE stopBits)
    {
    	DCB   PortDCB;
    	DWORD dwError;
    
    	PortDCB.DCBlength = sizeof (DCB);
    
    	//得到端口的默认设置信息
    	GetCommState(hPort, &PortDCB);
    
    	//改变DCB结构设置
    	PortDCB.BaudRate = baudRate;		//波特率 
    	PortDCB.ByteSize = byteSize;	//每字节的位数 number of bits/byte, 4-8 
    	PortDCB.Parity = parity;		//奇偶校验 0-4=no,odd,even,mark,space 
    
    	PortDCB.StopBits = (stopBits == 2) ? 2 : 0;	//每字节一位停止位 0,1,2 = 1, 1.5, 2 
    
    	//根据DCB结构配置端口 
    
    	if (!SetCommState(hPort, &PortDCB))
    	{
    		this->close();
    
    		//不能配置串行端口
    		dwError = GetLastError();
    
    		char buf[256];
    		sprintf(buf, "Unable to configure the serial port err:%d
    ", dwError);
    		//::MessageBox(NULL, buf, "Err code", MB_OK);
    
    		return FALSE;
    	}
    	return TRUE;
    }
    
    BOOL CCom::InitCommTimeouts()
    {
    	COMMTIMEOUTS CommTimeouts;
    	DWORD dwError;
    
    	//得到超时参数
    	GetCommTimeouts(hPort, &CommTimeouts);
    	//改变COMMTIMEOUTS结构设置
    
    	CommTimeouts.ReadIntervalTimeout = MAXDWORD;
    	CommTimeouts.ReadTotalTimeoutMultiplier = 0;
    	CommTimeouts.ReadTotalTimeoutConstant = 0;
    	CommTimeouts.WriteTotalTimeoutMultiplier = 10;
    	CommTimeouts.WriteTotalTimeoutConstant = 1000;
    
    	//设置端口超时值 
    	if (!SetCommTimeouts(hPort, &CommTimeouts))
    	{
    		this->close();
    
    		//不能设置超时值
    		printf("Unable to set the time-out parameters
    ");
    		dwError = GetLastError();
    		return FALSE;
    	}
    
    	return TRUE;
    }
    
    BOOL CCom::close()
    {
    	if (hPort != INVALID_HANDLE_VALUE)
    	{
    		//关闭串口
    		CloseHandle(hPort);
    		hPort = INVALID_HANDLE_VALUE;
    
    		return TRUE;
    	}
    	else
    	{
    		return TRUE;
    	}
    }
    
    DWORD CCom::read(char * buf, DWORD size)
    {
    	if (hPort == INVALID_HANDLE_VALUE)
    		return 0;
    
    	//从串口读取数据
    	if (!ReadFile(hPort, buf, size, &size, NULL))
    	{
    		//不能从串口读取数据
    		printf("Error in read from serial port
    ");
    		size = 0;
    	}
    
    	return size;
    }
    
    int CCom::readSyn(char *buf, DWORD wCount, DWORD wWaitTickCount)
    {
    	//读满 wCound 个字符
    	DWORD nReadCount = 0;
    
    	DWORD wTimeBegin = ::GetTickCount();
    
    	while (nReadCount < wCount)
    	{
    		int count = read(buf + nReadCount, wCount - nReadCount);
    		if (count < 0)
    			return -1;
    
    		if (count == 0)
    		{
    			if (wWaitTickCount > 0 && ::GetTickCount() - wTimeBegin > wWaitTickCount)
    				return 0;
    
    			//Sleep(1);
    		}
    		else
    			nReadCount += count;
    	}
    
    	return 1;
    }
    

    接下来是com.h文件(串口配置头文件)

    #include <windows.h>
    
    class CCom
    {
    public:
    	BOOL open(LPTSTR lpszPortName,
    		DWORD baudRate,
    		BYTE byteSize,
    		BYTE parity,
    		BYTE stopBits); //打开串口
    	BOOL close(); //关闭串口
    
    	int readSyn(char * buf, DWORD wCount, DWORD wWaitTickCount = 0);
    	DWORD read(char * buf, DWORD size);
    
    	DWORD write(TCHAR *buf, DWORD dwBytesToWrite); //写数据
    
    	CCom();
    	virtual ~CCom();
    
    private:
    	HANDLE hPort;
    
    	char LineBuf[256];
    	DWORD dataLen;
    
    	BOOL InitDCB(DWORD baudRate,
    		BYTE byteSize,
    		BYTE parity,
    		BYTE stopBits); //配置串口
    	BOOL InitCommTimeouts(); //设置超时参数
    };
    

    工程原本的.c文件。我的工程叫UartComdll。所以这里叫UartComdll.c

    // UartComdll.cpp : 定义 DLL 应用程序的导出函数。
    //
    
    #include "stdafx.h"
    #include "Com.h"
    #include <stdio.h>
    #include <stdlib.h>
    #include "UartCom.h"
    #include <string.h>
    #include <iostream>
    using namespace std;
    char* uartbuff;
    void delay(unsigned int xms)  // xms代表需要延时的毫秒数
    {
    	unsigned int x, y;
    	for (x = xms; x>0; x--)
    	for (y = 110000; y>0; y--);
    }
    unsigned char Hex_buf[6];                                 //继电器串口接收缓存
    unsigned char MotorHex_buf[100];                          //步进电机
    unsigned char i = 0x01;
    CCom gCom;
    BOOL gbInit = false;
    int gCurCom = 0;
    TPackaMsg boradMes;
    
    
    #define MSG_HEAD1	0xFF
    #define MSG_HEAD2	0xFE
    
    void UartClose(void)
    {
    	if (gbInit)
    	{
    		gCom.close();
    		gbInit = false;
    	}
    }
    
    
    int UartInit(int nCom)
    {
    	if (!gbInit || nCom != gCurCom)
    	{
    		char sNum[8] = { 0 };
    		char sComName[16] = "com";
    		strcat(sComName, _itoa(nCom, sNum, 10));
    
    		UartClose();
    
    		gbInit = gCom.open(sComName, 115200, 8, 0, 1);
    		if (gbInit)
    		{
    			gCurCom = nCom;
    			return 1;
    		}
    		return -1;
    	}
    
    	if (!gbInit)
    		return -1;	//打开端口失败
    	return 1;
    }
    
    
    
    
    void substring(char *s, char ch1, char ch2, char *substr)
    {
    	while (*s && *s++ != ch1);
    	while (*s && *s != ch2) *substr++ = *s++;
    	*substr = '';
    }
    
    
    //继电器串口控制发送函数
    int ControlRelaytoUart( unsigned char buf){
    	unsigned char index;
    	unsigned char checkvalue;
    	Hex_buf[0] = 0xff;
    	Hex_buf[1] = 0xfe;   //装载串口一帧数据头部
    	Hex_buf[2] = 0x01;    //板子编号
    	Hex_buf[3] = buf;     //继电器
    	Hex_buf[4] = 0xde;  //
    	checkvalue = 0;
    	for (index = 0; index < 5; index++){ checkvalue ^= Hex_buf[index]; }
    	Hex_buf[5] = checkvalue;   //装载尾部
    	gCom.write((TCHAR*)Hex_buf,sizeof(Hex_buf));
    	return 1;
    }
    
    int ControlRelay(int id, unsigned char controldata){                 //接口调用发数据给串口
    	if (id == 1){//id是板子编号
    		ControlRelaytoUart(controldata);//进入串口发送数据函数
    	}
    	return 1;
    }
    void testrelay(unsigned char num){
    	for (i; i < 17; i++){
    		ControlRelay(1, num);
    		num = num + 0x01;
    		delay(10000);
    	}
    }
    
    
    
    

    接下来就是在工程原本的头文件编写了,引出来我们的封装函数可以被第三方调用。
    UartComdll.h

    #ifndef _RS232DLL_H_
    #define _RS232DLL_H_
    
    #define Uart_API __declspec(dllexport)  
    
    #define player 4
    
    extern "C"   //输出dll中的接口,为程序所用,也为自己的测试程序所用
    {
    
    	Uart_API void UartClose(void);
    	Uart_API int UartInit(int nCom);
    	Uart_API int ControlRelay(int id, unsigned char controldata);
    	Uart_API void testrelay(unsigned char num);
    }
    
    #endif
    

    自此,全部完成。我们只需要点击

    就可以得到我们的dll和lib。在你的工程debug目录下的文件夹中就可以找到,有些是x64的,就要去x64中财可以找到。

    至于怎么引用,可以看我下一篇博客。
    https://blog.csdn.net/weixin_43673603/article/details/107855147
    如果自己根据我的博客配置还不成功的,可能是自己调用的版本问题,注意版本一致, 这个是绝对可以运行成功的,如果实在看不懂,那你花点积分下载我的资源吧,我也救不你。哈哈哈哈~~·
    资源地址:点击下载

    一键三连呀!
  • 相关阅读:
    edgecore
    十问 Linux 虚拟内存管理 (glibc)
    Covered Path
    Journey Planning
    K for the Price of One
    Candies!
    2种方式解决nginx负载下的Web API站点里swagger无法使用
    分布式环境下的数据一致性问题的方案讨论
    static,你还敢用吗?
    分离EF connectionString里的db连接串
  • 原文地址:https://www.cnblogs.com/jee-cai/p/14095322.html
Copyright © 2011-2022 走看看