【嵌入式-stm32电位器控制以及旋转编码器控制LED亮暗】

嵌入式-stm32电位器控制LED亮暗

  • 任务1
  • 代码1
    • Key.c
    • Key.h
    • Timer.c
    • Timer.h
    • PWM.c
    • PWM.h
    • main.c
  • 实验现象1
  • 任务2
  • 代码2
    • Key.c
    • Key.h
    • main.c
  • 实验现象2
  • 问题与解决
  • 总结

源码框架取自江协科技,在此基础上做扩展开发。

任务1

本文主要介绍利用stm32f103C8T6实现电位器控制PWM的占空比大小来改变LED亮暗程度,按键实现使用定时器非阻塞式,其中一个按键切换3个LED的控制状态,另一个按键是重置当前的LED为熄灭状态。

代码1

Key.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "oled.h"
#include "PWM.h"
#include "AD.h"
#include "Key.h"
#include <stdio.h>extern uint16_t ADValue;			//定义AD值变量
uint8_t Key_Num;
/*** 函    数:按键初始化* 参    数:无* 返 回 值:无*/
void Key_Init(void)
{/*开启时钟*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);/*GPIO初始化*/GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_10;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);	
}  // 定义模式枚举  
typedef enum {  MODE_PWM_CH2 = 0,  MODE_PWM_CH3,  MODE_PWM_CH4,  MODE_MAX  
} PWM_MODE;  // 全局变量  
volatile PWM_MODE currentMode = MODE_PWM_CH2;  
volatile uint16_t pwmValue = 0;  
volatile uint8_t resetFlag = 0;  
volatile uint8_t systemActive = 0;  //新增系统激活标志// 初始化显示函数  
void Initial_Display(void) {  // 清屏  OLED_Clear();  // 显示初始状态  OLED_ShowString(1, 1, "System Ready");  OLED_ShowString(2, 1, "Active KEY1 ");  // 初始化时关闭所有LED  PWM_SetCompare2(0);  PWM_SetCompare3(0);  PWM_SetCompare4(0);   
}  uint8_t Key_GetNum(void)
{uint8_t Temp;           Temp = Key_Num;         //读取按键键值Key_Num = 0; 					  //清零,防止重复触发return Temp;
}uint8_t Key_GetState(void)
{if(GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_8) == 0){return 1;}if(GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_10) == 0){return 2;}return 0;   //无按键按下
}void Key_Tick(void)         
{static uint8_t Count; //静态计数器,记录中断次数static uint8_t CurrState, PrevState;Count++;if(Count >= 20)   //20ms执行一次按键扫描(中断周期为1ms){Count = 0;PrevState = CurrState;         //保存前一次按键状态CurrState = Key_GetState();    //读取当前按键状态//检测按键释放动作(下降沿)if(CurrState == 0 && PrevState != 0){Key_Num = PrevState;    //记录按键值(1或者2)}}
}// 设置PWM的函数  
void SetPWM(uint16_t value) {  switch (currentMode) {  case MODE_PWM_CH2:  PWM_SetCompare2(value);  break;  case MODE_PWM_CH3:  PWM_SetCompare3(value);  break;  case MODE_PWM_CH4:  PWM_SetCompare4(value);  break;  }  
}  // 更新显示模式函数  
void Update_ModeDisplay(void) {  // 清除原有模式显示  OLED_Clear();  // 根据当前模式显示  switch (currentMode) {  case MODE_PWM_CH2:  OLED_ShowString(1, 1, "Mode: CH2");  break;  case MODE_PWM_CH3:  OLED_ShowString(1, 1, "Mode: CH3");  break;  case MODE_PWM_CH4:  OLED_ShowString(1, 1, "Mode: CH4");  break;  }  // 显示初始PWM值  OLED_ShowString(2, 1, "PWM:   0");  
}  /*OLED显示70.5%函数*/
void ShowPwm_Percent(uint8_t Line, uint8_t Colum, uint16_t pwmValue)
{char str[16];uint16_t integer = pwmValue / 10;  //整数部分如70uint16_t decimal = pwmValue % 10;  //小鼠部分如5sprintf(str, "%4d.%1d%%",integer,decimal);OLED_ShowString(Line,Colum,str);
}// 按键控制函数  
void Key_control(void) {  uint8_t keyNum = Key_GetNum();  // 处理按键1:模式切换  if (keyNum == 1) {  // 重置标志清零  resetFlag = 0; if(systemActive == 0){systemActive = 1;currentMode = MODE_PWM_CH2;Update_ModeDisplay();}else{// 切换模式  currentMode++;  if (currentMode >= MODE_MAX) {  currentMode = MODE_PWM_CH2;  }  // 更新模式显示  Update_ModeDisplay(); }				}  // 处理按键2:重置为全暗  if (keyNum == 2) {  // 设置重置标志  resetFlag = 1;  // 将当前通道设置为0  SetPWM(0);  pwmValue = 0;  // 显示PWM值  OLED_ShowNum(2, 5, pwmValue, 3);  }  // 仅在非重置状态下读取ADC和设置PWM  if (resetFlag == 0 && systemActive) {  // 读取ADC并设置PWM  //uint16_t adcValue = AD_GetValue();  pwmValue = (AD_GetValue() * 1000)/ 4095 ;  // 设置当前通道PWM  SetPWM(pwmValue);  // 显示PWM值 OLED_ShowNum(3, 1, pwmValue, 4);  // 直接显示pwmValue的值			ShowPwm_Percent(2, 4, pwmValue);//OLED_ShowNum(2, 5, pwmValue, 3);  }  
}  

