| 以下两种延时方式来源:Arduino_Core_STM32源码delayMicroseconds(uint32_t us)函数的实现。 
 利用SysTick再实现微秒延时函数 虽然SysTick已经被配置为1ms中断一次的模式,但每个1ms之间SysTick的当前值寄存器是一直在计数的(每计一个数的时间是1/SytemCoreClock)我们便可以利用该机制实现微秒延时函数。 
 void delayMicroseconds(uint32_t us) {     __IO uint32_t currentTicks = SysTick->VAL;   /* Number of ticks per millisecond */   const uint32_t tickPerMs = SysTick->LOAD + 1;   /* Number of ticks to count */   const uint32_t nbTicks = ((us - ((us > 0) ? 1 : 0)) * tickPerMs) / 1000;   /* Number of elapsed ticks */   uint32_t elapsedTicks = 0;   __IO uint32_t oldTicks = currentTicks;   do {     currentTicks = SysTick->VAL;     elapsedTicks += (oldTicks < currentTicks) ? tickPerMs + oldTicks - currentTicks :                     oldTicks - currentTicks;     oldTicks = currentTicks;   } while (nbTicks > elapsedTicks); } 
 以上函数可以直接复制到工程中使用,不需要额外的任何配置。 Note 
 虽然函数参数us为uint32_t类型,但是延时数不能过大,原因自己分析。建议超过1ms的延时时间使用HAL_Delay()。 
 利用DWT(数据观测点)实现微秒延时函数 对于DWT大家可以搜索具体了解,这里我也不是很了解,就直说实现方法好了。 
 dwt.h文件 /**   ******************************************************************************   * @brief   Header for dwt.c module   ******************************************************************************   * @attention   *   * Copyright (c) 2019, STMicroelectronics   * All rights reserved.   *   * This software component is licensed by ST under BSD 3-Clause license,   * the "License"; You may not use this file except in compliance with the   * License. You may obtain a copy of the License at:   *                        opensource.org/licenses/BSD-3-Clause   *   ******************************************************************************   */ 
 /* Define to prevent recursive inclusion -------------------------------------*/ #ifndef _DWT_H_ #define _DWT_H_ 
 #include "stm32f4xx.h" #include 
 #ifndef UNUSED #define UNUSED(x) (void)x #endif 
 #ifdef DWT_BASE uint32_t dwt_init(void); 
 #ifdef __cplusplus extern "C" { #endif 
 //uint32_t dwt_init(void); void dwt_access(bool ena); 
 static inline uint32_t dwt_max_sec(void) {   return (UINT32_MAX / SystemCoreClock); }; 
 static inline uint32_t dwt_max_msec(void) {   return (UINT32_MAX / (SystemCoreClock / 1000)); }; 
 static inline uint32_t dwt_max_usec(void) {   return (UINT32_MAX / (SystemCoreClock / 1000000)); }; 
 static inline uint32_t dwt_getCycles(void) {   return (DWT->CYCCNT); }; 
 #ifdef __cplusplus } #endif 
 #endif /* DWT_BASE */ #endif /* _DWT_H_ */ 
 dwt.c文件 /**   ******************************************************************************   * @file    dwt.c   * @author  Frederic Pillon   * @brief   Provide Data Watchpoint and Trace services   ******************************************************************************   * @attention   *   * Copyright (c) 2019, STMicroelectronics   * All rights reserved.   *   * This software component is licensed by ST under BSD 3-Clause license,   * the "License"; You may not use this file except in compliance with the   * License. You may obtain a copy of the License at:   *                        opensource.org/licenses/BSD-3-Clause   *   ******************************************************************************   */ 
 #include "dwt.h" 
 #ifdef DWT_BASE #ifdef __cplusplus extern "C" { #endif 
 
 uint32_t dwt_init(void) { 
   /* Enable use of DWT */   if (!(CoreDebug->DEMCR & CoreDebug_DEMCR_TRCENA_Msk)) {     CoreDebug->DEMCR |= CoreDebug_DEMCR_TRCENA_Msk;   } 
   /* Unlock */   dwt_access(true); 
   /* Reset the clock cycle counter value */   DWT->CYCCNT = 0; 
   /* Enable  clock cycle counter */   DWT->CTRL |=  DWT_CTRL_CYCCNTENA_Msk; 
   /* 3 NO OPERATION instructions */   __asm volatile(" nop      nt"                  " nop      nt"                  " nop      nt"); 
   /* Check if clock cycle counter has started */   return (DWT->CYCCNT) ? 0 : 1; } 
 void dwt_access(bool ena) { #if (__CORTEX_M == 0x07U)   /*    * Define DWT LSR mask which is (currentuly) not defined by the CMSIS.    * Same as ITM LSR one.    */ #if !defined DWT_LSR_Present_Msk #define DWT_LSR_Present_Msk ITM_LSR_Present_Msk #endif #if !defined DWT_LSR_Access_Msk #define DWT_LSR_Access_Msk ITM_LSR_Access_Msk #endif   uint32_t lsr = DWT->LSR; 
   if ((lsr & DWT_LSR_Present_Msk) != 0) {     if (ena) {       if ((lsr & DWT_LSR_Access_Msk) != 0) { //locked         DWT->LAR = 0xC5ACCE55;       }     } else {       if ((lsr & DWT_LSR_Access_Msk) == 0) { //unlocked         DWT->LAR = 0;       }     }   } #else /* __CORTEX_M */   UNUSED(ena); #endif /* __CORTEX_M */ } 
 #ifdef __cplusplus } #endif 
 #endif 
 delayMicroseconds()函数 void delayMicroseconds(uint32_t us) { #if defined(DWT_BASE) && !defined(DWT_DELAY_DISABLED)   int32_t start  = dwt_getCycles();   int32_t cycles = us * (SystemCoreClock / 1000000); 
   while ((int32_t)dwt_getCycles() - start < cycles); #endif } 
 |