手把手教你做stm32简易串口上位机(接收数据用)

8 篇文章 0 订阅

笔者的开发板是正点原子的stm32F103zet6迷你板。串口的使用是USART1.单片机相关串口的程序就不讲解,编写上位机程序是使用C++语言,在VS2017里面编写,下面进入正题。

一、相关知识

大家可以先参考一下这篇blog,C++串口通信里面详细讲解了C++串口的相关知识,以及一些函数的讲解。
下面我也会根据他的blog再讲解。

二、实现过程

1、打开串口:
使用函数:HANDLE CreateFile();

HANDLE CreateFile(
LPCTSTR  lpFileName
DWORD   dwDesiredAccess
DWORD   dwSharedMode
LPSECURITY_ATTRIBUTES  lpSecurityAttributes
DWORD   dwCreationDisposition
DWORD   dwFlagsAndAttributes
HANDLE   hTemplateFile
)

LPCTSTR lpFileName :串口的名字,不同位置的usb接口都有一个名字,通常是写成“COM4”,有一些要写成 L"COM4";加不加L取决于vs项目属性-常规-字符集选的是多字节字符集还是Unicode字符集,选多字节字符集则不用L。

dwDesiredAccess:将串行口指定为“读访问权限”、“写访问权限”或“读写访问权限”。可选GENERIC_READ 、GENERIC_WRITE、 GENERIC_READ | GENERIC_WRITE

dwShareMode:指定共享属性,由于串口不能共享,该参数必须置为0;
(PS:所谓共享属性,是指一个物理串口的数据给多个应用程序使用或串口使用,一般来说,串口是独占方式打开的,有且只有一个应用实例能对一个串口进行打开、读写操作。例如COM1是输入串口,从COM1口读出的数据可以供COM2、COM3等使用,也就是共享。)

lpSecurityAttributes:引用安全性属性结构,缺省值为NULL;
dwCreationDistribution:创建标志,对串口操作该参数必须置为OPEN_EXISTING;

dwFlagsAndAttributes:属性描述,用于指定该串口是否进行异步操作,该值为FILE_FLAG_OVERLAPPED,表示使用异步的I/O;该值为0,表示同步I/O操作;这里因为是跟stm32通信,我们选择FILE_ATTRIBUTE_NORMAL

hTemplateFile:对串口而言该参数必须置为NULL。

以下是应用的一个例子:

