STM32—CAN通信

文章目录

  • 一、CAN通信简介
    • 1.1 CAN简介
    • 1.2 CAN协议特点
    • 1.3 CAN通信的帧类型
    • 1.4 数据帧结构
    • 1.5 CAN的位时序
    • 1.6 CAN的仲裁功能
  • 二、STM32F1的CAN
    • 2.1 bxCAN简介
    • 2.2 bxCAN工作模式
      • 2.2.1 初始化模式
      • 2.2.2 正常模式
      • 2.2.3 睡眠模式
      • 2.2.4 静默模式
      • 2.2.5 环回模式
    • 2.3 位时序和波特率
  • 三、CAN配置步骤
  • 四、实战项目
    • 4.1 CAN初始化
    • 4.2 CAN发送
    • 4.3 CAN接收
    • 4.4 CAN收发测试
    • 4.5 补充说明

一、CAN通信简介

1.1 CAN简介

CAN全称是Controller Area Network,控制器局域网络,是ISO国际标准化的串行通信协议。CAN是国际上应用最广泛的现场总线之一。CAN通信只有两根信号线,分别是CAN_H和CAN_L,CAN 控制器根据这两根线上的电位差来判断总线电平。总线申平分为显性电平和隐性申平,二者必居其一。发送方通过使总线电平发生变化,将消息发送给接收方。

  • 2.0V---------逻辑0------显性电平
  • 0 V-----------逻辑1------隐性电平

CAN总线遵从“线与”机制,显性电平可以覆盖隐性电平。这就导致只有所有节点都发送隐形电平时总线才处于隐性状态
CAN通信示意图

1.2 CAN协议特点

  • 多主控
    在总线空闲时所有单元都可以发送消息。当两个以上单元发送消息时,会根据标识符(ID)决定发送的优先级。
  • 通信速度较快,最高可达1Mbps。通信距离较远
    当速度为1Mbps时,传输距离小于40m。当速度小于500Kbps时,传输距离最远可达10Km。
  • 具有错误检测,错误通知和错误恢复功能
    CAN总线上的任意一个单元都可以检测错误,当任意一个单元检测出错误时,会立刻通知其他单元。正在发送消息的单元一旦检测出错误,会强制结束当前发送。强制结束的单元会不断重新发送消息,直到发送成功。
  • 故障封闭功能
    CAN可以判断出错误的类型是总线上暂时的数据错误还是持续的数据错误。当总线上持续出现数据错误时,可以将引起故障的单元从总线上隔离出去。
  • 连接节点多
    理论上连接单元没有数量限制,但是连接单元越多,速度就会越低。

1.3 CAN通信的帧类型

CAN通信有5种帧类型

帧类型帧用途
数据帧用于发送单元向接收单元传送数据的帧
遥控帧用于接收单元向具有相同ID的发送单元请求数据的帧
错误帧用于当检测出错误时向其它单元通知错误的帧
过载帧用于接收单元通知其尚未做好接收准备的帧
帧间隔用于将数据帧及遥控锁与前面的帧分离开来的帧

在上述的几种帧里,数据很和遥控帧有标准帧扩展帧两种。标准帧有11位ID,扩展帧有29位ID。

1.4 数据帧结构

CAN通信数据帧的构成如下

CAN通信数据帧结构

  • 帧起始
    表示数据帧开始的段,标准帧和扩展帧的帧起始都是由1个位的显性电平组成。
  • 仲裁段
    表示数据帧优先级的段。
    仲裁段

RTR是用来表示是否是远程帧(遥控帧)。RTR为0是数据帧,RTR为1是远程帧。扩展帧中的IDE是标识符的选择位,如果为0,使用标准标识符,如果为1,使用扩展标识符。扩展帧的SRR相当于标准帧中的RTR位。标准帧的ID禁止高七位是隐性电平

  • 控制段
    控制段由6位构成,表示数据段的字节数。
    控制段
    扩展帧的r0r1是保留位,保留位必须全部以显性电平发送。DLC是数据的长度码,数据的字节数范围是0~8。IDE是标识符的选择位,如果为0,使用标准标识符,如果为1,使用扩展标识符。
  • 数据段
    数据段可以包含0~8个字节的数据。从MSB(最高位)开始传输。标准帧和扩展帧的数据段相同。
  • CRC段
    CRC段用于校验,检查帧传输是否存在错误。CRC段包含15位CRC序列和1位CRC界定符。标准帧和扩展帧的CRC段相同。
    CRC段
  • ACK段
    ACK段用来确认是否正常接收。由ACK槽和ACK界定符2位组成。标准帧和扩展帧的ACK段相同。
    ACK段
  • 帧结束
    由7位隐形电平组成,表示帧的结束。标准帧和扩展帧的帧结束相同。

