STM32速成笔记—串口通信

news2025/1/11 8:19:00

文章目录

  • 一、什么是串口通讯
  • 二、串口通讯有什么用
  • 三、STM32的串口通信
  • 四、串口通信相关概念
    • 4.1 波特率
    • 4.2 全双工和半双工
    • 4.3 同步通信和异步通信
  • 五、硬件连接
  • 六、串口通讯程序配置
    • 6.1 使能串口时钟和GPIO时钟
    • 6.2 初始化GPIO
    • 6.3 初始化串口参数
    • 6.4 使能串口
    • 6.5 串口接收中断
    • 6.6 串口接收中断服务函数
    • 6.7 串口发送函数
  • 七、拓展
    • 7.1 printf重定向
    • 7.2 接收帧解析
  • 八、实战项目
    • 8.1 前期准备
    • 8.2 项目要求
    • 8.3 串口程序
      • 8.3.1 初始化串口
      • 8.3.2 串口接收中断服务函数
      • 8.3.3 接收帧解析函数
      • 8.3.3 main函数

一、什么是串口通讯

串口通信是指外部设备与主控芯片之间,通过数据信号线、地线等,按位进行数据传输的一种通信方式,属于串行通信方式。串行通信是指使用一条数据线依次逐位传输数据,每一位数据占据固定长度的时间。可以看一下简单的串行通信示意图。
串口通信示意图

二、串口通讯有什么用

这里简单列举一下串口通信的用途

  • 下载程序
  • 外设与单片机通信
    单片机给外设发送一些指令或者配置信息,外设给单片机回传一些信息。
  • 打印信息
    比如将ADC采集到的电压发送给上位机的串口调试助手,或者实时监测某一个变量的变化。

三、STM32的串口通信

普中核心板上使用的STM32F103ZET6有三个USART,两个UART,他们都支持串口通信功能。USART(通用同步异步收发器)与UART(通用异步收发器)相比,多了一个同步功能,可以认为USART是UART的增强型。

四、串口通信相关概念

4.1 波特率

引用专业的说法,波特率表示单位时间内传送的码元符号的个数,它是对符号传输速率的一种度量。其实意思就是波特率表示1s内传输码元的个数。在单片机中数字都是二进制的01表示的,所以波特率可以说是1s内传输01的个数。常见的波特率有38400、9600和115200等。

波特率通常由波特率发生器产生,串口要想实现收发首先要有波特率发生器,网上介绍波特率发生器的作用是输入时钟转换出需要的波特率CLK。个人理解,波特率发生器就是提供一个时钟,这样才能发送出正确波特率的信息,比如1和0需要多久的高/低电平表示。

在串口通信时如果收发双方波特率不相同会导致通信失败,要么是接收不到,要么是接收到的是乱码。

4.2 全双工和半双工

  • 全双工可以简单解释为,我在接收消息的同时,你也可以发送消息。
  • 半双工可以简单解释为,我在接收消息时,没办法发送消息。类似于对讲机,你说话时占用了信道,对方无法跟你讲话,只有当你说完了,他才可以对你讲话。

4.3 同步通信和异步通信

同步通信和异步通信的区别在于通信双方是否需要时钟同步。同步通信的接收双方之间除了需要数据线之外,还需要一根时钟线,而异步通信不需要。关于二者的详细定义与区别,请大家自行搜索。

五、硬件连接

串口通信只需几条线即可在两个系统间交换信息,特别适用于计算机与计算机、计算机与外设之间的通信,常用的串口通信接口标准有很多,比如RS-232C、RS-232、RS-485等。但是放在单片机开发里,最简单的串口通信就是用四根线VCC、GND、TXD和RXD实现通信。

串口通信硬件连接示意图

普中核心板上常用的是USART1,其引脚对应如下

  • TXD——PA9
  • RXD——PA10

六、串口通讯程序配置

下面以配置USART1为例,来简单展示一下USART的配置方法。

6.1 使能串口时钟和GPIO时钟

// 使能USART1,GPIOA时钟
RCC_APB2PeriphClockCmd (RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE);

6.2 初始化GPIO

初始化USART1用到的GPIO。TXD引脚设置为复用推挽式输出,RXD引脚设置为输入浮空。

// USART1_TX   GPIOA.9
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;   // PA.9
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;   // 复用推挽输出
	GPIO_Init(GPIOA, &GPIO_InitStructure);   // 初始化GPIOA.9

	// USART1_RX	  GPIOA.10初始化
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;   // PA10
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;   // 输入浮空
	GPIO_Init(GPIOA, &GPIO_InitStructure);   // 初始化GPIOA.10  