HANDLE hcom;//全局变量串口通信
hcom = CreateFile("COM9", GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
	if (hcom == INVALID_HANDLE_VALUE)
	{
		fprintf(stderr, "打开串口失败!\n");
		exit(0);
	}

2、设置串口的属性。

BOOL  GetCommState(
      HANDLE hFile
      LPDCB  lpDCB
);

GetCommState函数的第一个参数hFile是由CreateFile函数返回指向已打开串行口的句柄。第二个参数指向设备控制块DCB。DCB是一个非常重要的数据结构,几乎所有的串行口属性和状态都存储在该结构的成员变量中。

HANDLE hFile:填写刚刚建立的串口句柄。
LPDCB lpDCB:定义一个dcb, 第二个参数指向设备控制块DCB。如果函数调用成功,则返回值为非0;若函数调用失败,则返回值为0。

应用实例:

DCB dcb;
GetCommState(hcom, &dcb);

3、设置发送和接收缓冲区

BOOL  SetupComm(
     HANDLE hFile
     DWORD dwInQueue
     DWORD dwOutQueue
);

当一个串行口打开时,可以为该串口分配一个发送缓冲区和一个接收缓冲区。串行口发送缓冲区和接收缓冲区的配置可以由函数SetupComm实现。如果不调用SetupComm,系统会为该串口分配默认的发送缓冲区和接收缓冲区。但是为了保证缓冲区的大小与实际需要的一致,最好调用该函数进行设置。
这里我们是这样设置的:

SetupComm(hcom, 1024, 1024);//设置缓冲区大小

4、设置波特率,奇偶校验这些

dcb.BaudRate = 9600;//波特率
	dcb.ByteSize = 8;
	dcb.Parity = 0;
	dcb.StopBits = 1;

5、串行数据的发送和接收
接收:
利用ReadFile函数可以读取将串行口接收到的数据。ReadFile函数原型如下:

BOOL  ReadFile(
HANDLE  hFile
LPVIOD   lpBuffer
DWORD   nNumberOfBytesToRead
LPDWORD   lpNumberOfBytesRead
LPOVERLAPPED  lpOverlapped
);

HANDLE hFile:hFile指向已经打开的串行口句柄;
lpBuffer:指向一个读取数据缓冲区;nNumberOfBytesToRead:指定要从串行设备中读取的字节数;
lpNumberOfBytesRead:指明实际从串行口中读出的字节数;
lpOverlapped指向一个OVERLAPPED结构变量,该结构变量中包含一个同步事件。

例子:

unsigned char lpBuffer[2];//设置的要接收的数据
		DWORD dwBytesRead = 2;//设置实际接收的数据
		if (ReadFile(hcom, lpBuffer, dwBytesRead, &dwBytesRead, NULL))//这一句其实就已经读好了数据。
		{
			tmp1 = lpBuffer[0] - '0';
			tmp2 = lpBuffer[1] - '0';
			PreY = Y;
			Y = tmp1 * 10 + tmp2;
			//printf("接收数据成功!\n");
		}

因为我在stm32中发送int型的数据是用printf(“%d”,a);这样的,所以发过来就是一个一个的字符;比如a=10;那么接收到的数据就存在lpBuffer[2]中;lpBuffer[0]=‘1’,lpBuffer[1]=‘0’;

			tmp1 = lpBuffer[0] - '0';
			tmp2 = lpBuffer[1] - '0';
			PreY = Y;
			Y = tmp1 * 10 + tmp2;

所以上面这段我是用来把字符型转化为整型。
数据发送的我就不讲解了,文章后面会附上参考的blog,里面有详细的讲解。

以下的代码是我做了一个心率检测的项目的小作品,通过STM32将心率值传输到电脑,还使用了一个easyx图形库,使程序更加美观。如下图:在这里插入图片描述
easyx的使用很简单,想做上位机但是不会C#,qt,labview的,可以试一试。

#include<iostream>
#include<windows.h>
#include"easyx.h"
#include <graphics.h>
#include <conio.h>
#include<time.h>
#include<stdlib.h>
#include <mmsystem.h>
#pragma comment(lib,"winmm.lib")
using namespace std;

const int HIGHT = 313;//窗口的高
const int WIDTH = 500;//窗口的宽
const int X_hight = 293;//x轴的高
IMAGE background0, background2, background1;
HANDLE hcom;//全局变量串口通信
int tmp1,tmp2,PreY=0,Y=0,X=18,ImageFlag;//对应于接收到的十位和各位
bool word_flag=TRUE;//用来改变字体


DWORD WINAPI playMusic(LPVOID lpParamer)//重新开一个线程
{
	mciSendString("open heart.mp3", 0, 0, 0);
	mciSendString("play heart.mp3 wait", 0, 0, 0);
	mciSendString("close heart.mp3", 0, 0, 0);

	return 0;
}


void BackGround()//加载背景
{
	loadimage(&background0, "b0.jpg");	
	loadimage(&background1, "b11.jpg");	
	loadimage(&background2, "b2.jpg");
	//putimage(0, 0, &background2);
}


void WordStyle()//改变字体
{
	if (word_flag)
	{
		settextstyle(35, 0, _T("宋体"));//输出文本
		word_flag=!word_flag;
	}
	else
	{
		settextstyle(35, 0, _T("华文楷体"));//输出文本
		word_flag = !word_flag;
	}
}


void Mouse()//鼠标检测的
{
	while (MouseHit())
	{
		MOUSEMSG m;
		m = GetMouseMsg();
		switch (m.uMsg)
		{
			case WM_LBUTTONDOWN:putimage(0, 0, &background0); X = 18;  break;
			case WM_MBUTTONDOWN:putimage(0, 0, &background1); X = 18;  break;
			case WM_RBUTTONDOWN:putimage(0, 0, &background2); X = 18; break;
			case  WM_MOUSEWHEEL: WordStyle(); break;
			//default:break;
		}

	}
	
}


int main()
{
	initgraph(WIDTH, HIGHT);// 绘图窗口初始化

	BackGround();
	*********************************//串口通信初始化//*************************************************************
	hcom = CreateFile("COM9", GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
	if (hcom == INVALID_HANDLE_VALUE)
	{
		fprintf(stderr, "打开串口失败!\n");
		exit(0);
	}
	SetupComm(hcom, 1024, 1024);//设置缓冲区大小
	DCB dcb;
	GetCommState(hcom, &dcb);
	dcb.BaudRate = 9600;//波特率
	dcb.ByteSize = 8;
	dcb.Parity = 0;
	dcb.StopBits = 1;
//SetCommState(hcom, &dcb);

	while (1)
	{
		//*************************************读取数据*********************************************
		unsigned char lpBuffer[2];
		DWORD dwBytesRead = 2;
		if (ReadFile(hcom, lpBuffer, dwBytesRead, &dwBytesRead, NULL))
		{
			tmp1 = lpBuffer[0] - '0';
			tmp2 = lpBuffer[1] - '0';
			PreY = Y;
			Y = tmp1 * 10 + tmp2;
			//printf("接收数据成功!\n");
		}
		else
		{
			TCHAR a[] = _T("串口异常,请检查设备");
			outtextxy(150, 50, a);
			Sleep(2000);
			break;
		}

		//*************************************画曲线部分*********************************************
		setlinestyle(PS_SOLID | PS_JOIN_BEVEL, 2);
		line(X, X_hight - PreY, X + 5, X_hight - Y);

		Mouse();//鼠标检测;

		if (Y > 90)
		{
			CreateThread(NULL, NULL, playMusic, NULL, NULL, NULL);
			settextcolor(RED);
		}
		else
		{
			settextcolor(WHITE);
		}
		TCHAR s[25];
		if (word_flag)
			_stprintf_s(s, _T("当前心率:  %d  "), Y);//转化为字符串	
		else
			_stprintf_s(s, _T("当前心率:%d"), Y);

		outtextxy(150, 50, s);//输出字符串

		if (X <= 500)//横坐标移动
		{
			X += 5;
		}
		else
		{
			X = 0;
			cleardevice();//清屏幕
		}	
	}
	return 0;
}

三、参考资料:

https://blog.csdn.net/uncle123456/article/details/84716169

https://blog.csdn.net/XTUPWM/article/details/88395249

这两位博主讲的都很好,我很多也是从他们那里学来的,哈哈哈!
另外:需要心率检测模块MAX30102的资料/源码的,
自取啦:
链接:https://pan.baidu.com/s/1ELyAaQ5-Sn_KHfRWK90fmQ
提取码:mx9r
复制这段内容后打开百度网盘手机App,操作更方便哦

  • 24
    点赞
  • 230
    收藏
    觉得还不错? 一键收藏
  • 15
    评论
### 回答1: 正点原子上位机xcom是一款基于正点原子模块和微型单片机技术开发的上位机综合开发工具,具备实时显示、数据采集、控制、算法分析等功能。其主要特点如下: 1. 兼容性强:支持多种主流单片机开发板,如STM32、51单片机等;同时,还可以与多种常见的传感器和执行器相结合,方便进行物联网项目的开发。 2. 界面友好:xcom采用图形化界面设计,用户可以直观地操作各种工具,大大提高了开发效率。 3. 功能丰富:xcom提供多种数据采集手段、数据处理算法,同时支持大量传感器测量范围、采集频率、精度等参数设置,可以满足多种应用场景需求。 4. 易于使用:采用非常简单的拖拉式操作方式,用户只需简单设置参数即可完成各种控制、采集和算法分析任务。 总之,正点原子上位机xcom提供了简单、易用、功能丰富的开发工具,可以帮助开发者快速、高效地完成物联网项目的开发和调试工作,是物联网开发必备的工具之一。 ### 回答2: 正点原子上位机xcom是一种基于ATMEL MCU开发的高性能单片机。它具有可编程性、可扩展性与多样化功能,在工业控制、仪器仪表、机器人、智能家居等领域广泛应用。xcom在设计上采用开放式标准接口,可以通过USB和以太网等方式与外部设备或网络进行通信。 xcom的硬件设计基于ARM Cortex-M系列核心,配备了高速的DMA控制器和SRAM缓冲,可以为复杂的实时控制和数据处理提供强大的计算能力。同时,xcom还具有可编程的GPIO和PWM接口,可用于控制各种外部设备,并具有多路串口和CAN总线接口,方便与其他设备进行通讯。 xcom的软件设计采用开放的C/C++开发环境,配套丰富的库函数和代码示例,可快速开发各种应用。同时,xcom支持多种操作系统和编程语言,如Linux、RTOS、Python等,可以根据应用需求选择最适合的开发环境。另外,xcom还具有可靠性、可用性和可维护性等特点,并支持远程升级和配置,使其具有更高的实用性和可扩展性。 总之,正点原子上位机xcom是一款高性能、可扩展、可编程的单片机,具有广泛的应用场景和深远的发展前景,将成为未来智能化设备控制和数据处理的核心技术之一。 ### 回答3: 正点原子上位机xcom,是一款基于STM32芯片的嵌入式开发板,也是国内知名的开源硬件品牌——正点原子推出的一款产品。它除了具备传统嵌入式开发板的各种通用接口,还带有TFT彩色屏幕、摄像头、WIFI、蓝牙等功能模块,可以直接驱动,方便用户快速搭建各种嵌入式应用系统。同时,这款开发板的使用非常简单,初学者也能轻松上手,快速实现自己的IOT创意。 选择正点原子上位机xcom的原因在于它非常适合我自己的需求。我擅长使用C语言进行嵌入式编程,又希望自己的项目能够实现一些高级功能,比如WIFI控制、摄像头实时监测等等。正点原子上位机xcom提供了一整套完备的示例代码和开发工具,极大地便利了我的开发过程。此外,它的价格也相当亲民,比同类硬件产品更加经济实惠,也符合我个人的预算要求。 总的来说,正点原子上位机xcom是一款适合广大嵌入式编程爱好者和IOT创客使用的硬件产品,具有易上手、功能强大、性价比高等优点。尤其对于初学者来说,这款开发板能够带他们深入了解嵌入式系统设计、掌握最新的物联网技术,从而促进个人技能的提升和发展。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 15
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值