Key.h

#ifndef __KEY_H
#define __KEY_Hvoid Key_Init(void);
uint8_t Key_GetNum(void);
void Key_control(void);
void Initial_Display(void);
void SetPWM(uint16_t value);
void Key_Tick(void);
uint8_t Key_GetState(void);#endif

Timer.c

#include "stm32f10x.h"                  // Device headervoid Timer_Init(void)
{RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);TIM_InternalClockConfig(TIM3);TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;TIM_TimeBaseInitStructure.TIM_Period = 1000 - 1;TIM_TimeBaseInitStructure.TIM_Prescaler = 72 - 1;TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;TIM_TimeBaseInit(TIM3, &TIM_TimeBaseInitStructure);TIM_ClearFlag(TIM3, TIM_FLAG_Update);TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE);NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);NVIC_InitTypeDef NVIC_InitStructure;NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;NVIC_Init(&NVIC_InitStructure);TIM_Cmd(TIM3, ENABLE);
}/*
void TIM2_IRQHandler(void)
{if (TIM_GetITStatus(TIM2, TIM_IT_Update) == SET){TIM_ClearITPendingBit(TIM2, TIM_IT_Update);}
}
*/

Timer.h

#ifndef __TIMER_H
#define __TIMER_Hvoid Timer_Init(void);#endif

PWM.c

#include "stm32f10x.h"                  // Device header/*** 函    数:PWM初始化* 参    数:无* 返 回 值:无*/void TIM2_PWM_Init(void)
{TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;TIM_OCInitTypeDef TIM_OCInitStructure;GPIO_InitTypeDef GPIO_InitStruct;	// 打开定时器2时钟RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);																RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);		GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP; //GPIO采用复用推挽输出模式GPIO_InitStruct.GPIO_Pin = GPIO_Pin_3|GPIO_Pin_2|GPIO_Pin_1; //TIM2同时产生三路PWM波 在管脚123 a11GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;  //GPIO速度50MHZGPIO_Init(GPIOA,&GPIO_InitStruct);  //初始化函数 让刚刚配置的参数 输入到对应寄存器里面// 配置定时器2为PWM模式TIM_TimeBaseStructure.TIM_Period = 999; // PWM周期TIM_TimeBaseStructure.TIM_Prescaler = 720; // 72MHz/(71+1) = 1MHz,计数频率为1MHzTIM_TimeBaseStructure.TIM_ClockDivision = 0;TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);// 配置TIM2通道2为PWM模式TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;TIM_OCInitStructure.TIM_Pulse = 0; // 初始占空比0%TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;TIM_OC2Init(TIM2, &TIM_OCInitStructure);TIM_OC3Init(TIM2, &TIM_OCInitStructure);TIM_OC4Init(TIM2, &TIM_OCInitStructure);// 使能TIM2TIM_Cmd(TIM2, ENABLE);
}
/*** 函    数:PWM设置CCR* 参    数:Compare 要写入的CCR的值,范围:0~1000* 返 回 值:无* 注意事项:CCR和ARR共同决定占空比,此函数仅设置CCR的值,并不直接是占空比*           占空比Duty = CCR / (ARR + 1)*/void PWM_SetCompare2(uint16_t Compare)
{TIM_SetCompare2(TIM2 ,Compare );	//设置CCR1的值
}
void PWM_SetCompare3(uint16_t Compare)
{TIM_SetCompare3(TIM2 ,Compare );	//设置CCR1的值
}
void PWM_SetCompare4(uint16_t Compare)
{TIM_SetCompare4(TIM2 ,Compare );	//设置CCR1的值
}