1.5 CAN的位时序

由发送单元在非同步状态下每秒钟发送的位数称为位速率。一个位可以分成4段。

  • 同步段 SS
  • 传播时间段 PTS
  • 相位缓冲段1 PBS1
  • 相位缓冲段2 PBS2

上面的这些段由称为Time Quantum(Tq)的最小时间单位组成。1个位分成4个段,一个段又分成若干个Tq,这成为位时序
位构成
采样点是读取总线电平,并将读到的电平作为位值的点。

1.6 CAN的仲裁功能

在总线空闲态,最先开始发送消息的单元获得发送权。当多个单元同时开始发送时,各发送单元从仲裁段的第一位开始进行仲裁。连续输出显性电平最多的单元可继续发送

仲裁过程

二、STM32F1的CAN

2.1 bxCAN简介

STM32F1芯片自带bxCAN 控制器 (Basic Extended CAN),即基本扩展CAN,可与 CAN 网络进行交互,它支持 2.0A 和 B 版本的CAN 协议。STM32F1的bxCAN有以下特点

  • 支持 CAN 协议 2.0A 和 20B 主动模式
  • 波特率最高达 1Mbps
  • 支持时间触发通信
  • 具有 3 个发送邮箱
  • 具有 3 级深度的 2 个接收 FIFO
  • 可变的过滤器组(STM32F103ZET6有14个)

bxCAN模块可以完全自动地接收和发送CAN报文,且完全支持标准标识符(11位)和扩展标识符(29位)。

2.2 bxCAN工作模式

bXCAN有3个主要的工作模式:初始化模式正常模式睡眠模式。除此之外,还有测试模式,静默模式,环回模式。

2.2.1 初始化模式

首先看一下CAN主控制寄存器 (CAN_MCR)的INRQ位。

寄存器介绍CAN_MCR
寄存器CAN_MSR介绍

通过介绍可以直到,想要进入初始化模式,软件先将CAN_MCR的INRQ位置1。然后等待硬件将CAN主状态寄存器(CAN_MSR)的INAK位置1。此时进入初始化模式。

当bxCAN处于初始化模式时,禁止报文的接收和发送,并且CANTX引脚输出隐性位(高电平)。

2.2.2 正常模式

在初始化完成后,软件应该让硬件进入正常模式,以便正常接收和发送报文。继续看上面对于CAN主控制寄存器INRQ位的介绍。软件将INRQ位清0,可以使CAN从初始化模式进入正常模式。此时等待硬件将CAN主状态寄存器的INAK位清0即可。

2.2.3 睡眠模式

bxCAN可工作在低功耗的睡眠模式。在该模式下,bxCAN的时钟停止了,但软件仍然可以访问邮箱寄存器。

寄存器CAN_MSR介绍

可以看出,软件将CAN主控制寄存器的SLEEP置1,即可请求进入睡眠模式。清零该位,退出睡眠模式。另外,如果CAN_MCR寄存器的AWUM位为’1’,一旦检测到CAN总线的活动,硬件就自动对SLEEP位清’0’来唤醒bxCAN。

2.2.4 静默模式

将CAN_BTR寄存器的SILM位置’1’,来选择静默模式。

寄存器CAN_BTR介绍

在静默模式下,bxCAN可以正常地接收数据帧和远程帧,但只能发出隐性位,而不能真正发送报文。如果bxCAN需要发出显性位(确认位、过载标志、主动错误标志),那么这样的显性位在内部被接回来从而可以被CAN内核检测到,同时CAN总线不会受到影响而仍然维持在隐性位状态。因此,静默模式通常用于分析CAN总线的活动,而不会对总线造成影响-显性位(确认位、错误帧)不会真正发送到总线上。

静默模式

2.2.5 环回模式

将CAN_BTR寄存器的LBKM位置’1’,来选择环回模式。在环回模式下,bxCAN把发送的报文当作接收的报文并保存(如果可以通过接收过滤)在接收邮箱里。

寄存器CAN_BTR寄存器介绍

