【STM32嵌入式系统设计与开发】——12IWDG(独立看门狗应用)

这里写目录标题

  • 一、任务描述
  • 二、任务实施
    • 1、ActiveBeep工程文件夹创建
    • 2、函数编辑
      • (1)主函数编辑
      • (2)USART1初始化函数(usart1_init())
      • (3)USART数据发送函数( USART1_Send_Data())
      • (4)USART数据发送函数( USART1_IRQHandler())
      • (5)系统时间初始化函数( SystemTinerInit())
      • (6)等待计时函数( WaitTimerOut())
      • (7)系统时间定时器中断服务函数( TIM3_IRQHandler())
      • (8)获取系统计时时间函数( GetSystemTimer())
      • (9)外部中断4初始化函数( EXTIX_Init())
      • (10)外部中断4服务函数( EXTI4_IRQHandler())
      • (11)独立看门狗初始化函数(IWDG_Init())
      • (12)喂独立看门狗函数(IWDG_Feed())
    • 3、宏定义
      • 定时器宏定义
      • 中断宏定义
      • 独立看门狗宏定义
    • 4、知识链接
      • (1)独立看门狗
      • (2)独立看门狗时间计算
    • 5、工程测试


STM32资料包:
百度网盘下载链接:链接:https://pan.baidu.com/s/1mWx9Asaipk-2z9HY17wYXQ?pwd=8888
提取码:8888


一、任务描述

在这里插入图片描述

二、任务实施

观察电路图:
TXD(底板) ————————> PA10
RXD(底板) ————————> PA9
DK1(底板) ————————> PC4
D1 (底板) ————————> PA8
使用USB-AB型数据线,连接15核心板USB口,串口发送接收到的数据。在这里插入图片描述

1、ActiveBeep工程文件夹创建

步骤1:复制工程模板“1_Template”重命名为“9_IWDG”。
在这里插入图片描述

步骤2:修改项目工程名,先删除projects文件夹内除了Template.uvprojx文件外的所有内容并修改为“IWDG.uvprojx”。并删除output/obj和output/lst中的所有文件。
请添加图片描述

步骤3:运行“Exit.uvprojx”打开目标选项“Options for Target”中的“Output”输出文件,并修改可执行文件名称为“IWDG”点击“OK”保存设置。最后点击“Rebuild”编译该工程生成Usart文件。
请添加图片描述
步骤4:复制“2_SingleKey”中的"1_LED"和"SingleKey"文件复制到hardware中。
请添加图片描述
步骤5:在“system”中新建“iwdg”文件夹,并新建“iwdg.c”和“iwdg.h”文件。在这里插入图片描述
步骤5:工程组文件中添加“led.c”和“SingleKey.c”文件。
请添加图片描述
步骤5:工程组文件中添加“iwdg.c”和“iwdg.h”文件。
在这里插入图片描述
步骤6:目标选项添加添加头文件路径。
在这里插入图片描述

2、函数编辑

(1)主函数编辑

置的硬件设备,用于监视单片机的运行情况。如果程序出现了错误或者陷入了无限循环,独立看门狗就会启动,重置单片机,让其恢复到安全状态。
在这里插入图片描述
步骤1:端口初始化准备

	//函数初始化,端口准备delay_init();                   //启动滴答定时器usart1_init(9600);              //USART1初始化LED_Init();                     //板载LED初始化SystemTinerInit(1000-1,7200-1); //系统时间初始化 定时100msExpKeyInit();                   //开发板按键初始化LED = 0;  delay_ms(800);                  //让人看得到灭IWDG_Init(4,625);               //与分频数为64,重载值为625,溢出时间为1sLED = 1;delay_ms(800);

