用STM32手搓一个体温计

这篇具有很好参考价值的文章主要介绍了用STM32手搓一个体温计。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

前言

市面上大多额温计、测温计精度偏差太大

而水银温度计,等待时间又太久

因此就产生了用STM32自己做一个——精度高、便宜、测速快 产品级 温度仪/体温计的想法。

用STM32手搓一个体温计,c++

01

这个温度仪的亮点?

1精度高,误差小于0.1度

2带有显示屏,可实时查看测温数据

3支持蓝牙传输,支持BLE和SPP

4可直接用手机或电脑接收温度数据

5可以DIY设置报警功能,方便看护儿童时,不知道什么时候又温度升高了,都不敢打个盹

6运行时间长,可以连续使用12个小时以上

7解决 儿童生病时不愿意配合量体温,以及腋下量体温时,不能实时了解发烧情况

那如何通过【硬件、外型、软件设计】实现出这样一个产品级的体温计呢?一起来看看!

02

硬件设计

用STM32手搓一个体温计,c++

系统框图

用STM32手搓一个体温计,c++

原理图

用STM32手搓一个体温计,c++

PCB

下面就将电路拆分为5个部分,一 一分析一下电路原理。

01

电源部分

充放电芯片采用TP4057,该芯片便宜,支持升压,确保系统供电稳定。

这里通过Type-C给产品充电。

由于系统采用3.3V供电,因此用LDO进行电压转换。

用STM32手搓一个体温计,c++

其锂电池采用200ma容量,规格如下:

用STM32手搓一个体温计,c++

连接方式是焊接,预留了2.54排针,方便调试。

用STM32手搓一个体温计,c++

Type-C选择了microUSB

用STM32手搓一个体温计,c++

02主控部分

系统主控采用stm32f030c8t6。

用STM32手搓一个体温计,c++

该主控可实现以下功能:

  • ADC采集,也就是检测锂电池电量;

  • 串口通信,控制蓝牙芯片,进行数据发送;

  • IIC通信,控制温度传感器,配置以及初始化,最重要的是读取温度

  • IIC通信,显示电量、蓝牙连接状、实时温度、测量时长

  • 其它可扩展功能

03蓝牙部分

这里采用KT6368A设计。

该芯片支持双模,具体规格如下:

用STM32手搓一个体温计,c++

PS:为了降低功耗建议使用KT6328A,但是这个只支持BLE哦。

相关原理图如下所示:

用STM32手搓一个体温计,c++

04显示部分

自己采用了0.91寸的OLED模块,只有4个管脚,如下所示:

用STM32手搓一个体温计,c++

我用的是白光的,我试了蓝光的也挺好的,这个看个人喜好了。

用STM32手搓一个体温计,c++

05温度传感器

温度传感器在本项目中,无疑是很重要的存在。

MTS4和T117是我能找到的最具性价比的芯片,精度高且便宜。

T117参数:

用STM32手搓一个体温计,c++

  • 超低温至-103℃测温!

  • 测温精度±0.1℃@+28~+43℃

  • I2C/单总线数字输出

  • 分辨率16位/0.004℃

  • 测温最快2.2ms可配置

  • 支持4路I2C通信地址;

  • 超低功耗2μA@1HZ

  • 1.8V~5.5V宽电压供电

  • 2*2*0.75mm小尺寸

  • DFN6L封装

  • 内置112bit EEPROM用户空间。

MTS4参数:

用STM32手搓一个体温计,c++

  • I2C协议数字输出

  • 可同时兼容数字单总线

  • 测温精度±0.1℃@+28℃~+43℃

  • 16位输出0.004℃

  • 测温速度15.3ms/8.5ms/5.2ms/2.2ms可配置

  • 超低功耗3.1μA(AVG=16,1次测量/s)

  • 1.6x1.2x0.55mm超小尺寸

  • DFN4L封装

  • 内置112bit EEPROM用户空间