环回模式可用于自测试。为了避免外部的影响,在环回模式下CAN内核忽略确认错误(在数据/远程帧的确认位时刻,不检测是否有显性位)。在环回模式下,bxCAN在内部把Tx输出回馈到Rx输入上,而完全忽略CANRX引脚的实际状态。发送的报文可以在CANTX引脚上检测到。

环回模式

2.3 位时序和波特率

STM32将每一位分成三段

  • 同步段 SS
  • 时间段1 BS1
  • 时间段2 BS2

位时序

其中tpclk是APB1总线的时钟频率,默认为36MHz。

三、CAN配置步骤

  • 使能CAN时钟,将对应引脚复用映射为CAN功能
    STM32F103ZET6只有一个CAN,对应引脚如下
复用功能没有重映射部分重映射完全重映射
CAN_RXPA11PB8PD0
CAN_TXPA12PB9PD1

CAN_RX配置为上拉输入模式,CAN_TX配置为复用推挽输出。

  • 设置CAN工作模式,波特率等
    库函数提供了一个结构体和一个函数来配置。初始化函数为
uint8_t CAN_Init(CAN_TypeDef* CANx, CAN_InitTypeDef* CAN_InitStruct)

结构体成员如下

结构体成员作用
CAN_Prescaler设置Tq长度,范围为0~1023,实际为配置值加1
CAN_Mode设置CAN的工作模式
CAN_SJW指定CAN硬件允许延长或缩短位以执行重新同步的最大时间量,可以配置为1~4Tq
CAN_BS1设定BS1段的长度,范围是1~16Tq
CAN_BS2设定BS2段的长度,范围是1~8Tq
CAN_TTCM是否使用时间触发功能
CAN_ABOM是否使用自动离线管理,使用的话可以在结点离线后,适时的自动恢复,不需要软件干预
CAN_AWUM是否使用自动唤醒
CAN_NART是否使用自动重传
CAN_RFLM是否使用锁存接收,接收FIFO溢出时,不使能该功能,则新的会覆盖旧的。使能该功能,会丢弃新的数据。
CAN_TXFP设置发送报文优先级判定方法,使能时以报文发送邮箱的先后顺序发送,不使能,按照ID优先级发送。

波特率 = Fpclk1 / ((CAN_BS1 + CAN_BS2 + 1)* CAN_Prescaler)

  • 设置CAN筛选器(过滤器)
    库函数也提供了筛选器的配置函数
void CAN_FilterInit(CAN_FilterInitTypeDef* CAN_FilterInitStruct)

结构体内容如下

结构体成员作用
CAN_FilterIdHigh如果筛选器工作在32位模式,该成员存储ID的高16位。如果筛选器工作在16位模式,该成员存储的是完整的16位筛选ID。
CAN_FilterIdLow存储要筛选的ID,如果筛选器工作在32位模式,该成员存储ID的低16位。如果工作在16位模式,该成员存储第二个ID。
CAN_FilterMaskIdHigh存储要筛选的掩码。当筛选器工作在标识符模式,与上面的第一个成员功能相同。当筛选器工作在掩码模式时,改为存储的是掩码的高16位,或者是一个完整的16位掩码。
CAN_FilterMaskIdLow存储要筛选的掩码。当筛选器工作在标识符模式,与上面的第二个成员功能相同。当筛选器工作在掩码模式时,改为存储的是掩码的低16位,或者是一个完整的16位掩码。
CAN_FilterFIFOAssignment报文通过删选后,该报文存储到哪个FIFO,可选择FIFO0,FIFO1
CAN_FilterNumber选择要使用的筛选器编号,0~27
CAN_FilterMode设置筛选器的工作模式,可以设置为列表模式和掩码模式
CAN_FilterScale设置筛选器的位宽,32位或16位
CAN_FilterActivation是否激活筛选器
  • 选择CAN中断类型,开启中断
    库函数提供了一个中断的配置函数
void CAN_ITConfig(CAN_TypeDef* CANx, uint32_t CAN_IT, FunctionalState NewState)

CAN的中断类型有很多,这里就不再一一介绍了。