在这里插入图片描述
步骤2:实现一个简单的计时器,并在每秒打印一次计时信息。利用LED状态的改变来指示系统正在运行。

	while(1){	IWDG_Feed();//如果DK1按下,则喂狗LED = 0;delay_ms(100);LED = 1;delay_ms(100);if(!DK1)    //按下DK1按键delay_ms(1000);delay_ms(20);}	

在这里插入图片描述

(2)USART1初始化函数(usart1_init())

配置了 PA9 为复用推挽输出,用于 USART1 的 TXD,并配置了 PA10 为浮空输入,用于 USART1 的 RXD。并配置了 USART1 的参数,包括波特率、数据位长度、停止位数、校验位、硬件流控制和工作模式。

/*********************************************************************@Function  : USART1初始化@Parameter : bound : 波特率 @Return    : N/A
**********************************************************************/   	
void usart1_init(uint32_t bound)
{GPIO_InitTypeDef GPIO_InitStructure;             										          // 定义 GPIO 初始化结构体USART_InitTypeDef USART_InitStructure;            										          // 定义 USART 初始化结构体NVIC_InitTypeDef NVIC_InitStructure;              										          // 定义 NVIC 初始化结构体/* 时钟使能:启用 USART1 和 GPIOA 的时钟 */RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA, ENABLE);/* 引脚复用配置 */  // 配置 PA9 为复用推挽输出,用于 USART1 的 TXDGPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;   		                             // 设置 GPIO 端口GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;                                // 设置 GPIO 速度GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; 								 // 设置 GPIO 模式为复用推挽GPIO_Init(GPIOA, &GPIO_InitStructure);          							     // 初始化 GPIO// 配置 PA10 为浮空输入,用于 USART1 的 RXDGPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;                                      // 设置 GPIO 端口GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;                           // 设置 GPIO 模式为浮空输入GPIO_Init(GPIOA, &GPIO_InitStructure);                                          // 初始化 GPIO/* NVIC 中断配置 */ NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;                               // 设置中断通道为 USART1NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 3;                       // 设置抢占优先级为3NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;                              // 设置子优先级为3NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                                 // 使能中断通道NVIC_Init(&NVIC_InitStructure);                                                 // 初始化 NVIC/* USART1 配置 */ USART_InitStructure.USART_BaudRate = bound;                                     // 设置波特率USART_InitStructure.USART_WordLength = USART_WordLength_8b;                     // 设置数据位长度为8位USART_InitStructure.USART_StopBits = USART_StopBits_1;                          // 设置停止位为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);                                       // 初始化 USART1/*中断配置*/USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);                                //开接受中断 USART_ITConfig(USART1,USART_IT_IDLE,ENABLE);                                //开空闲中断USART_ITConfig(USART1,USART_IT_TXE,ENABLE);                                 //开发送中断	USART_Cmd(USART1, ENABLE);                                                  //启用USART1USART_DataTypeStr.Usart_Tc_State = SET;	                                    //置位发送允许标志	      
}

在这里插入图片描述

(3)USART数据发送函数( USART1_Send_Data())

初始化PD14端口,并为推挽输出。

/*********************************************************************@Function  : USART数据发送函数@Parameter : Data 	 :要发送的数据缓存.Lenth  :发送长度@Return    : 发送状态   1 :失败   0 :成功
**********************************************************************/
char USART1_Send_Data(char* Data,uint8_t Lenth) 
{uint8_t uNum = 0;if(USART_DataTypeStr.Usart_Tc_State == 1)                       //判断发送标志位是否置1{USART_DataTypeStr.Usart_Tc_State = 0;                       //将发送标志位清零,表示数据已经成功放入缓存,等待发送USART_DataTypeStr.Usart_Tx_Len = Lenth;                     //获取需要发送的数据的长度       for(uNum = 0;uNum < USART_DataTypeStr.Usart_Tx_Len;uNum ++)   //将需要发送的数据放入发送缓存{USART_DataTypeStr.Usart_Tx_Buffer[uNum] = Data[uNum];}USART_ITConfig(USART1,USART_IT_TXE,ENABLE);			            //数据放入缓存后打开发送中断,数据自动发送}return USART_DataTypeStr.Usart_Tc_State;                        //返回放数据的状态值,为1表示发送失败,为0表示发送成功了
}

在这里插入图片描述

(4)USART数据发送函数( USART1_IRQHandler())

/*********************************************************************@Function  : USART1中断服务函数@Parameter : N/A @Return    : N/A
**********************************************************************/
void USART1_IRQHandler(void)                
{uint8_t Clear = Clear;                                                                           // 定义清除标志的变量,并初始化为自身static uint8_t uNum = 0;                                                                          // 静态变量,用于循环计数if(USART_GetITStatus(USART1,USART_IT_RXNE) != RESET)                                                // 判断读数据寄存器是否为非空{USART_ClearFlag(USART1, USART_IT_RXNE);                                                           // 清零读数据寄存器,其实硬件也可以自动清零USART_DataTypeStr.Usart_Rx_Buffer[USART_DataTypeStr.Usart_Rx_Num ++] = \(uint16_t)(USART1->DR & 0x01FF);                                                              // 将接收到的数据存入接收缓冲区(USART_DataTypeStr.Usart_Rx_Num) &= 0xFF;                                                     // 防止缓冲区溢出} else if(USART_GetITStatus(USART1,USART_IT_IDLE) != RESET)   // 检测空闲{Clear = USART1 -> SR;                                                                         // 读SR位Clear = USART1 -> DR;                                                                       // 读DR位,USART_DataTypeStr.Usart_Rx_Len = USART_DataTypeStr.Usart_Rx_Num;                              // 获取数据长度for(uNum = 0; uNum < USART_DataTypeStr.Usart_Rx_Len; uNum ++)          {USART_DataTypeStr.Usart_Rx_Data[uNum] = USART_DataTypeStr.Usart_Rx_Buffer[uNum];      // 将接收到的数据复制到接收数据缓冲区}USART_DataTypeStr.Usart_Rx_Num = 0;                                                           // 清空接收计数器USART_DataTypeStr.Usart_Rc_State = 1;                                                         // 数据读取标志位置1,读取串口数据}if(USART_GetITStatus(USART1,USART_IT_TXE) != RESET)                                                  // 判断发送寄存器是否为非空{USART1->DR = \((USART_DataTypeStr.Usart_Tx_Buffer[USART_DataTypeStr.Usart_Tx_Num ++]) & (uint16_t)0x01FF);    // 发送数据(USART_DataTypeStr.Usart_Tx_Num) &= 0xFF;                                                       // 防止缓冲区溢出if(USART_DataTypeStr.Usart_Tx_Num >= USART_DataTypeStr.Usart_Tx_Len){   USART_ITConfig(USART1,USART_IT_TXE,DISABLE);                                                // 发送完数据,关闭发送中断USART_DataTypeStr.Usart_Tx_Num = 0;                                                         // 清空发送计数器USART_DataTypeStr.Usart_Tc_State = 1;                                                       // 发送标志置1,可以继续发送数据了} 		}}

在这里插入图片描述

(5)系统时间初始化函数( SystemTinerInit())

Tout=((arr+1)*(psc+1))/Ft us,Ft=定时器工作频率,单位:Mhz;初始化TIM3定时器,配置定时器的周期值、预分频值、计数模式等参数,并使能定时器及其中断

/*********************************************************************@Function  : 系统时间初始化@Parameter : arr:自动重装值。psc:时钟预分频数@Return    : N/A@Read 			:Tout=((arr+1)*(psc+1))/Ft us,Ft=定时器工作频率,单位:Mhz
**********************************************************************/
void SystemTinerInit(uint16_t arr, uint16_t psc)
{TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;               // 定义TIM基本参数结构体NVIC_InitTypeDef NVIC_InitStructure;                         // 定义中断优先级配置结构体/* 时钟使能 */RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);         // 使能TIM3时钟/* TIM配置 */TIM_TimeBaseStructure.TIM_Period = arr;                      // 设置定时器的周期值TIM_TimeBaseStructure.TIM_Prescaler = psc;                   // 设置定时器的预分频值TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;      // 设置时钟分频因子为1TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  // 设置计数模式为向上计数TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);              // 初始化TIM3定时器/* 允许中断 */TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE);                   // 使能TIM3更新(溢出)中断/* NVIC 配置 */NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;              // 设置TIM3中断通道NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;    // 设置TIM3中断的抢占优先级为0NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;           // 设置TIM3中断的子优先级为3NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;              // 使能TIM3中断通道NVIC_Init(&NVIC_InitStructure);                              // 初始化NVIC/* 使能TIMx */TIM_Cmd(TIM3, ENABLE);                                       // 使能TIM3定时器
}

(6)等待计时函数( WaitTimerOut())

定时器超时检测功能,根据传入的参数 gTimer 和系统时钟计数器,判断定时器是否超时,并返回相应的状态。

/*********************************************************************@Function  : 等待计时@Parameter : gTimer :等待时间,100ms一个单位@Return    : 1表示超时,0表示未超时
**********************************************************************/
uint8_t WaitTimerOut(uint32_t gTimer)
{	uint32_t GTr = 0;                         // 定义变量用于存储定时器剩余时间if(gTimer==0) return 1;                   // 如果等待时间为0,则直接返回1,表示不等待GTr = SystemTimer % gTimer;	              // 计算定时器剩余时间if((GTr==0) && (!Rti) && (Gti != gTimer)) // 如果定时器剩余时间为0,且上次未检测到超时,并且当前定时器时间不等于上次记录的时间{ Rti=1;                                // 设置标志表示检测到定时器超时Gti = gTimer;                         // 更新记录的定时器时间return 1;                             // 返回1表示超时}else if((GTr!=0) && (Rti))                // 如果定时器剩余时间不为0,且上次检测到超时,则将标志置为0Rti=0;if(!GetTimer) GetTimer = SystemTimer;	  // 如果记录定时器开始时间为0,则将其设置为当前系统时间if(SystemTimer - GetTimer == gTimer)      // 如果当前系统时间减去记录的定时器开始时间等于设定的等待时间,则返回1表示超时{ GetTimer = 0;                         // 将记录的定时器开始时间清零,准备下一次记录return 1;                             // 返回1表示超时}return 0;                                 // 返回0表示未超时
}

在这里插入图片描述

(7)系统时间定时器中断服务函数( TIM3_IRQHandler())

实现TIM3定时器的中断服务程序,每次定时器溢出时,增加 SystemTimer 计数值,并在计数到60时归零,同时清除中断标志位。

/*********************************************************************@Function  : 系统时间定时器中断服务函数@Parameter : N/A@Return    : N/A
**********************************************************************/
void TIM3_IRQHandler(void)   
{	// 检查定时器更新中断是否触发if(TIM_GetITStatus(TIM3, TIM_IT_Update) == SET) // 溢出中断{SystemTimer++;                                // 系统时间计数器加1if(SystemTimer == 60)	                        // 如果系统时间计数器达到60,则重置为0,并且清零记录的定时器开始时间{	SystemTimer = 0;GetTimer = 0;}}// 清除定时器更新中断标志位TIM_ClearITPendingBit(TIM3, TIM_IT_Update);     // 清除中断标志位
}

在这里插入图片描述

(8)获取系统计时时间函数( GetSystemTimer())

/*********************************************************************@Function  : 获取系统计时时间@Parameter : N/A@Return    : N/A
**********************************************************************/
uint32_t GetSystemTimer(void)
{return SystemTimer;
}

在这里插入图片描述

(9)外部中断4初始化函数( EXTIX_Init())

/*********************************************************************@Function  : 外部中断4初始化@Parameter : N/A@Return    : N/A
**********************************************************************/
void EXTIX_Init(void)
{EXTI_InitTypeDef EXTI_InitStructure;                      // 定义外部中断配置结构体NVIC_InitTypeDef NVIC_InitStructure;                      // 定义中断控制器配置结构体/*时钟使能*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);	      // 使能 AFIO 时钟,用于配置外部中断的映射/*中断线配置*/   GPIO_EXTILineConfig(GPIO_PortSourceGPIOC, GPIO_PinSource4); // 配置外部中断线,将 PC4 映射到外部中断4EXTI_InitStructure.EXTI_Line = EXTI_Line4;	              // 设置外部中断线为 EXTI4EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;	      // 设置外部中断模式为中断模式EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;     // 设置触发方式为下降沿触发EXTI_InitStructure.EXTI_LineCmd = ENABLE;                   // 使能外部中断线EXTI_Init(&EXTI_InitStructure);	 	                      // 初始化外部中断配置/*NVIC配置*/NVIC_InitStructure.NVIC_IRQChannel = EXTI4_IRQn;	          // 设置中断向量为外部中断4NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x02;// 设置抢占优先级为2NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x03;       // 设置子优先级为3NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;	          // 使能外部中断4NVIC_Init(&NVIC_InitStructure);                             // 初始化中断控制器配置/*关闭蜂鸣器*/beep = 0;                                                 // 初始化蜂鸣器状态为关闭
}

(10)外部中断4服务函数( EXTI4_IRQHandler())

/*********************************************************************@Function  : 外部中断4服务程序@Parameter : N/A@Return    : N/A
**********************************************************************/
void EXTI4_IRQHandler(void)
{delay_ms(10);//消抖if(DK1==0)				 beep =!beep;	EXTI_ClearITPendingBit(EXTI_Line4); //清除LINE4上的中断标志位  
}

在这里插入图片描述

(11)独立看门狗初始化函数(IWDG_Init())

/*********************************************************************@Function  : 初始化独立看门狗@Parameter : prer : 分频数:0~7(只有低3位有效!)rlr  : 重装载寄存器值:低11位有效.@Return    : N/A@Read 			: 1、分频因子=4*2^prer.但最大值只能是256!2、时间计算(大概):Tout=((4*2^prer)*rlr)/40 (ms).
**********************************************************************/
void IWDG_Init(uint8_t prer,uint16_t rlr) 
{	IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable);  //使能对寄存器IWDG_PR和IWDG_RLR的写操作IWDG_SetPrescaler(prer);                       //设置IWDG预分频值:设置IWDG预分频值为64IWDG_SetReload(rlr);                           //设置IWDG重装载值IWDG_ReloadCounter();                          //按照IWDG重装载寄存器的值重装载IWDG计数器IWDG_Enable();                                 //使能IWDG
}

在这里插入图片描述

(12)喂独立看门狗函数(IWDG_Feed())

/*********************************************************************@Function  : 喂独立看门狗@Parameter : N/A@Return    : N/A@Read 			: 不喂狗会自动复位系统				
**********************************************************************/
void IWDG_Feed(void)
{   IWDG_ReloadCounter();                          //重新加载						   
}

在这里插入图片描述

3、宏定义

步骤1:主函数添加所需的头文件,主源文件部分报错消失

#include ".\iwdg\iwdg.h"/***********Hardweare***************/
#include "led.h"
#include "SingleKey.h"

在这里插入图片描述

步骤2:添加中断源文件所需的头文件

#include ".\iwdg\iwdg.h"
#include "stm32f10x_iwdg.h" 

在这里插入图片描述

步骤3:添加宏定义

#define USART_RX_LEN  200               // 接收缓冲区最大长度
#define USART_TX_LEN  200               // 发送缓冲区最大长度
#define UART_NUM      10                // 串口结构体最大对象数量

在这里插入图片描述
步骤4:添加函数声明

void usart1_init(uint32_t bound);
extern USART_DataTypeDef USART_DataTypeStr; 
char USART1_Send_Data(char* Data,uint8_t Lenth);

在这里插入图片描述
步骤5:添加数据类型和宏的头文件

//定义串口数据结构体
typedef struct USART_DataType 
{uint8_t Usart_Rx_Len;          // 接收缓冲区长度uint8_t Usart_Tx_Len;          // 发送缓冲区长度uint8_t Usart_Rx_Num;          // 接收数据计数uint8_t Usart_Tx_Num;          // 发送数据计数uint8_t Usart_Rc_State;        // 接收状态标志位uint8_t Usart_Tc_State;        // 发送状态标志位char Usart_Rx_Buffer[USART_RX_LEN]; // 接收缓冲区char Usart_Tx_Buffer[USART_TX_LEN]; // 发送缓冲区char Usart_Rx_Data[USART_RX_LEN];   // 接收数据char Usart_Tx_Data[USART_TX_LEN];   // 发送数据
} USART_DataTypeDef;

在这里插入图片描述
步骤6:定义一个串口数组变量

USART_DataTypeDef USART_DataTypeStr={0};

在这里插入图片描述

定时器宏定义

步骤1:创建一个宏定义保护

#ifndef __TIMER_H
#define __TIMER_H#endif

在这里插入图片描述

步骤2:添加函数声明

void SystemTinerInit(uint16_t arr,uint16_t psc);//系统时间初始化函数
uint32_t GetSystemTimer(void);                  //获取系统计时时间函数
uint8_t WaitTimerOut(uint32_t gTimer);          //等待计时函数

在这里插入图片描述

步骤3:添加数据类型和宏的头文件

#include <stdint.h> 

在这里插入图片描述

中断宏定义

步骤1:创建一个宏定义保护

#ifndef __TIMER_H
#define __TIMER_H#endif

在这里插入图片描述
步骤2:添加函数声明

void EXTIX_Init(void);	

在这里插入图片描述
步骤3:添加数据类型和宏的头文件

#include <stdint.h> 

在这里插入图片描述

独立看门狗宏定义

步骤1:创建一个宏定义保护

#ifndef __IWDG_H
#define __IWDG_H#endif

在这里插入图片描述
步骤2:添加函数声明

void IWDG_Init(uint8_t prer,uint16_t rlr);
void IWDG_Feed(void);

在这里插入图片描述
步骤3:添加数据类型和宏的头文件

#include <stdint.h> 

在这里插入图片描述

4、知识链接

(1)独立看门狗

在这里插入图片描述

在 STM32 单片机中,独立看门狗也是类似的。它是一个内置的硬件设备,用于监视单片机的运行情况。如果程序出现了错误或者陷入了无限循环,独立看门狗就会启动,重置单片机,让其恢复到安全状态,以防止系统崩溃或者出现不可预料的问题。就像在厨房里一样,独立看门狗在单片机中扮演着保护系统安全的角色。

(2)独立看门狗时间计算

初始化独立看门狗为1S:
IWDG_Init(uint8_t prer,uint16_t rlr)
时间计算(大概):Tout=((4*2^prer)rlr)/40 (ms)
分频因子=4
2^prer.
但最大值只能是256!
1000ms = 4x2^4x625/40ms

5、工程测试

在这里插入图片描述

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

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

相关文章

苹果App Store上架工具介绍

文章目录 摘要引言正文1. Xcode2. [appuploder](https://www.applicationloader.net/)3. [克魔助手](https://keymob.com/) 4.[ipa guard](https://www.ipaguard.com/)总结参考资料 摘要 苹果App Store作为iOS应用程序的主要分发渠道&#xff0c;上架应用程序需要遵守规定和通…

Hive3.0.0出库库表中timestamp字段读取为null

在利用sqoop1.99.7做数据迁移的时候&#xff0c;从mysql导出表格到hive建立对应的表格&#xff0c;字段中使用了timestamp类型&#xff0c;在读取数据的时候&#xff0c;发现数据为null。查找问题方法如下&#xff1a; 1、查询库表字段类型 命令&#xff1a;desc tablen…

00000基础搭建vue+flask前后端分离项目

我完全是参考的这个vue3flask前后端分离环境速建_flask vue3-CSDN博客 安装了node_js&#xff08;添加了环境变量&#xff09; 环境变量 把原来的镜像源换成了淘宝镜像源 npm config set registry https://registry.npmmirror.com/ 查看版本证明安装成功 npm - v 安装npm i…

正则表达式不会用?一篇教你快速搞懂 !

目录 前言一、基础字符二、一系列常用的字符&#xff1b;1、一些元字符&#xff08;Meta-characters&#xff09; 三、一些高级概念1、贪婪与懒惰匹配2、两个实例加深理解1.颜色值的匹配&#xff1a;RGBS值2.ipv4 地址匹配 四、正则表达式常用语法**1.Flags&#xff08;标志符或…

c语言中动态内存管理

说到内存&#xff0c;大家一定都知道。但是有一种函数可以实现动态内存管理&#xff0c;下面大家一起学习。 文章目录 一、为什么要有动态内存管理&#xff1f;二、malloc 和 free1.malloc2.free 三、calloc 和 realloc1.calloc2.realloc3.常见的动态内存的错误3.1对NULL指针的…

面试题 之 webpack

1.说说你对webpack理解&#xff1f;解决什么问题&#xff1f; Webpack 是实现前端项目的模块化&#xff0c;用于现代 JavaScript 应用程序的静态模块打包工具&#xff0c;被webpack 直接引用的资源打包进 bunde.js的资源&#xff0c;当webpack 处理应用程序时,它会在内部构建一…

理解CPU与执行指令原理

本文侧重介绍cpu的工作任务&#xff0c;与cpu执行指令的过程是怎么样的&#xff1f; 目录 1.理解CPU 1.1.CPU的功能 1.2.CPU的逻辑构成 2.认识指令 2.1.什么是指令 2.2.CPU执行指令的准备工作(重点) 3.指令的执行过程 前景知识&#xff1a; 什么是计算机 就是遵循冯诺依…

阿里云部署宝塔,设置了安全组还是打不开。

1.在安全组是开放正确的端口好。8888要开&#xff0c;但是不只是开放8888&#xff0c;举个例子&#xff0c;https://47.99.53.222:17677/49706cf7这个&#xff0c;要开放17677这个端口号。 2.安全组要挂载到实例上&#xff0c;从三个点的进入点击管理实例&#xff0c;加到对应的…

深入聊聊企业数字化转型这个事儿

01 什么是数字化&#xff1f; 聊数字化&#xff0c;就不得不聊聊信息化、智能化。佛性的说&#xff1a;信息化是数字化的前世&#xff0c;智能化是数字化的来生&#xff01;我习惯用一个结构化的图形来表示事物之间的关系&#xff0c;信息化、数字化、智能化的关系如下&#…

[flask] flask的基本介绍、flask快速搭建项目并运行

笔记 Flask Flask 本身相当于一个内核&#xff0c;其他几乎所有的功能都要用到扩展&#xff08;邮件扩展Flask-Mail&#xff0c;用户认证Flask-Login&#xff0c;数据库Flask-SQLAlchemy&#xff09;&#xff0c;都需要用第三方的扩展来实现。比如可以用 Flask 扩展加入ORM、…

JVM本地方法

本地方法接口 NAtive Method就是一个java调用非java代码的接口 本地方法栈&#xff08;Native Method Statck&#xff09; Java虚拟机栈用于管理Java方法的调用&#xff0c;而本地方法栈用于管理本地方法的调用。 本地方法栈&#xff0c;也是线程私有的。 允许被实现成固定或…

想做抖音小店又不会该怎么办?先学会做店出单逻辑,再入门

大家好&#xff0c;我是电商花花。 现在在这个巨大的电商市场中&#xff0c;很多人都被电商的巨大红利给勾起来了&#xff0c;在这个抖音小店的黑马项目中&#xff0c;很多人都在其中赚钱获利&#xff0c;吸引了一批又一批商家。 相信现在对电商感兴趣的仍不在少数&#xff0…

Navicat BI 工具 | 连接数据

早前&#xff0c;海外 LearnBI online 博主 Adam Finer 对 Navicat Charts Creator 这款 BI&#xff08;商业智能&#xff09;工具进行了真实的测评。上一期&#xff0c;我们介绍了这位博主对 Navicat BI 工具的初始之感。今天&#xff0c;我们来看看从连接数据的角度&#xff…

Tomcat 下载以及安装

Tomcat安装及配置教程主要分为四步&#xff1a; 步骤一&#xff1a;首先确认自己是否已经安装JDK 1. cmd&#xff1a;查看java的版本 步骤二&#xff1a;下载安装Tomcat 1. 下载tomcat :Apache Tomcat - Welcome! 2. 选择对应的tomcat版本&#xff1a; 3. 进行安装&#…

C# 登录界面代码

背景 MVVM 是一种软件架构模式&#xff0c;用于创建用户界面。它将用户界面&#xff08;View&#xff09;、业务逻辑&#xff08;ViewModel&#xff09;和数据模型&#xff08;Model&#xff09;分离开来&#xff0c;以提高代码的可维护性和可测试性。 MainWindow 类是 View&a…

38 mars3d 对接地图图层 绘制点线面员

前言 这里主要是展示一下 mars3d 的一个基础的使用 主要是设计 接入地图服务器的 卫星地图, 普通的二维地图, 增加地区标记 基础绘制 点线面园 等等 测试用例 <template><div style"width: 1920px; height:1080px;"><div class"mars3dClas…

【论文笔记】RobotGPT: Robot Manipulation Learning From ChatGPT

【论文笔记】RobotGPT: Robot Manipulation Learning From ChatGPT 文章目录 【论文笔记】RobotGPT: Robot Manipulation Learning From ChatGPTAbstractI. INTRODUCTIONII. RELATED WORK1. LLMs for Robotics2. Robot Learning III. METHODOLOGY1. ChatGPT Prompts for Robot …

【leetcode】双“指针”

标题&#xff1a;【leetcode】双指针 水墨不写bug 我认为 讲清楚为什么要用双指针 比讲怎么用双指针更重要&#xff01; &#xff08;一&#xff09;快乐数 编写一个算法来判断一个数 n 是不是快乐数。 「快乐数」 定义为&#xff1a; 对于一个正整数&#xff0c;每一次将该数…

我们常用Linux命令总结

Linux作为一种自由和开放源代码的操作系统&#xff0c;广泛应用于各种计算机系统中&#xff0c;尤其是服务器环境。在Linux系统中&#xff0c;命令行是管理和操作系统的主要方式之一&#xff0c;熟练掌握常用的Linux命令对于系统管理员、开发人员和其他使用者来说都是至关重要的…

HDLBits刷题Day28,3.2.5.14 3.2.5.14 one-hot FSM

3.2.5.14 one-hot FSM 问题描述 给定以下具有 1 个输入和 2 个输出的状态机&#xff1a; 假设此状态机使用 one-hot 编码&#xff0c;其中state[0]到state[9]分别对应于状态 S0 到 S9。除非另有说明&#xff0c;否则输出为零。 仅实现状态机的状态转换逻辑和输出逻辑部分。您在…