对应原理图:

用STM32手搓一个体温计,c++

06PCB部分

PCB采用4层板设计,板厚1.6mm。为了能打免费PCB,整体长度设计在10cm以内。

这里的装配孔是m3,建议可以m2。

用STM32手搓一个体温计,c++

03

外型设计

外型我做了3D外壳和面板。

01

外壳部分

3D外壳使用嘉立创EDA设计。

由于我是个新手,所以设计的比较简单。

用STM32手搓一个体温计,c++

实物是这样子:

用STM32手搓一个体温计,c++

02

面板部分

这个也是采用嘉立创EDA设计的,为了省钱。

*因为有很多位置需要计算,所以这里有个技巧,可以先导出PCB的DXF,方便定位。

做了很多颜色,实现3D外壳和面板自由了。

设计如下:

用STM32手搓一个体温计,c++

实物如下:

用STM32手搓一个体温计,c++

04

程序设计

”01

开发工具和环境

采用CubeMXKeil5结合的方式,使用C语言进行开发。

02程序设计思路

1电量采样通过timer+DMA+DAC进行数据采集,timer定时2S,为啥2秒,降低功耗。

之前1s感觉刷新有点快。

2检测蓝牙是否连接,读取PA1管脚状态,若连接显示器显示OK,否则是NO。

3每两秒读取一次温度,根据状态寄存器进行判别是否温度转换完成。

4

OLED显示2s更新一次。

效果如下图所示:

用STM32手搓一个体温计,c++

03

温度传感器驱动

目前只焊接成功了T117芯片,这里给大家展示一下这个驱动的编写,如下所示。

这里是头文件