6.3 初始化串口参数

库函数提供了一个结构体,用于初始化串口。其中包括

    USART_InitTypeDef USART_InitStructure;
    
    // USART 初始化设置
	USART_InitStructure.USART_BaudRate = bound;   // 串口波特率
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;   // 字长为8位数据格式
	USART_InitStructure.USART_StopBits = USART_StopBits_1;   // 一个停止位
	USART_InitStructure.USART_Parity = USART_Parity_No;   // 无奇偶校验位
	// 无硬件数据流控制
	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
	USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;   // 收发模式
	USART_Init(USART1, &USART_InitStructure);   // 初始化串口1

6.4 使能串口

USART_Cmd(USART1, ENABLE);   // 使能串口1 

6.5 串口接收中断

平时开发过程中经常需要开启串口接收中断,配置串口接收中断的方法与上一篇的外部中断有些类似,主要包括以下步骤

  • 配置中断分组(通常在main函数中初始化中配置)
  • 设置中断优先级
  • 使能中断

配置中断优先级

    NVIC_InitTypeDef NVIC_InitStructure;

    // Usart1 NVIC 配置
	NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;   // 抢占优先级3
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;   // 子优先级3
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;   // IRQ通道使能
	NVIC_Init(&NVIC_InitStructure);   // 根据指定的参数初始化VIC寄存器

使能串口接收中断

USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);   // 开启串口接收中断

6.6 串口接收中断服务函数

通常接收到的数据会是一帧,很少是一个单独的字符,这里给出一个接收一帧数据的串口中断服务函数。需要注意的是,在初始化串口时,需要使能空闲中断。

使能空闲中断的程序如下

USART_ITConfig(USART1, USART_IT_IDLE, ENABLE);   // 使能空闲中断
/*
 *==============================================================================
 *函数名称:USART1_IRQHandler
 *函数功能:USART1中断服务函数
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
 */
u32 gReceCount = 0;   // 接收计数变量
u32 gClearCount = 0;   // 清空接收数组计数变量
u8 gReceFifo[1500];   // 接收数组
u8 gReceEndFlag = 0;   // 接收完成标志位 

void USART1_IRQHandler(void)  
{
	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)   //接收到一个字节  
	{
		gReceFifo[gReceCount++] = USART_ReceiveData(USART1);
	}
	else if(USART_GetITStatus(USART1,USART_IT_IDLE) != RESET)   //接收到一帧数据
	{
		USART1->SR;   // 先读SR
		USART1->DR;   // 再读DR
		
		gReceEndFlag = 1;   // 接收完成标志置1 
	} 
}

接收完成后,接收完成标志位会置1。此时,对接收到的帧进行解析处理。解析完成后需要清除接收数组,同时,不要忘记清除接收完成标志位。

/*
 *==============================================================================
 *函数名称:Uart_Rece_Pares
 *函数功能:解析串口接收内容
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
 */
void Uart_Rece_Pares(void)   // 串口接收内容解析函数
{
	if (gReceEndFlag  == 1)   // 如果接收完成
	{
		// 解析接收内容
		
		// 清空接收数组
		for (gClearCount = 0;gClearCount < gReceCount;gClearCount ++)
		{
			gReceFifo[gClearCount] = ' ';
		}
			
		gReceEndFlag = 0;   // 清除接收完成标志位
		gReceCount = 0;   // 清零接收计数变量
	}
}

6.7 串口发送函数

//串口发送函数
void USART1_Send(u8*str)
{
	u8 index=0;
	do
	{
		USART_SendData(USART1,str[index++]);
		while(USART1,str[index++]);
		while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);
	}
	while(str[index]!=0);
}

其实这里最根本的USART_SendData()本质就是将数据搬运到串口发送的寄存器。当然除了直接用发送函数发送,也可以直接重定向之后用printf发送,这里就不详细介绍了,有需要的友友可以直接去看普中或者正点的教程视频。

七、拓展

7.1 printf重定向

关于重定向的概念这里就不再做介绍了,重定向之后就可以在程序中使用printf直接打印或者发送字符串,不再需要串口发送函数。重定向的方法就是在串口的.c文件中添加下面的程序

// 加入以下函数可以使用printf
#pragma import(__use_no_semihosting)             
// 标准库需要的支持函数                 
struct __FILE 
{ 
	int handle; 
}; 

FILE __stdout;       
//定义_sys_exit()以避免使用半主机模式    
void _sys_exit(int x) 
{ 
	x = x; 
} 
//重定义fputc函数 
int fputc(int ch, FILE *f)
{      
	while((USART1->SR&0X40)==0);//循环发送,直到发送完毕   
    USART1->DR = (u8) ch;      
	return ch;
}