PWM.h

#ifndef __PWM_H
#define __PWM_Hvoid TIM2_PWM_Init(void);void PWM_SetCompare2(uint16_t Compare);
void PWM_SetCompare3(uint16_t Compare);
void PWM_SetCompare4(uint16_t Compare);#endif

main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "Key.h"
#include "sys.h"
#include "AD.h"
#include "PWM.h"
#include "Timer.h"/*全局变量*/
uint16_t ADValue;			//定义AD值变量int main(void)
{/*模块初始化*/OLED_Init();		//OLED初始化Key_Init();			//按键初始化AD_Init();			//AD初始化TIM2_PWM_Init();	//定时器2PWM初始化Timer_Init();/*OLED显示静态字符*/Initial_Display();while (1){//KeyNum=Key_GetNum();	//获取键码值Key_control();			//按键PWM控制}
}//中断服务函数
//每次TIM3溢出时触发中断,调用Key_Tick()进行按键扫描
//清除中断标志,避免重复进入中断
void TIM3_IRQHandler(void)
{if (TIM_GetITStatus(TIM3, TIM_IT_Update) == SET){Key_Tick();TIM_ClearITPendingBit(TIM3, TIM_IT_Update);}
}

实验现象1

以下是通过电位器控制PWM输出大小的值进而调暗LED
在这里插入图片描述
在这里插入图片描述
通过网盘分享的文件:电位器改变PWM输出控制LED
链接: https://pan.baidu.com/s/1JrevfJ2GTsBqLyRb4Do39g 提取码: 6688
在这里插入图片描述

任务2

旋转编码器控制LED亮暗:
1、LED亮度控制:旋转编码器调节PWM占空比,控制LED亮度。
2、状态显示:OLED实时显示当前PWM占空比(格式为XX.X%)。
3、模式切换:通过独立按键切换PWM输出通道(如CH2、CH3、CH4)。
4、系统激活与重置:按键控制系统的启动和重置。
接线图片来自江协议科技
在这里插入图片描述

代码2

1、模块化代码架构
编码器驱动:通过外部中断检测旋转方向,更新计数值。
PWM生成:配置定时器(如TIM2)的PWM模式,动态调节占空比。
OLED显示:格式化显示占空比和模式信息。
主控制逻辑:整合按键、编码器和PWM功能,实现状态机控制。