C++
#ifndef __SENSER_H#define __SENSER_H
#include "stm32f0xx_hal.h"
typedef enum{//Temp_Cmd//测温模式CONTI_CONVERT = 0x00, //连续测量温度STOP_CONVERT = 0x40, //停止测量温度SINGLE_CONVERT = 0xc0, //单次测量温度//加热模式OFF_HEATING = 0xf0, //低位清0,加热关闭ON_HEATING = 0x0A, //加热开启
//Temp_Cfg//测温频率FRE_8times = 0x00, //每秒8次FRE_4times = 0x20, //每秒4次FRE_2times = 0x40, //每秒2次FRE_1times = 0x60, //每秒1次
FRE_2s = 0x80, //每2秒1次FRE_4s = 0xa0, //每4秒1次FRE_8s = 0xc0, //每8秒1次FRE_16s = 0xe0, //每16秒1次//平均次数AVG_1 = 0xe7, //位清0,转换时间2.1msAVG_8 = 0x08, //转换时间5.2msAVG_16 = 0x10, //转换时间8.5msAVG_32 = 0x18, //转换时间15.3ms//低功耗模式OFF_PD = 0xfe, //位清0,不进入低功耗模式ON_PD = 0x01, //进入低功耗模式
//EE_CmdEE_DOWN = 0xb6, //装载EE值到寄存器EE_COPY = 0x08, //将寄存器中数值保存到EE中EE_RESET = 0x6a, //软复位,装载EE值到寄存器,与EE对应的部分,寄存器值恢复到EE保存值,不与EE对应的部分,寄存器值恢复到默认值
//Alert_Mode//报警开关OFF_ALERT = 0x00, //清0,报警关ON_ALERT = 0x80, //报警开//ModeTL_CLEAR = 0xbf, //位清0,TL为报警清除门限阈值TL_ALERT = 0x40, //TL为报警门限下阈值//极性ALERT_LO = 0xdf, //位清0,低电平有效ALERT_HI = 0x20, //高电平有效//报警端口模式选择ALERT_IO = 0xef, //位清0,用作温度报警CONVERT_FINI = 0x10, //用作测温完成标志} I2C_CMD;

typedef enum{Temp_lsb = 0x00, //Temp_msb = 0x01, //Crc_temp = 0x02, //Status = 0x03, //Temp_Cmd = 0x04, //默认值0x40:停止测量,不加热Temp_Cfg = 0x05, //默认值0x69:每秒1次,AVG_8,进入低功耗Alert_Mode = 0x06, //默认值0x00:报警关,报警模式为TL解除报警,报警低电平有效,标志位表示温度报警Th_lsb = 0x07,Th_msb = 0x08,Tl_lsb = 0x09,Tl_msb = 0x0A,Crc_scratch = 0x0B,EE_Cmd = 0x17, //默认值0x00:无操作Romcode1 = 0x18,Romcode2 = 0x19,Romcode3 = 0x1A,Romcode4 = 0x1B,Romcode5 = 0x1C,Romcode6 = 0x1D,Romcode7 = 0x1E,crc_romcode = 0x1F,} REG;
void T117_Init(void);uint8_t T117_ID(void);//获取IDuint8_t T117_R_REG(uint8_t REG,uint8_t *DAT);//读寄存器uint8_t T117_W_REG(uint8_t REG,uint8_t DAT);//写寄存器uint8_t T117_R_TEMP(float *DAT); //读温度#endif

这里是C文件

​​​​​​​文章来源地址https://www.toymoban.com/news/detail-833638.html

C++
#include "SENSOR.h"#include "myiic.h"
#define T117_WADD 0x80#define T117_RADD 0x81
//测温指令寄存器(Temp_Cmd),地址 0x04//#define Temp_Cmd 0x04
//配置寄存器(Temp_Cfg),地址 0x05//#define Temp_Cfg 0x05
//状态寄存器(Status),地址 0x03//#define Status 0x03/*位 内容描述 默认数值7 温度高线报警跟踪 0:温度报警未触发 1:温度报警触发 ‘0’6 温度低线报警跟踪 0:温度报警未触发 1:温度报警触发 ‘0’5 温度转换状态 0:温度转换完成 1:温度转换过程中 ‘0’4 E2PROM 状态 0:未处于读写状态 1:处于读写状态 ‘0’3 加热状态 0:未处于加热状态 1:处于加热状态 ‘0’2 温度报警错误提示 0:TH 大于 TL 1:TH 小于等于 TL ‘0’1:0 预留 ‘00’*/
//报警模式寄存器(Alert_Mode),地址 0x06//#define Alert_Mode 0x06
/*位 内容描述 默认数值7 报警功能开关(Alert_en)0:关闭1:开启 ‘0’6 报警模式(IM)选择0:高于 TH 报警+低于 TL 解除报警1:高于 TH 报警+低于 TL 报警 ‘0’5 报警极性(POL)选择0:低有效1:高有效 ‘0’4 报警端口输出模式(FUNC)选择0:输出温度报警标志位1:输出测温完成标志位 ‘0’3:0 预留 ‘0000’
*/

void T117_Init(void){
T117_W_REG(Temp_Cmd,0x00);//连续测温不加热
T117_W_REG(Temp_Cfg,0x99);//每秒0.5次,32次平均,进入低功耗
T117_W_REG(Alert_Mode,0x10);//不报警,输出测温完成标志位}

uint8_t T117_ID(void) //获取ID{uint8_t temp=0;IIC2_Start();IIC2_Send_Byte(T117_WADD);temp=IIC2_Wait_Ack();if(temp) {IIC2_Stop();return 1;}IIC2_Send_Byte(0x19);temp=IIC2_Wait_Ack();if(temp) {IIC2_Stop();return 2;}
IIC2_ReStart();IIC2_Send_Byte(T117_RADD);temp=IIC2_Wait_Ack();if(temp) {IIC2_Stop();return 3;}temp=IIC2_Read_Byte();IIC2_NAck();IIC2_Stop();return temp;}
uint8_t T117_R_REG(uint8_t REG,uint8_t *DAT) //读寄存器{uint8_t temp=0;IIC2_Start();IIC2_Send_Byte(T117_WADD);temp=IIC2_Wait_Ack();if(temp) {IIC2_Stop();return 1;}IIC2_Send_Byte(REG);temp=IIC2_Wait_Ack();if(temp) {IIC2_Stop();return 2;}
IIC2_ReStart();IIC2_Send_Byte(T117_RADD);temp=IIC2_Wait_Ack();if(temp) {IIC2_Stop();return 3;}*DAT=IIC2_Read_Byte();IIC2_NAck();IIC2_Stop();return 0;}
uint8_t T117_W_REG(uint8_t REG,uint8_t DAT) //写寄存器{uint8_t temp=0;IIC2_Start();IIC2_Send_Byte(T117_WADD);temp=IIC2_Wait_Ack();if(temp) {IIC2_Stop();return 1;}
IIC2_Send_Byte(REG);temp=IIC2_Wait_Ack();if(temp) {IIC2_Stop();return 2;}
IIC2_Send_Byte(DAT);temp=IIC2_Wait_Ack();if(temp) {IIC2_Stop();return 3;}IIC2_Stop();return 0;
}
uint8_t T117_R_TEMP(float *DAT) //读寄存器{uint16_t rx=0;uint8_t data=0;
if(T117_R_REG(0x01,&data)) return 1;//rx = data<<8;rx = data;rx <<=8;if(T117_R_REG(0x00,&data)) return 2;rx += data;*DAT=(int16_t)rx;*DAT=25.0+(*DAT)/256.0;return 0;
}