7.2 接收帧解析

这里的接收帧解析比较简单,比如有些项目要求接收到某些特定字符执行某些操作。这时需要根据接收帧的长度和固定位置的字符来解析命令。

比如项目要求上位机(电脑)发送“BEEP ON”时,蜂鸣器响。这时在解析时只要接收到长度为6,第5和第6个字符分别为“O”,“N”时,开启蜂鸣器即可。

// 解析接收内容
if (gReceCount == 6 && gReceFifo[5] == 'O' && gReceFifo[6] == 'N')
{
   // 开启蜂鸣器
}

当然上面的只是粗略的卡命令,也可以写的更详细。

八、实战项目

8.1 前期准备

  • CH340驱动
  • USB转TTL,用于单片机与电脑的通信
  • 串口调试助手

刚买来的普中核心板,不拔短接片的话可以直接通过USB下载程序,或者与电脑进行串口通信,串口为USART1。注意一定要插图中标出来的USB接口,另一个只能用来供电。
串口通信跳线帽连接

8.2 项目要求

  • 单片机上电发送“Sys Ready!”
  • 电脑串口助手发送“LED1 ON”(带回车换行),LED1点亮,同时单片机回复“OK!”
  • 电脑串口助手发送“LED1 OFF”(带回车换行),LED2熄灭,同时单片机回复“OK!”

8.3 串口程序

8.3.1 初始化串口

首先是串口初始化程序,需要开启接收中断和空闲中断。

/*
 *==============================================================================
 *函数名称:uart_init
 *函数功能:初始化USART1
 *输入参数:bound:波特率
 *返回值:无
 *备  注:可以修改成输入初始化哪个USART
 *==============================================================================
 */
void uart_init(u32 bound)
{
	// 相关结构体定义
	GPIO_InitTypeDef GPIO_InitStructure;
	USART_InitTypeDef USART_InitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;

	// 使能USART1,GPIOA时钟
	RCC_APB2PeriphClockCmd (RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA, ENABLE);	

	// USART1_TX   GPIOA.9
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;   // PA.9
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;   // 复用推挽输出
	GPIO_Init(GPIOA, &GPIO_InitStructure);   // 初始化GPIOA.9

	// USART1_RX	  GPIOA.10初始化
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;   // PA10
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;   // 浮空输入
	GPIO_Init(GPIOA, &GPIO_InitStructure);   // 初始化GPIOA.10  

	// Usart1 NVIC 配置
	NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;   // 抢占优先级3
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;   // 子优先级3
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;   // IRQ通道使能
	NVIC_Init(&NVIC_InitStructure);   // 根据指定的参数初始化VIC寄存器

	// USART 初始化设置
	USART_InitStructure.USART_BaudRate = bound;   // 串口波特率
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;   // 字长为8位数据格式
	USART_InitStructure.USART_StopBits = USART_StopBits_1;   // 一个停止位
	USART_InitStructure.USART_Parity = USART_Parity_No;   // 无奇偶校验位
	// 无硬件数据流控制
	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
	USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;   // 收发模式
	USART_Init(USART1, &USART_InitStructure);   // 初始化串口1
  
  USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);   // 开启串口接收中断
	USART_ITConfig(USART1, USART_IT_IDLE, ENABLE);   // 使能空闲中断
	
  USART_Cmd(USART1, ENABLE);   // 使能串口1 
}

其次需要加上重定向函数,直接复制上面的即可。

8.3.2 串口接收中断服务函数

/*
 *==============================================================================
 *函数名称:USART1_IRQHandler
 *函数功能:USART1中断服务函数
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
 */
u32 gReceCount = 0;   // 接收计数变量
u32 gClearCount = 0;   // 清空接收数组计数变量
u8 gReceFifo[1500];   // 接收数组
u8 gReceEndFlag = 0;   // 接收完成标志位 

void USART1_IRQHandler(void)  
{
	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)   //接收到一个字节  
	{
		gReceFifo[gReceCount++] = USART_ReceiveData(USART1);
	}
	else if(USART_GetITStatus(USART1,USART_IT_IDLE) != RESET)   //接收到一帧数据
	{
		USART1->SR;   // 先读SR
		USART1->DR;   // 再读DR
		
		gReceEndFlag = 1;   // 接收完成标志置1 
	} 
}

8.3.3 接收帧解析函数

/*
 *==============================================================================
 *函数名称:Uart_Rece_Pares
 *函数功能:解析串口接收内容
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
 */