Key.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "oled.h"
#include "PWM.h"
#include "AD.h"
#include "Key.h"
#include "Encoder.h"
#include <stdio.h>uint8_t Key_Num;
signed Key_Encoder_Count = 0;
/*** 函    数:按键初始化* 参    数:无* 返 回 值:无*/
void Key_Init(void)
{/*开启时钟*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);/*GPIO初始化*/GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_10;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);	
}  // 定义模式枚举  
typedef enum {  MODE_PWM_CH2 = 0,  MODE_PWM_CH3,  MODE_PWM_CH4,  MODE_MAX  
} PWM_MODE;  // 全局变量  
volatile PWM_MODE currentMode = MODE_PWM_CH2;  
volatile uint16_t pwmValue = 0;  
volatile uint8_t resetFlag = 0;  
volatile uint8_t systemActive = 0;  //新增系统激活标志// 初始化显示函数  
void Initial_Display(void) {  // 清屏  OLED_Clear();  // 显示初始状态  OLED_ShowString(1, 1, "System Ready");  OLED_ShowString(2, 1, "Active KEY1 ");  // 初始化时关闭所有LED  PWM_SetCompare2(0);  PWM_SetCompare3(0);  PWM_SetCompare4(0);   
}  uint8_t Key_GetNum(void)
{uint8_t Temp;           Temp = Key_Num;         //读取按键键值Key_Num = 0; 					  //清零,防止重复触发return Temp;
}uint8_t Key_GetState(void)
{if(GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_8) == 0){return 1;}if(GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_10) == 0){return 2;}return 0;   //无按键按下
}void Key_Tick(void)         
{static uint8_t Count; //静态计数器,记录中断次数static uint8_t CurrState, PrevState;Count++;if(Count >= 20)   //20ms执行一次按键扫描(中断周期为1ms){Count = 0;PrevState = CurrState;         //保存前一次按键状态CurrState = Key_GetState();    //读取当前按键状态//检测按键释放动作(下降沿)if(CurrState == 0 && PrevState != 0){Key_Num = PrevState;    //记录按键值(1或者2)}}
}// 设置PWM的函数  
void SetPWM(uint16_t value) {  switch (currentMode) {  case MODE_PWM_CH2:  PWM_SetCompare2(value);  break;  case MODE_PWM_CH3:  PWM_SetCompare3(value);  break;  case MODE_PWM_CH4:  PWM_SetCompare4(value);  break;  }  
}  // 更新显示模式函数  
void Update_ModeDisplay(void) {  // 清除原有模式显示  OLED_Clear();  // 根据当前模式显示  switch (currentMode) {  case MODE_PWM_CH2:  OLED_ShowString(1, 1, "Mode: CH2");  break;  case MODE_PWM_CH3:  OLED_ShowString(1, 1, "Mode: CH3");  break;  case MODE_PWM_CH4:  OLED_ShowString(1, 1, "Mode: CH4");  break;  }  // 显示初始PWM值  OLED_ShowString(2, 1, "PWM:   0");  
}  /*OLED显示70.5%函数*/
void ShowPwm_Percent(uint8_t Line, uint8_t Colum, uint16_t pwmValue)
{char str[16];uint16_t integer = pwmValue / 10;  //整数部分如70uint16_t decimal = pwmValue % 10;  //小鼠部分如5sprintf(str, "%4d.%1d%%",integer,decimal);OLED_ShowString(Line,Colum,str);
}// 按键控制函数  
void Key_control(void) {  uint8_t keyNum = Key_GetNum();  // 处理按键1:模式切换  if (keyNum == 1) {  // 重置标志清零  resetFlag = 0; if(systemActive == 0){systemActive = 1;currentMode = MODE_PWM_CH2;Update_ModeDisplay();}else{// 切换模式  currentMode++;  if (currentMode >= MODE_MAX) {  currentMode = MODE_PWM_CH2;  }  // 更新模式显示  Update_ModeDisplay(); }				}  // 处理按键2:重置为全暗  if (keyNum == 2) {  // 设置重置标志  resetFlag = 1;  // 将当前通道设置为0  SetPWM(0);  pwmValue = 0;  // 显示PWM值  OLED_ShowNum(2, 5, pwmValue, 3);  }  // 仅在非重置状态下读取ADC和设置PWM  if (resetFlag == 0 && systemActive) {  Key_Encoder_Count += Encoder_Get();if(Key_Encoder_Count < 0){Key_Encoder_Count = 0;}if(Key_Encoder_Count > 100){Key_Encoder_Count = 100;}pwmValue = (Key_Encoder_Count * 10) ;  // 设置当前通道PWM  SetPWM(pwmValue);  // 显示PWM值 OLED_ShowNum(3, 1, pwmValue, 4);  // 直接显示pwmValue的值			ShowPwm_Percent(2, 4, pwmValue);//OLED_ShowNum(2, 5, pwmValue, 3);  }  
}  

Key.h

#ifndef __KEY_H
#define __KEY_Hvoid Key_Init(void);
uint8_t Key_GetNum(void);
void Key_control(void);
void Initial_Display(void);
void SetPWM(uint16_t value);
void Key_Tick(void);
uint8_t Key_GetState(void);#endif

main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "Key.h"
#include "sys.h"
#include "AD.h"
#include "PWM.h"
#include "Timer.h"
#include "Encoder.h"int main(void)
{/*模块初始化*/OLED_Init();		//OLED初始化Key_Init();			//按键初始化TIM2_PWM_Init();	//定时器2PWM初始化Timer_Init();Encoder_Init();/*OLED显示静态字符*/Initial_Display();while (1){Key_control();			//按键PWM控制}
}//中断服务函数
//每次TIM3溢出时触发中断,调用Key_Tick()进行按键扫描
//清除中断标志,避免重复进入中断
void TIM3_IRQHandler(void)
{if (TIM_GetITStatus(TIM3, TIM_IT_Update) == SET){Key_Tick();TIM_ClearITPendingBit(TIM3, TIM_IT_Update);}
}