04

蓝牙驱动

关于KT6368A的驱动实现,可以去梁山派的模块移植手册查看详情,具体实现我这里就不再赘述。

手册入口【建议收藏】:https://lceda001.feishu.cn/wiki/JNvYwEU5SiGldFkNcxncYXhZnZc

用STM32手搓一个体温计,c++

模块移植手册-第一部

05

OLED驱动

这个0.91寸显示屏,网上有很多移植好的,这里也不再赘述,可以去梁山派的模块移植手册查看详情。

用STM32手搓一个体温计,c++

06

其它串口IIC驱动

这个是老生常谈。

串口使用的是USART2,支持printf

IIC采用IO模拟方式,这样不受IO位置控制了。

具体啰嗦的代码就不这里贴了,节省空间哦,可以在开源原文里下载源文件。

07

main函数实现

这里是整个实现方式,对应前面的设计思路,这里包含一些中断回调函数的实现。

​​​​​​​

C++/* USER CODE BEGIN Header *//********************************************************************************* @file : main.c* @brief : Main program body******************************************************************************* @attention** Copyright (c) 2024 STMicroelectronics.* All rights reserved.** This software is licensed under terms that can be found in the LICENSE file* in the root directory of this software component.* If no LICENSE file comes with this software, it is provided AS-IS.********************************************************************************//* USER CODE END Header *//* Includes ------------------------------------------------------------------*/#include "main.h"#include "adc.h"#include "dma.h"#include "tim.h"#include "usart.h"#include "gpio.h"
/* Private includes ----------------------------------------------------------*//* USER CODE BEGIN Includes */#include "oled.h"#include "bsp_KT6368A.h"#include "myiic.h"#include "SENSOR.h"#include "usart.h"/* USER CODE END Includes */
/* Private typedef -----------------------------------------------------------*//* USER CODE BEGIN PTD */
/* USER CODE END PTD */
/* Private define ------------------------------------------------------------*//* USER CODE BEGIN PD */#define VOT 3.287/* USER CODE END PD */
/* Private macro -------------------------------------------------------------*//* USER CODE BEGIN PM */
/* USER CODE END PM */
/* Private variables ---------------------------------------------------------*/
/* USER CODE BEGIN PV */uint16_t adc_value[4]={0};uint8_t Flag=0;float temp=0,x,y,z;uint32_t second=0;uint8_t hh,mm,ss;
/* USER CODE END PV */
/* Private function prototypes -----------------------------------------------*/void SystemClock_Config(void);/* USER CODE BEGIN PFP */
/* USER CODE END PFP */
/* Private user code ---------------------------------------------------------*//* USER CODE BEGIN 0 */
/* USER CODE END 0 */
/*** @brief The application entry point.* @retval int*/int main(void){/* USER CODE BEGIN 1 */uint8_t DAT;uint16_t len;uint8_t i=0;/* USER CODE END 1 */
/* MCU Configuration--------------------------------------------------------*/
/* Reset of all peripherals, Initializes the Flash interface and the Systick. */HAL_Init();
/* USER CODE BEGIN Init */
/* USER CODE END Init */
/* Configure the system clock */SystemClock_Config();
/* USER CODE BEGIN SysInit */
/* USER CODE END SysInit */
/* Initialize all configured peripherals */MX_GPIO_Init();MX_DMA_Init();MX_ADC_Init();MX_TIM1_Init();MX_USART2_UART_Init();/* USER CODE BEGIN 2 */
T117_Init();OLED_Init();
HAL_Delay(2000);
//Set_SppName();HAL_Delay(1000);//Set_BLEName();HAL_Delay(1000);//Set_Power();HAL_ADC_Start_DMA(&hadc,(uint32_t *)(&(adc_value[0])),4);HAL_TIM_PWM_Start(&htim1,TIM_CHANNEL_4);




/* USER CODE END 2 */
/* Infinite loop *//* USER CODE BEGIN WHILE */while (1){/* USER CODE END WHILE */
/* USER CODE BEGIN 3 */char s[20];if(Flag){Flag=0;x=adc_value[0]/4095.0*VOT*2;if(x>=4.2){sprintf(s,"99%%");}else if(x>=4.06){sprintf(s,"90%%");}else if(x>=3.98){sprintf(s,"80%%");}else if(x>=3.92){sprintf(s,"70%%");}else if(x>=3.87){sprintf(s,"60%%");}else if(x>=3.82){sprintf(s,"50%%");}else if(x>=3.79){sprintf(s,"40%%");}else if(x>=3.77){sprintf(s,"30%%");}else if(x>=3.74){sprintf(s,"20%%");}else if(x>=3.68){sprintf(s,"10%%");}else if(x>=3.45){sprintf(s," 5%%");}else if(x>=3.33){sprintf(s," 1%%");}else{sprintf(s," 0%%");}
OLED_ShowString(0,0,s,8,1);//6*8
hh=second/3600;mm=(second%3600)/60;ss=second%60;
sprintf(s,"%02d:%02d:%02d",hh,mm,ss);OLED_ShowString(0,24,s,8,1);//6*8
// temp=adc_value[2]/4095.0*VOT;// y=25+(1.43-temp)/0.0043;// sprintf(s,"%6.3f",y);// OLED_ShowString(0,18,s,12,1);//6*8}if(T117_R_REG(0x03,&DAT)==0){if((DAT&0x20)==0){// DAT=HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_4);// printf("A4:%d\r\n",DAT);// if(!(T117_R_REG(0x00,&DAT))){printf("LSB:%x\r\n",DAT);}// if(!(T117_R_REG(0x01,&DAT))){printf("MSB:%x\r\n",DAT);}DAT=T117_R_TEMP(&z);sprintf(s,"%5.2fC",z);OLED_ShowString(0,7,s,16,1);
}}
if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_1)){if(USART_RX_STA&0x8000){len=USART_RX_STA&0x3fff;//得到此次接收到的数据长度HAL_UART_Transmit(&huart2,(uint8_t*)USART_RX_BUF,len,1000); //发送接收到的数据while(__HAL_UART_GET_FLAG(&huart2,UART_FLAG_TC)!=SET); //等待发送结束printf("\r\n\r\n");//插入换行USART_RX_STA=0;printf("\r\nLEN:%d\r\n",len);}if(DAT==0){printf("T=%5.2fC\r\n",z);}sprintf(s,"BT:OK");OLED_ShowString(18,0,s,8,1);//6*8printf("V=%6.3fV\r\n",x);printf("%02d:%02d:%02d\r\n",hh,mm,ss);

}else{sprintf(s,"BT:NO");OLED_ShowString(18,0,s,8,1);//6*8}
OLED_ShowChinese(48,0,0,16,1);OLED_ShowChinese(64,0,1,16,1);OLED_ShowChinese(80,0,2,16,1);OLED_ShowChinese(96,0,3,16,1);OLED_ShowChinese(112,0,4,16,1);//OLED_ShowChinese(48,16,5,16,1);OLED_ShowChinese(64,16,6,16,1);OLED_ShowChinese(80,16,7,16,1);OLED_ShowChinese(96,16,8,16,1);OLED_ShowChinese(112,16,9,16,1);
OLED_Refresh();HAL_Delay(2000);}/* USER CODE END 3 */}
/*** @brief System Clock Configuration* @retval None*/void SystemClock_Config(void){RCC_OscInitTypeDef RCC_OscInitStruct = {0};RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
/** Initializes the RCC Oscillators according to the specified parameters* in the RCC_OscInitTypeDef structure.*/RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI;RCC_OscInitStruct.HSIState = RCC_HSI_ON;RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT;RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK){Error_Handler();}
/** Initializes the CPU, AHB and APB buses clocks*/RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK|RCC_CLOCKTYPE_PCLK1;RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_HSI;RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1;
if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_0) != HAL_OK){Error_Handler();}}
/* USER CODE BEGIN 4 */
void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc){//HAL_ADC_Stop(&hadc1);//HAL_ADC_Stop_DMA(&hadc1);//HAL_GPIO_TogglePin(GPIOC,GPIO_PIN_13);Flag=1;second += 2;HAL_ADC_Start_DMA(hadc,(uint32_t *)(&(adc_value[0])),4);
}
/* USER CODE END 4 */
/*** @brief This function is executed in case of error occurrence.* @retval None*/void Error_Handler(void){/* USER CODE BEGIN Error_Handler_Debug *//* User can add his own implementation to report the HAL error return state */__disable_irq();while (1){}/* USER CODE END Error_Handler_Debug */}
#ifdef USE_FULL_ASSERT/*** @brief Reports the name of the source file and the source line number* where the assert_param error has occurred.* @param file: pointer to the source file name* @param line: assert_param error line source number* @retval None*/void assert_failed(uint8_t *file, uint32_t line){/* USER CODE BEGIN 6 *//* User can add his own implementation to report the file name and line number,ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) *//* USER CODE END 6 */}#endif /* USE_FULL_ASSERT */