void Uart_Rece_Pares(void)   // 串口接收内容解析函数
{
	if (gReceEndFlag  == 1)   // 如果接收完成
	{
		// 解析接收内容
		if (gReceFifo[6] == 'N')
		{
			Med_Led_StateCtrl (LED1,LED_ON);   // 点亮LED1
			printf ("OK!\r\n");
		}
		
		if (gReceFifo[6] == 'F' && gReceFifo[7] == 'F')
		{
			Med_Led_StateCtrl (LED1,LED_OFF);   // 熄灭LED1
			printf ("OK!\r\n");
		}
		
		// 清空接收数组
		for (gClearCount = 0;gClearCount < gReceCount;gClearCount ++)
		{
			gReceFifo[gClearCount] = ' ';
		}
			
		gReceEndFlag = 0;   // 清除接收完成标志位
		gReceCount = 0;   // 清零接收计数变量
	}
}

8.3.3 main函数

int main(void)
{
	Med_Mcu_Iint();   // 系统初始化
	printf ("Sys Ready!\r\n");
	
	while(1)
  {
		Uart_Rece_Pares ();   // 接收帧解析
	}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/655788.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

3ds MAX 多维材质

有的时候&#xff0c;我们想在一个图形中添加两个材质 比如我们绘制了一个球体&#xff0c;想要表示这是足球&#xff0c;但是没有黑白方块的贴图 除了绘制一个贴图外&#xff0c;我们可以用多维材质直接实现。 这里给茶壶对象的盖子和壶身设置两种贴图&#xff1a; 首先打…

MVI设计模式

一.各种框架对比 https://blog.csdn.net/qq_36390114/article/details/126160017 1. MVC&#xff08;Model-View-Controller&#xff09; 模型-视图-控制器 MVC的目的就是为了M和V代码分离&#xff0c;降低耦合性。 Model&#xff1a;数据来源&#xff0c;网络请求数据和数据…

DJ8-1 shell 的启动和终止、重定向、管道

目录 8.1 shell 的启动和终止 8.2 输入输出重定向 8.2.0 标准输入输出 8.2.1 输出重定向 > 8.2.2 输入重定向 < 8.2.3 常见输入输出重定向形式 8.2.4 标准错误输出重定向 8.3 管道 Linux 系统中的 shell 具有两大功能&#xff1a; 是一个命令解释器&…

express框架学习笔记

express简介 express是一个基于Node.js平台的极简的、灵活的WEB应用开发框架。express是一个封装好的工具包&#xff0c;封装了很多功能&#xff0c;便于我们开发WEB应用&#xff08;HTTP服务&#xff09; express使用 新建express文件夹新建文件test01.js&#xff0c;代码如…

特征点Features2D类介绍

文章目录 Features2D类介绍1. cv::AgastFeatureDetector2. cv::AKAZE3. cv::BRISK4. cv::FastFeatureDetector5. cv::GFTTDetector6. cv::KAZE7. cv::MSER8. cv::SimpleBlobDetector9. cv::StarDetector10. cv::SIFT11. cv::SURF12. cv::FastFeatureDetector13. cv::AgastFeatu…

leetcode2385. 感染二叉树需要的总时间(java)

感染二叉树需要的总时间 感染二叉树需要的总时间递归 dfs代码演示 二叉树专题 感染二叉树需要的总时间 来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 链接&#xff1a;https://leetcode.cn/problems/amount-of-time-for-binary-tree-to-be-infected 给你一棵二叉树的…

每日一练 | 华为认证真题练习Day60

1、启用GRE的keepalive功能后&#xff0c;GRE隧道的本端会周期性的每10s向对端发送一次keepalive报文。 A. 对 B. 错 2、AAA协议是RADIUS协议。 A. 对 B. 错 3、路由器Radius信息配置如下&#xff1a;下列说法正确的有&#xff1f;&#xff08;多选&#xff09; A. 计费服务…

[Spring Cloud]:学习笔记·(一)

文章目录 摘要1 认识微服务1.1 单体架构与分布式架构1.2 分布式架构与微服务1.3 微服务架构 摘要 摘要&#xff1a;分布式&#xff1b;微服务&#xff1b;springcloud 1 认识微服务 1.1 单体架构与分布式架构 架构方式解释优点缺点单体架构将业务所有功能集中在一个项目中开…

java从入门到起飞(二)——运算符

目录 前提——运算符概念算数运算符注意事项&#xff1a;字符的“”操作字符串的“”操作 赋值运算符注意事项&#xff1a; 自增自减运算符注意事项&#xff1a; 关系运算符注意事项&#xff1a; 逻辑运算符短路逻辑运算符注意事项&#xff1a; 三元运算符计算规则&#xff1a;…

OpenCV Mat类

文章目录 Mat类Mat类数据类型读取Mat类支持的运算OpenCV Mat数据类型指针ptr的使用多维矩阵创建 Mat类 Mat类数据类型读取 S 有符号整型 U 无符号整型 F 浮点型CV_8U - 8位无符号整数&#xff08;0…255&#xff09;CV_8S - 8位有符号整数&#xff08;-128…127&#xff…

Python基础(1)——Python简介

Python基础&#xff08;1&#xff09;——Python简介 文章目录 Python基础&#xff08;1&#xff09;——Python简介目标Python介绍Python版本总结 目标 了解PythonPython的应用领域Python的版本 Python介绍 Python是时下最流行、最火爆的编程语言之一&#xff0c;具体原因如…

SAP从入门到放弃系列之BOM组-Part1

目录 BOM组两种模式&#xff1a; 创建BOM的方式 方式一&#xff1a;直接在每个工厂分别创建BOM。 方式二&#xff1a;创建BOM组&#xff0c;然后每个工厂参考创建 方式三&#xff1a;创建BOM组&#xff0c;每个工厂参考创建&#xff0c;针对有特殊的工厂复制BOM组后进行调…

人机交互学习-9 以用户为中心的设计

以用户为中心的设计 以用户为中心的设计思想工程设计过程的三个假设以人为本设计思想三个方面的假设以用户为中心的设计四原则UDC项目包含的方法 用户参与设计用户参与的重要性选择哪些用户&#xff1f;为什么要让用户参与&#xff1f; 用户参与的形式参与式设计PICTIVECARD 理…

Nacos-手写注册中心基本原理

本文已收录于专栏 《中间件合集》 目录 概念说明需求分析核心功能代码实现AService模块BService模块NacosService模块NacosSDK模块 注意事项总结提升 概念说明 注册中心是微服务架构中的纽带&#xff0c;类似于“通讯录”&#xff0c;它记录了服务和服务地址的映射关系。在分布…

香蕉派(Banana Pi) BPI-M2 Zero 评测试,与树莓派 Zero同尺寸的开发板

Banana Pi M2 Zero 是一款微型计算机&#xff0c;配备四核处理器并内置 Wi-Fi 和蓝牙。这是一款非常适合基本计算任务甚至轻度游戏的小型设备。在这篇评论中&#xff0c;我们将了解 M2 Zero 的性能、功能和价值。 什么是 Banana Pi BPI-M2 Zero Banana Pi M2 Zero 是由深圳公…

MongoDB(Windows版)安装

首先需要下载 官网&#xff1a;MongoDB: The Developer Data Platform | MongoDB 安装过程 需要安装的版本 第一步&#xff1a;安装时&#xff0c;Custom是指可以自定义安装路径&#xff0c;然后傻瓜式安装即可&#xff08;注意&#xff1a;先不要安装图形化工具&#xff0…

2023-06-17 LeetCode每日一题(分割圆的最少切割次数)

2023-06-17每日一题 一、题目编号 2481. 分割圆的最少切割次数二、题目链接 点击跳转到题目位置 三、题目描述 圆内一个 有效切割 &#xff0c;符合以下二者之一&#xff1a; 该切割是两个端点在圆上的线段&#xff0c;且该线段经过圆心。该切割是一端在圆心另一端在圆上…

【Vue】学习笔记-Vue Router activated deactivated 路由守卫

Vue Router activated deactivated 路由守卫 activated deactivated路由守卫1.全局守卫2.独享守卫3.组件内守卫全局路由守卫路由器的两种工作模式 activated deactivated activated 和 deactivated 是路由组件所独有的两个钩子&#xff0c;用于捕获路由组件的激活状态 具体使用…

管理类联考——英语——趣味篇——阅读——考题的来源

Part One考研英语阅读——Part A 1.卫报 《卫报》( The Guardian)是英国的全国性综合内容日报。与《泰晤士报》、《每日电讯报》被合称为英国三大报。由约翰爱德华容泰勒创办于1821年5月5日。该报注重报道国际新闻&#xff0c;擅长发表评论和分析性专题文章。一般公众视《卫报…

【数据分析之道-Matplotlib(九)】Matplotlib棉棒图

文章目录 专栏导读1、Matplotlib棉棒图stem()基本语法2、Matplotlib棉棒图stem()定义样式2.1linefmt参数2.2markerfmt参数2.3举例一&#xff1a;直线样式2.4举例二&#xff1a;圆点样式 3、棉棒图案例实战3.1绘制每月销量的棉棒图3.2绘制每月销量与平均销量之差 专栏导读 ✍ 作…