实验现象2

在这里插入图片描述

在这里插入图片描述

问题与解决

一上电程序卡死,原因是Timer3的中断服务函数忘记清除相应的标志位。

总结

旋转编码器和电位器控制LED亮暗的区别
在这里插入图片描述核心逻辑在于旋转编码器时中断服务函数检测旋转方向,更新计数值,而电位器时ADC采样。

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

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

相关文章

图扑可视化点亮智慧城市垃圾分类新未来

图扑基于 HT 开发的智慧城市废弃物可视化管理系统&#xff0c;通过智能感知与三维可视化技术&#xff0c;构建全流程数字化监管平台。系统实现固体废物从源头投放到终端处置的全程可视化追踪&#xff0c;提供智能收运路径规划与资源回收管理方案&#xff0c;助力城市环境治理向…

Elasticsearch安全加固指南:启用登录认证与SSL加密

在之前文章中我们介绍了Elasticsearch安全与权限控制&#xff0c;本篇文章我们将详细介绍 启用登录认证与SSL加密实践配置操作 。 1 为什么需要安全加固&#xff1f; Elasticsearch默认不启用安全功能&#xff0c;会导致以下风险&#xff1a; 未授权访问&#xff1a;任何人都能…

前端知识点---本地存储(javascript)

localStorage 是浏览器提供的一个 本地存储 API&#xff0c;可以在用户的浏览器中存储数据&#xff0c;数据不会随页面刷新而丢失。 1. 基本用法 (1) 存储数据&#xff08;setItem&#xff09; localStorage.setItem("username", "zhangsan");存储 “use…

神经网络能不能完全拟合y=x² ???

先说结论&#xff1a;关键看激活函数的选择 ReLU神经网络对非线性函数的拟合分析 ReLU神经网络对非线性函数&#xff08;如 y x 2 y x^2 yx2&#xff09;的拟合只能是逼近&#xff0c;而无法实现数学意义上的完全重合。这一结论源于ReLU的分段线性本质与目标函数的非线性结…

14.流程自动化工具:n8n和家庭自动化工具:node-red

n8n 安装 docker方式 https://docs.n8n.io/hosting/installation/docker/ #https://hub.docker.com/r/n8nio/n8n docker pull n8nio/n8n:latest docker rm -f n8n; docker run -it \ --network macvlan --hostname n8n \ -e TZ"Asia/Shanghai" \ -e GENERIC_TIME…

哈密尔顿路径(Hamiltonian Path)及相关算法题目

哈密尔顿路径要求访问图中每个顶点恰好一次&#xff0c;通常用于解决旅行商问题&#xff08;TSP&#xff09;或状态压缩DP问题。 哈密尔顿路径&#xff08;Hamiltonian Path&#xff09;是指在一个图中经过每个顶点恰好一次的路径。如果这条路径的起点和终点相同&#xff08;即…

面试算法高频02-树

树类型对比 数据结构定义节点特点遍历方式常见操作时间复杂度&#xff08;平均&#xff09;时间复杂度&#xff08;最坏&#xff09;空间复杂度&#xff08;最坏&#xff09;与其他结构关系应用场景树有根节点&#xff0c;分层级&#xff0c;包含父子、兄弟节点及子树关系的非…

数论4 组合数

目录 前言 求法一 代码 求法二 代码 求法三 代码 求法四 代码 前言 今天要将最后一部分&#xff0c;主要涉及组合数的四种求法。 前置知识 组合数的通项公式&#xff1a; 组合数的递推公式&#xff1a; 卢卡斯定理&#xff1a; 我们今天需要求的四种求法主要基…

构建自己的私有 Git 服务器:基于 Gitea 的轻量化部署实战指南

对于个人开发者、小型团队乃至企业来说&#xff0c;将项目代码托管在 GitHub、Gitee 等公共平台虽然方便&#xff0c;但也存在一定的隐私与可控性问题。 搭建一套私有 Git 代码仓库系统&#xff0c;可以实现对源码的完全控制&#xff0c;同时不依赖任何第三方平台&#xff0c;…

Linux操作系统 4.Linux实用操作