到了这里,关于用STM32手搓一个体温计的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 嵌入式项目分享 单片机远程wifi红外无接触体温测量系统 - 物联网 stm32

    🔥 这两年开始毕业设计和毕业答辩的要求和难度不断提升,传统的毕设题目缺少创新和亮点,往往达不到毕业答辩的要求,这两年不断有学弟学妹告诉学长自己做的项目系统达不到老师的要求。 为了大家能够顺利以及最少的精力通过毕设,学长分享优质毕业设计项目,今天

    2024年02月20日
    浏览(62)
  • 通信工程毕设 基于Stm32的便携体测仪(心率 体温) - 单片机 嵌入式 物联网

    🔥 这两年开始毕业设计和毕业答辩的要求和难度不断提升,传统的毕设题目缺少创新和亮点,往往达不到毕业答辩的要求,这两年不断有学弟学妹告诉学长自己做的项目系统达不到老师的要求。 为了大家能够顺利以及最少的精力通过毕设,学长分享优质毕业设计项目,今天

    2024年02月20日
    浏览(58)
  • 【单片机毕设选题】单片机远程wifi红外无接触体温测量系统 - 物联网 stm32

    🔥 这两年开始毕业设计和毕业答辩的要求和难度不断提升,传统的毕设题目缺少创新和亮点,往往达不到毕业答辩的要求,这两年不断有学弟学妹告诉学长自己做的项目系统达不到老师的要求。 为了大家能够顺利以及最少的精力通过毕设,学长分享优质毕业设计项目,今天

    2024年02月20日
    浏览(52)
  • 【机器人/小车】自己动手用ESP32手搓一个智能机器人:ESP32-CAM AI Robot(文末附完整工程源码)

    目录 介绍 硬件需求  软件需求 步骤 总结 源码下载 ESP32-CAM是一款集成了Wi-Fi和蓝牙功能的微控制器模块,同时还集成了摄像头接口,使其成为一个非常适合构建智能机器人的选择。在本项目中,我将向您展示如何使用ESP32-CAM模块构建一个

    2024年04月15日
    浏览(64)
  • 97、基于stm32单片机智能药箱药盒温湿度体温光照时钟wifi手机APP监控(程序+原理图+PCB源文件+手机APP源码+硬件设计资料+元器件清单等)

    单片机类型选择 方案一:可以使用现在比较主流的单片机STC89C5单片机进行数据处理。这款单片机具有的特点是内存和51的单片机相比多了4KB内存,但是价格和51单片机一样。并且支持数据串行下载和调试助手。此款单片机是有ATMEL公司生产,可用5V电压编程,而且擦写时间仅需

    2024年02月13日
    浏览(51)
  • 记录--ThreeJs手搓一个罗盘特效

    最近在学Three.js.,对着文档看了一周多,正好赶上码上掘金的活动,就顺便写了一个小demo,手搓一个罗盘特效。 先来看一下太极的实现方式,这里我们使用CircleGeometry,将其分解开来可以看出是由圆形和半圆形组成 。 CircleGeometry 官网案例 radius 半径 segments 分段(三角面)的数

    2024年02月03日
    浏览(45)
  • 【STM32】STM32CubeMX创建第一个工程

    STM32CubeMX是STM32Cube工具家族中的一员,从MCU/MPU选型,引脚配置,系统时钟以及外设时钟设置,到外设参数配置,中间件参数配置,它给STM32开发者们提供了一种简单,方便,并且直观的方式来完成这些工作。所有的配置完成后,它还可以根据所选的IDE生成对应的工程和初始化

    2024年04月09日
    浏览(101)
  • unity开发笔记#230821-手搓一个虚拟摇杆

    做unity游戏时,如果是做手机端则需要加一个虚拟摇杆以便玩家控制角色移动(做pc端则直接获取键值即可较方便)。原理就是用Image制作一个大圆圈住一个Image小圆,玩家拖拽小圆控制角色移动。 中心思想是,以小圆中心为(0,0),获取拖拽的偏移量即可。 首先将图片资源拉入

    2024年02月12日
    浏览(45)
  • 【STM32】制作一个bootloader

    工作环境:STM32CubeMX+Keil 相关环境准备这里就不介绍了。 bootloader是什么 bootloader就是单片机启动时候运行的一段小程序,这段程序负责单片机固件的更新,也就是单片机选择性的自己给自己下载程序。可以更新,可以不更新,更新的话,boot loader更新完程序后,跳转到新程序

    2024年02月08日
    浏览(58)
  • 【ArcGIS Pro二次开发】(45):手搓一个工具执行进度框

    我在之前做的工具中,UI这部分基本没怎么深入,都是直接用现成的控件。 其中有一个问题比较突出,就是没有工具执行的进度框提示。曾经也用过系统自带的信息提示框和进度条,但太简陋,确实不好用。于是就想抄一个进度框来用,自然而然想到了ArcGIS中地理工具执行时

    2024年02月13日
    浏览(52)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包