#define IS_CAN_IT(IT)        (((IT) == CAN_IT_TME) || ((IT) == CAN_IT_FMP0)  ||\
                             ((IT) == CAN_IT_FF0)  || ((IT) == CAN_IT_FOV0)  ||\
                             ((IT) == CAN_IT_FMP1) || ((IT) == CAN_IT_FF1)   ||\
                             ((IT) == CAN_IT_FOV1) || ((IT) == CAN_IT_EWG)   ||\
                             ((IT) == CAN_IT_EPV)  || ((IT) == CAN_IT_BOF)   ||\
                             ((IT) == CAN_IT_LEC)  || ((IT) == CAN_IT_ERR)   ||\
                             ((IT) == CAN_IT_WKU)  || ((IT) == CAN_IT_SLK))
  • CAN发送和接收消息
    CAN发送消息的函数是
uint8_t CAN_Transmit(CAN_TypeDef* CANx, CanTxMsg* TxMessage)

发送之前需要配置好消息的结构体,消息结构体成员如下

结构体成员作用
StdId存储报文的11位的标准标识符,范围是0x0~0x7FF
ExtId存储报文的29位扩展标识符,范围是0x0~0x1FFFFFFF
IDE配置使用哪个标识符,配置为STD,为标准帧。配置为EXT,为扩展帧。
RTR报文类型的标志,可以配置为CAN_RTR_Data,表示报文为数据帧。配置为CAN_RTR_Remote,表示报文为遥控帧。遥控帧没有数据段。
DLC存储数据段的长度,0~8。如果报文为遥控帧,该值配置为0。
Data[8]存储数据段数据

CAN接收函数为

void CAN_Receive(CAN_TypeDef* CANx, uint8_t FIFONumber, CanRxMsg* RxMessage)

CAN接收结构体与发送结构体基本相同,多了一个

结构体成员作用
FMI存储筛选器的编号,表示接收到的报文是从哪个筛选器进入FIFO的。
  • CAN状态获取
    库函数提供了很多可以获取CAN状态标志的函数,比如
uint8_t CAN_TransmitStatus(CAN_TypeDef* CANx, uint8_t TransmitMailbox)

FlagStatus CAN_GetFlagStatus(CAN_TypeDef* CANx, uint32_t CAN_FLAG);

四、实战项目

4.1 CAN初始化

/*
 *==============================================================================
 *函数名称:Drv_Can_Init
 *函数功能:初始化CAN
 *输入参数:tsjw:重新同步跳跃宽度(Tsjw);tbs1:BS1长度;tbs2:BS2长度;
						brp:Tq大小;mode:CAN工作模式
 *返回值:无
 *备  注:无
 *==============================================================================
 */
void Drv_Can_Init (u8 tsjw,u8 tbs1,u8 tbs2,u16 brp,u8 mode)
{
	// 结构体定义
	GPIO_InitTypeDef GPIO_InitStructure;
	CAN_InitTypeDef CAN_InitStructure;
	CAN_FilterInitTypeDef CAN_FilterInitStructure;
	
	// 开启时钟
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_CAN1, ENABLE);   // 打开CAN1时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);   // PA端口时钟打开
	
	// 初始化GPIO
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;   // PA11 CAN_RX   
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;   // 上拉输入模式
	GPIO_Init(GPIOA, &GPIO_InitStructure);	

	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;   // PA12 CAN_TX   
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;   // 复用推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;   // IO口速度为50MHz
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	// 初始化CAN
	CAN_InitStructure.CAN_TTCM=DISABLE;   // 非时间触发通信模式   
	CAN_InitStructure.CAN_ABOM=DISABLE;   // 软件自动离线管理	  
	CAN_InitStructure.CAN_AWUM=DISABLE;   // 睡眠模式通过软件唤醒(清除CAN->MCR的SLEEP位)
	CAN_InitStructure.CAN_NART=ENABLE;   // 使用报文自动传送 
	CAN_InitStructure.CAN_RFLM=DISABLE;   // 报文不锁定,新的覆盖旧的  
	CAN_InitStructure.CAN_TXFP=DISABLE;   // 优先级由报文标识符决定 
	CAN_InitStructure.CAN_Mode= mode;   //CAN工作模式设置 
	CAN_InitStructure.CAN_SJW=tsjw;   // 重新同步跳跃宽度(Tsjw)为tsjw+1个时间单位 CAN_SJW_1tq~CAN_SJW_4tq
	CAN_InitStructure.CAN_BS1=tbs1;   // Tbs1范围CAN_BS1_1tq ~CAN_BS1_16tq
	CAN_InitStructure.CAN_BS2=tbs2;   // Tbs2范围CAN_BS2_1tq ~	CAN_BS2_8tq
	CAN_InitStructure.CAN_Prescaler=brp;   //分频系数(Fdiv)为brp+1	
	CAN_Init(CAN1, &CAN_InitStructure);   // 初始化CAN1
	
	// 初始化过滤器
	CAN_FilterInitStructure.CAN_FilterNumber=0;   // 过滤器0
	CAN_FilterInitStructure.CAN_FilterMode=CAN_FilterMode_IdMask;   // 掩码模式
	CAN_FilterInitStructure.CAN_FilterScale=CAN_FilterScale_32bit;   // 32位 
	CAN_FilterInitStructure.CAN_FilterIdHigh=0x0000;   // 32位ID
	CAN_FilterInitStructure.CAN_FilterIdLow=0x0000;
	CAN_FilterInitStructure.CAN_FilterMaskIdHigh=0x0000;   // 32位MASK
	CAN_FilterInitStructure.CAN_FilterMaskIdLow=0x0000;
	CAN_FilterInitStructure.CAN_FilterFIFOAssignment=CAN_Filter_FIFO0;   // 过滤器0关联到FIFO0
	CAN_FilterInitStructure.CAN_FilterActivation=ENABLE;   // 激活过滤器0
	CAN_FilterInit(&CAN_FilterInitStructure);   // 过滤器初始化
}