一、各类小技巧&#xff08;快捷键&#xff09; 1.CTRL C 强制停止 1.Linux某些程序的运行&#xff0c;如果想要强行停止它&#xff0c;可以使用ctrlc 2.命令输入错误&#xff0c;也可以通过快捷键ctrl c,退出当前输入&#xff0c;重新输入&#xff0c;或者ctrlc跳过当前这…

react redux的学习,单个reducer

redux系列文章目录 一 什么redux&#xff1f; redux是一个专门用于做状态管理的JS库(不是react插件库)。它可以用在react, angular, vue等项目中, 但基本与react配合使用。集中式管理react应用中多个组件共享的状 简单来说&#xff0c;就是存储页面的状态值的一个库&#xf…

PCI与PCIe接口的通信架构是主从模式吗?

PCI&#xff08;Peripheral Component Interconnect&#xff09;总线在通信架构上本质是主从模式&#xff0c;但其具体实现和角色分配在不同版本&#xff08;如传统PCI与PCI Express&#xff09;中存在差异。以下是详细分析&#xff1a; 传统PCI总线的主从模式 (1) 基本架构 主…

java项目挂机自动重启操作指南

前段时间有个伙伴问我&#xff0c;java项目挂机怎么自动重启。。。。。。今天就写一个 .sh脚本来实现应用挂机的自动重启功能 #!/bin/bash # 查询mita的进程个数 countps -ef | grep mita.jar | grep -v "grep" | wc -l # echo $count nowtimedate "%Y-%m-%d %H…

开放最短路径优先 - OSPF【LSA详细】

目录 LSA的头部结构 LSA类型 LSA数据包 LSA的主要作用是传递路由信息。 LSA的头部结构 共占20个字节&#xff0c;不同类型的LSA头部字段部分都是相同的。 链路状态老化时间(Link-State Age) 2个字节。指示该条LSA的老化时间&#xff0c;即它存在了多长时间&#xff0c;单位…

SpringBoot+Spring+MyBatis相关知识点

目录 一、相关概念 1.spring框架 2.springcloud 3.SpringBoot项目 4.注解 5.SpringBoot的文件结构 6.启动类原理 二、相关操作 1.Jar方式打包 2.自定义返回的业务状态码 3.Jackson 4.加载配置文件 5.异常处理 三、优化配置 1.简化sql语句 2.查询操作 复杂查询 一…

《双影奇境》手机版上线?ToDesk用跨平台技术实现「全设备云电脑3A游戏」

《双影奇境》是由Hazelight Studios研发发行的一款双人合作冒险类游戏&#xff0c;玩家们在游戏中将扮演米欧和佐伊两位风格迥异的女作家&#xff0c;剧情讲述的是她们被骗进入一台意在窃取创意的机器后便陷入了自己创作的故事之中&#xff0c;并且必须相互依靠&#xff0c;努力…

【教程】Windows下 Xshell 连接跳板机和开发机

需求 使用远程连接工具 Xshell 连接跳板机&#xff0c;再从跳板机连接开发机&#xff0c;用户登陆方式为使用密钥。 方法 首先&#xff0c;建立一个会话&#xff0c;用于配置跳板机信息和开发机转跳信息&#xff1a; 在【连接】页面&#xff0c;给跳板机取个名字&#xff0c…

如何快速入门物联网单片机开发?

背景 物联网单片机硬件开发涉及多个阶段&#xff0c;元器件是否“自己设计”取决于具体需求。以下是详细解答和学习方案&#xff1a; 一、元器件是否自己设计&#xff1f; 通用元器件&#xff1a; 大多数情况下&#xff0c;开发者直接使用现成的标准化元器件&#xff08;如电阻…

每日一题(小白)模拟娱乐篇11

由题可知就是要求计算一个数字&#xff0c;可以整除10进制的每一位&#xff0c;亦可以整除8进制和16进制的每一位。要求找出第2023个能够在三个进制下同时被10进制整除的数字。 Java中已经封装了进制转换的方法&#xff0c;以下是一些常用的转换方法&#xff1a;&#x1f447;…

阿里巴巴langengine二次开发大模型平台

阿里巴巴LangEngine开源了&#xff01;支撑亿级网关规模的高可用Java原生AI应用开发框架 - Leepy - 博客园 阿里国际AI应用搭建平台建设之路(上) - 框架篇 基于java二次开发 目前Spring ai、spring ai alibaba 都是java版本的二次基础能力 重要的是前端工作流 如何与 服务端的…