4.2 CAN发送

/*
 *==============================================================================
 *函数名称:Med_Can_Send_Msg
 *函数功能:发送报文
 *输入参数:msg:数据段指针;len:数据长度
 *返回值:0:发送成功;1:发送失败
 *备  注:固定ID为0x12
 *==============================================================================
 */
u8 Med_Can_Send_Msg (u8* msg,u8 len)
{	
	u8 mbox;
	u16 i = 0;
	CanTxMsg TxMessage;   // 定义发送报文结构体
	TxMessage.StdId = 0x12;   // 标准标识符
	TxMessage.ExtId = 0x12;   // 扩展标识符
	TxMessage.IDE = CAN_Id_Standard;   // 使用标准标识符
	TxMessage.RTR = 0;   // 消息类型为数据帧,一帧8位
	TxMessage.DLC = len;
	for(i = 0;i < len;i ++)
	{
		TxMessage.Data[i] = msg[i];   // 填充帧数据段
	}
	mbox = CAN_Transmit(CAN1,&TxMessage);   // 发送报文   
	i = 0;
	
	// 等待发送结束
	while((CAN_TransmitStatus(CAN1,mbox) == CAN_TxStatus_Failed) && (i < 0XFFF))
	{
		i++;
	}
	
	// 返回发送情况
	if(i >= 0XFFF)
	{
		return 1;
	}
	return 0;		
}

4.3 CAN接收

/*
 *==============================================================================
 *函数名称:Med_Can_Receive_Msg
 *函数功能:接收报文
 *输入参数:buf:数据缓存区指针
 *返回值:0:没有接收到数据;其他:接收数据长度
 *备  注:无
 *==============================================================================
 */
u8 Med_Can_Receive_Msg (u8 *buf)
{		   		   
 	u32 i;
	CanRxMsg RxMessage;   // 定义接收报文结构体
	
	// 没有接收到数据,直接退出 
	if( CAN_MessagePending(CAN1,CAN_FIFO0) == 0)
	{
		return 0;
	}
	CAN_Receive(CAN1, CAN_FIFO0, &RxMessage);   // 读取数据	
	
	for(i = 0;i < RxMessage.DLC;i ++)
	{
		buf[i] = RxMessage.Data[i];
	}
	
	return RxMessage.DLC;	
}

4.4 CAN收发测试

利用按键WK UP控制报文的发送,按下一次发送一次报文。配置CAN波特率为500Kbps,环回模式。利用串口打印接收数据。需要注意的是,STM32只有CAN控制器,想要实现报文的收发,需要自己连接CAN收发器。首先初始化CAN

	// 初始化CAN,500Kbps波特率
	Drv_Can_Init(CAN_SJW_1tq,CAN_BS1_9tq,CAN_BS2_8tq,4,CAN_Mode_LoopBack);

然后编写主程序

u8 gKeyValue = 0;   // 获取按键值

u8 gSendData[8] = {'1','2','3','4','5','6','7','8'};   // 发送内容数组
u8 gReceData[8];   // 接收内容数组
u8 gFlag = 0;   // 接收发送标志

int main(void)
{
	Med_Mcu_Iint();   // 系统初始化
	
	while(1)
  {
		gKeyValue = Med_KeyScan();
		
		// WK UP 按下发送消息
		if (gKeyValue == 1)
		{
			gFlag = Med_Can_Send_Msg(gSendData,8);
			
			// 发送失败
			if (gFlag)
			{
				printf ("Send Defeat!\r\n");
			}
			else
			{
				printf ("Send Success!\r\n");
			}
		}
		
		// 接收报文
		gFlag = Med_Can_Receive_Msg(gReceData);
		
		// 接收成功
		if (gFlag)
		{
			printf ("Receive Data:%s\r\n",gReceData);
		}
	}
}

测试结果如下

测试结果

4.5 补充说明

上面的CAN收发测试程序,发送的内容是字符串“12345678”。如果发送的是数字12345678。在串口打印接收数据时需要先将接收到的数据转换成字符,然后再打印。转换方法很简单,只需要对接收数组的每一位加48即可。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

leetCode刷题记录3-面试经典150题

文章目录 不要摆&#xff0c;没事干就刷题&#xff0c;只有好处&#xff0c;没有坏处&#xff0c;实在不行&#xff0c;看看竞赛题面试经典 150 题80. 删除有序数组中的重复项 II189. 轮转数组122. 买卖股票的最佳时机 II 不要摆&#xff0c;没事干就刷题&#xff0c;只有好处&…

阿里云部署 ChatGLM2-6B 与 langchain+ChatGLM

1.ChatGLM2-6B 部署 更新系统 apt-get update 安装git apt-get install git-lfs git init git lfs install 克隆 ChatGLM2-6B 源码 git clone https://github.com/THUDM/ChatGLM2-6B.git 克隆 chatglm2-6b 模型 #进入目录 cd ChatGLM2-6B #创建目录 mkdir model #进入目录 cd m…

安全攻击 --- XSS攻击

XSS跨站脚本攻击 &#xff08;1&#xff09;简介 OWASP TOP 10 之一&#xff0c;XSS被称为跨站脚本攻击&#xff08;Cross-Site-Scripting&#xff09;主要基于JavaScript&#xff08;JS&#xff09;完成攻击行为XSS通过精心构造JS代码注入到网页中&#xff0c;并由浏览器解释…

在nginx上部署nuxt项目

先安装Node.js 我安的18.17.0。 安装完成后&#xff0c;可以使用cmd&#xff0c;winr然cmd进入&#xff0c;测试是否安装成功。安装在哪个盘都可以测试。 测试 输入node -v 和 npm -v&#xff0c;&#xff08;中间有空格&#xff09;出现下图版本提示就是完成了NodeJS的安装…

前端开发实习总结参考范文

▼前端开发实习总结篇四 读了三年的大学&#xff0c;然而大多数人对本专业的认识还是不那么透彻&#xff0c;学的东西真正能够学以致用的东西很少&#xff0c;大家都抱怨没有实践的机会&#xff0c;在很多同学心里面对于本专业还是很茫然。直到即将毕业的时候才知道我们以前学…

【Linux后端服务器开发】HTTPS协议

目录 一、加密算法 二、中间人攻击 三、CA认证 一、加密算法 HTTPS协议是什么&#xff1f;HTTPS协议也是一个应用层协议&#xff0c;是在HTTP协议的基础上引入了一个加密层。 HTTP协议内容是按照文本的方式明文传输的&#xff0c;这就导致在传输过程中出现一些被篡改的情况…

ROS1ROS2之CmakeList.txt和package.xml用法详解

前言&#xff1a;目前还在学习ROS无人机框架中&#xff0c;&#xff0c;&#xff0c; 更多更新文章详见我的个人博客主页【前往】 文章目录 1. CMakeLists.txt与package.xml的作用2. 生成CMakeLists.txt2.1 ROS12.2 ROS2 3. CMakeLists.txt编写3.1 ROS13.2 ROS2 4. package.xml…

Ubuntu 20.04 Ubuntu18.04安装录屏软件Kazam

1.在Ubuntu Software里面输入Kazam&#xff0c;就可以找不到这个软件&#xff0c;直接点击install就可以了 2.使用方法&#xff1a; 选择Screencast&#xff08;录屏&#xff09; Fullscreen&#xff08;全屏&#xff09;-----Windows&#xff08;窗口&#xff09;--------Ar…

1、传统锁回顾(Jvm本地锁,MySQL悲观锁、乐观锁)

目录 1.1 从减库存聊起1.2 环境准备1.3 简单实现减库存1.4 演示超卖现象1.5 jvm锁1.6 三种情况导致Jvm本地锁失效1、多例模式下&#xff0c;Jvm本地锁失效2、Spring的事务导致Jvm本地锁失效3、集群部署导致Jvm本地锁失效 1.7 mysql锁演示1.7.1、一个sql1.7.2、悲观锁1.7.3、乐观…

fragment

fragment 在vue2中,组件必须有一个跟标签在vue3中,组件可以没有跟标签,内部会将多个标签包含在一个fragment虚拟元素中好处:减少标签层级,减小内存占用 teltport 什么是teltport teleport是一种能够将我们组件html结构移动到指定位置的技术 像是下面的代码不适用teleport:…

信息系统项目管理师(第四版)教材精读思维导图-第三章信息系统治理

请参阅我的另一篇文章&#xff0c;综合介绍软考高项&#xff1a; 信息系统项目管理师&#xff08;软考高项&#xff09;备考总结_计算机技术与软件专业技术_铭记北宸的博客-CSDN博客 目录 3.1 IT治理 3.2 IT审计 3.1 IT治理 3.2 IT审计

Java程序设计六大原则设计模式

Java程序设计六大原则 一、单一职责原则&#xff1a; 一个接口或者类只有一个原因引起变化&#xff0c;即一个接口或者类只有一个职责&#xff0c;负责一件事情。&#xff08;此原则同样适用于方法&#xff09; 好处&#xff1a;1、复杂性降低&#xff1b;2、可读性提高&…

elasticsearch查询操作(API方式)

说明&#xff1a;elasticsearch查询操作除了使用DSL语句的方式&#xff08;参考&#xff1a;http://t.csdn.cn/k7IGL&#xff09;&#xff0c;也可以使用API的方式。 准备 使用前需先导入依赖 <!--RestHighLevelClient依赖--><dependency><groupId>org.ela…

内存泄漏是什么?有什么危害

内存泄漏是什么&#xff1f;有什么危害 1. 前言1.内存泄漏是什么&#xff1f;2. 为什么会发生内存泄漏3. 内存泄漏的危害4. 总结 1. 前言 在各种项目开发中&#xff0c;内存泄漏是一个很严重的问题。对资源管理、性能优越、系统稳定性&#xff0c;以及是否安全产生极大印象。本…

AndroidStudio设计一个计算器

界面设计 activity_calcuator.xml 设计&#xff1a; <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas.android.com/apk/res/android"xmlns:app"http://schemas.android.com/apk/res-auto&qu…

2、基于redis实现分布式锁

目录 2.1. 基本实现2.2. 防死锁2.3. 防误删2.4. redis中的lua脚本2.4.1 redis 并不能保证2.4.2 lua介绍 2.5. 使用lua保证删除原子性 2.1. 基本实现 借助于redis中的命令setnx(key, value)&#xff0c;key不存在就新增&#xff0c;存在就什么都不做。同时有多个客户端发送setn…

Easy-Es笔记

一、Easy-ES概述 Easy-Es&#xff08;简称EE&#xff09;是一款由国内开发者打造并完全开源的ElasticSearch-ORM框架。在原生 RestHighLevelClient 的基础上&#xff0c;只做增强不做改变&#xff0c;为简化开发、提高效率而生。Easy-Es采用和MP一致的语法设计&#xff0c;降低…

HDFS异构存储详解

异构存储 HDFS异构存储类型什么是异构存储异构存储类型如何让HDFS知道集群中的数据存储目录是那种类型存储介质 块存储选择策略选择策略说明选择策略的命令 案例&#xff1a;冷热温数据异构存储对应步骤 HDFS内存存储策略支持-- LAZY PERSIST介绍执行使用 HDFS异构存储类型 冷…

C# winform子窗口向父窗口传值

这里我使用一个简单的方法。只需要在父窗口定义一个静态变量就行。 父窗体为Form1,子窗体为Form2。 public static int get_num0; 子窗体直接给get_num赋值即可。 Form1.get_num2; 这样父窗体就能获得get_num修改后这个值了

[start] m40 test

software & update 470 drive version # cd /etc/apt # mv sources.list sources.list.bak # sudo vi /etc/apt/sources.list # 默认注释了源码镜像以提高 apt update 速度&#xff0c;如有需要可自行取消注释 deb https://mirrors.tuna.tsinghua.edu.cn/ubuntu/ ja…
最新文章