首页 > 编程语言 >基于STM32的气体检测系统--有上位机小程序

基于STM32的气体检测系统--有上位机小程序

时间:2024-07-10 14:25:51浏览次数:29  
标签:机小 ESP8266 -- 气体 ppm STM32 include esp8266

       

 毕设搞完了,来分享一下,第一次搞,多包涵!

设计要求:

        技术要求: 1、原始数据:采集的气体浓度数据,包括CO2、可燃气体、有毒气体等;系统运行时的各种状态数据。 2、选择合适的气体传感器,并进行性能测试和评估;设计稳定且精确的模拟信号采集电路;嵌入式程序设计,实现数据采集、处理和界面显示;实现报警功能,确保及时预警和用户安全。 3、达到的指标:准确测量和实时监测不同气体浓度;检测范围和灵敏度符合相关要求; 数据显示界面友好且直观;报警功能准确可靠,能够及时提示用户;数据传输和远程监测功能稳定可靠。 4、 实验仿真等:原始数据采集和处理实验,验证嵌入式程序的功能和算法;报警功能实验,测试报警的准确性和响应速度;电源管理实验,评估电池寿命和充电效率;通信功能实验,测试数据传输的稳定性和可靠性。

        废话不多说直接软硬件设计!

硬件设计:

        本文所设计的便携式气体检测系统采用STM32F103C8T6作为核心处理单元,通过模块化系统,实现了对周围环境的全面监控和管理。
  通过温甲醛、一氧化碳、烟雾、可燃性等气体传感器对气体质量进行多方面监测,检测到的气体数据传输到主控芯片,再通过主控芯片的ADC数模转化模块转换和处理气体浓度信号。此外,用户还可以通过按键控制显示不同测量数据的气体浓度变化情况曲线,从而更直观地了解气体浓度的变化趋势。

          当气体浓度值超过设定的报警阈值时,STM32芯片通过启动蜂鸣器和闪烁红灯的方式,提醒用户当前环境有害气体超标,并在LCD屏幕上显示实时报警信息。

          为了实现仪器的智能化和数据的远程传输,本文使用WIFI通信模块将检测数据上传到云端。上位机小程序是另一部分,通过与云端的通信,从云端获取数据并存入本地数据库,并展示这些数据,借助云端和上位机小程序,用户可以随时随地监测和分析室内空气质量,从而做出相应的措施。此外,系统采用稳压模块5V供电,确保了电源的稳定性和系统的稳定运行。

硬件选型:

        stm32f103c8t6主控,DHT11温湿度传感器,SGP30甲醛浓度检测模块,MQ-2烟雾浓度检测模块,MQ-9可燃气浓度检测模块,MQ-7一氧化碳气浓度检测模块,TFT液晶显示模块,ESP8266-01S WIFI模块,声光报警模块:蜂鸣器和LED,按键

PCB板绘制:

        本文基于STM32的便携式气体检测系统硬件PCB板设计使用嘉立创eda设计,PCB板由嘉立创免费提供。

软件设计:

        本文基于STM32的便携式气体检测系统软件设计使用C语言编程设计,程序代码采用keil软件编写,程序注有清晰的中文注释。

主程序设计

        初始化完成后,系统进入主循环,核心是 timeCount 时间管理变量。timeCount 小于 40 时,系统进行实时监控和响应,包括检测 LED 和蜂鸣器状态、显示气体数据、判断报警等,根据 edit 变量确定后续操作。随着 timeCount 增加,达到 10 的倍数读取气体传感器数据并转换,40 的倍数向 上位机发送数据,超过 200 向服务器发送数据并归零。然后判断服务器是否下发数据,最后 timeCount 加 1 回到主循环起点继续执行对应操作,以实现高效准确监测和通信。 

main函数代码

/*************************************版本2*********************************/

/*   USART1   		TX--PA10		RX--PA9

	 ESP8266-01s  	TX--PA3			RX--PA2     RST--PC14
	 
	 1.8TFT+KEY
//              GND   电源地
//              VCC   接5V或3.3v电源
//              SCL   接PA0(SCL)	PB8
//              SDA   接PA1(SDA)	PB9
//              RES   接PB12
//              DC    接PB5
//              CS    接PA7 
//				BL	  接PB13

按键
//				K1    接PB0
//				K2    接PB1
//				K3    接PB3
//				K4    接PB4

蜂鸣器			IO    接PB15

DHT11           DATA  接PB14

SGP30           SDA--PB11    SCL---PB10

MQ7一氧化碳     AO--PA4    DO--

MQ2烟雾			AO--PA5    DO--

MQ9可燃气		AO--PA6    DO--

OLED 			SCL--PB6  SDA--PB7

/***************************************************************************/
//单片机头文件
#include "stm32f10x.h"
#include "delay.h"
#include "sys.h"
#include "bmp.h"
#include "DHT11.h"
#include "usart.h"
#include "sgp30.h"
#include "adc.h"
#include "buzzer.h"
#include "key.h"
#include "led.h"
#include "lcd_init.h"
#include "lcd.h"
#include "pic.h"
#include "exti.h"
#include "function.h"
//#include "lanya.h"
#include "timer.h"
#include "oled.h"
//网络协议层
#include "onenet.h"

//网络设备
#include "esp8266.h"

//C库
#include <string.h>
#include <stdio.h>
#include "math.h"

		u8 Num;
	    int  key_count;
		u8 voltage;

		u8 temp;
		u8 humi;
		u16 len;
		u32 CO2Data,TVOCData;				//定义CO2浓度变量与TVOC浓度变量
	   
		float CO_data,PM25_data,CH4_data;
		int HCHO_ppm,CO2_ppm,CO_ppm,PM25_ppm,CH4_ppm;
		int edit;
			
		unsigned long sgp30_dat;
	    u16 adc_CO,adc_CH4,adc_PM25;
		
		
		u8 LED_Status = 0;
		u8 alarm_flag = 0;				//是否报警标志
	
		char oled_buf[20];		//上传数据的buf
		
		char PUB_BUF[256];		//上传数据的buf
		const char *devSubtopic[] = {"/myGas_detection/sub"};
		const char devPubtopic[] = "/myGas_detection/pub";
		
		unsigned short timeCount = 0;	//发送间隔变量
		
		int count = 0;	//发送间隔变量
	
		unsigned char *dataPtr = NULL;

		
 int main(void)
 {

		NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置 NVIC 中断分组 2:2 位抢占优先级,2 位响应优先级

		LED_Init();
	 
		
		Adc_Init();		  		//ADC初始化	 
		delay_init();	    	 //延时函数初始化
		
		KEY_Init();//按键初始化
		LCD_Init();//LCD初始化
		LCD_Fill(0,0,LCD_W,LCD_H,WHITE);
	    SGP30_Init();   //初始化SGP30
		BEEP_Init();
	 
	 
		/* 初始化函数,如果DHT11存在响应则返回1,否则0 */
		DHT11_Init(); 
		
		
		Usart1_Init(115200);							//串口1,打印信息用
	
		Usart2_Init(115200);							//串口2,驱动ESP8266用
		
	 
		UsartPrintf(USART_DEBUG, " Hardware init OK\r\n");
		
	
		ESP8266_Init();					//初始化ESP8266
		
		OneNet_DevLink();
		//while(OneNet_DevLink())			//接入OneNET
		delay_ms(50);
	
		OneNet_Subscribe(devSubtopic, 1);	
		
		EXTIX_Init();
		
	
	while(1) 
	{		

			SHOW_zhujiem();
			baojing ();

			if (edit==1)
			{
			SHOW_curve(key_count);
			}
			else SHOW_flag(key_count);

	if(timeCount % 5 ==0)
	{
			DHT11_Read_Data(&temp, &humi);
						
			SGP30_Write(0x20,0x08);
			sgp30_dat = SGP30_Read();//读取SGP30的值
			CO2Data = (sgp30_dat & 0xffff0000) >> 16;//取出CO2浓度值
			CO2_ppm = KalmanFilter_CO2(CO2Data);
			TVOCData = sgp30_dat & 0x0000ffff;       //取出TVOC值
			HCHO_ppm = KalmanFilter_HCHO(TVOCData);
	
			//CO_ppm = MQ7_GetPPM();
			CO_ppm = KalmanFilter_CO( MQ7_GetPPM() );
	

			//PM25_ppm = MQ2_GetPPM();
			PM25_ppm = KalmanFilter_PM25(  MQ2_GetPPM() );
			

			
			CH4_ppm =KalmanFilter_CH4( MQ9_GetPPM() );

	}		

		if(timeCount >= 8)//5000ms/25=200 发送间隔5000ms		1min	8---10s						//发送间隔5s
		{
		
			UsartPrintf(USART_DEBUG, "OneNet_Publish\r\n");
			UsartPrintf(USART_DEBUG, "温度:%d\r\n 湿度:%d\r\n 甲醛:%d\r\n CO2:%d\r\n",temp,humi,TVOCData,CO2Data);
			UsartPrintf(USART_DEBUG, "CO:%d\r\n 烟雾:%d\r\n 可燃气:%d\r\n",CO_ppm,PM25_ppm,CH4_ppm);
			
			sprintf(PUB_BUF,"{\"Temp\":%d,\"Humi\":%d,\"HCHO\":%d,\"CO2\":%d,\"CO\":%d,\"PM2_5\":%d,\"CH4\":%d,\"LED\":%d,\"BEEP\":%d}",
			temp,humi,TVOCData,CO2Data,CO_ppm,PM25_ppm,CH4_ppm,LED_Status,alarm_flag);
			
			OneNet_Publish(devPubtopic, PUB_BUF);
			
			timeCount = 0;
			ESP8266_Clear();
		}
		
		dataPtr = ESP8266_GetIPD(3);
		if(dataPtr != NULL)
		OneNet_RevPro(dataPtr);
		
		delay_ms(10);
		
		timeCount++;
		
}
}
 

报警子程序设计

        系统进入报警子程序时先检查上位机控制,若有则报警器为手动控制模式,更新 alarm_is_free 变量并进行十秒声光自检以验证报警功能。若上位机未操作,系统检查气体数据,达阈值则启动声光报警器发出信号,让操作人员采取措施避免安全事故。

void baojing (void)
{
	if(alarm_is_free  == 10)				//报警器自动控制是否空闲
	{
	if (temp < 50 && humi < 90 && TVOCData < 100 &&  CO2Data < 1000 && CO_ppm < 100 && PM25_ppm < 500 && CH4_ppm <1000)
	{
		LED = 1;
		BEEP = 0;
	}else{
		
		LED = 0;
		BEEP = 1;
	}
}
	if (alarm_is_free < 10)alarm_is_free ++;
	delay_ms(1000);
	
//	UsartPrintf(USART_DEBUG, "alarm_is_free %d\r\n",alarm_is_free);
	
}

MQ系列数据的读取(以MQ2烟雾传感器为例)

        MQ-2烟雾传感器输出为0-5V的模拟电压信号,烟雾浓度越高电压越高,而STM32F103C8T6单片机的ADC输入通道的模拟量输入范围为0-3.3V,因此需要串联电阻进行分压,保护AD转换芯片。这样,将传感器的模拟输出引脚接到处理器的一个ADC引脚进行模数转换,即可得烟雾传感器的输出电压。又因烟雾浓度与电压信号关系为非线性,因此需拟合出气体浓度与输出电压的关系曲线,从而计算气体浓度。

        其中,x为RS/R0的比值,R0元件在洁净空气中的阻值,通常取6.64,RS=(Vc-VI)*RI/V,VC为回路电压,该系统取5V,为传感器模拟输出电中压,RL为5。

WiFi数据传输设计

        在进行物联网项目开发时,通常需要将微控制器与Wi-Fi模块相配合,以实现数据的采集、处理和远程传输。数据传输流程图如图4. 8所示,其配置步骤如下:

  1. 初始化过程:

串口初始化:设置串口波特率、数据位、停止位和校验位等。

ESP8266-01s初始化: Wi-Fi连接设置和MQTT客户端的初始化。

  1. 网络接入检测:

判断ESP8266-01s是否成功接入到OneNet平台,若未接入,则尝试连接,直至成功连接为止。

  1. MQTT主题订阅:

网络连接成功后,订阅指定的OneNet主题,接收来自MQTT服务器的消息。

  1. 数据打包与传输:

        STM32负责采集和处理传感器数据,将这些数据打包成MQTT消息格式,然后ESP8266-01s通过Wi-Fi网络将这些消息传输至MQTT服务器。

//esp8266.c
//单片机头文件
#include "stm32f10x.h"

//网络设备驱动
#include "esp8266.h"

//硬件驱动
#include "delay.h"
#include "usart.h"

//C库
#include <string.h>
#include <stdio.h>


#define ESP8266_WIFI_INFO		"AT+CWJAP=\"ONENET\",\"88888888\"\r\n"

#define ESP8266_ONENET_INFO		"AT+CIPSTART=\"TCP\",\"broker.emqx.io\",1883\r\n"

//#define ESP8266_ONENET_INFO		"AT+CIPSTART=\"TCP\",\"47.97.44.102\",1883\r\n"

//#define ESP8266_ONENET_INFO		"AT+CIPSTART=\"TCP\",\"emqx_Mjc0OT\",1883\r\n"


unsigned char esp8266_buf[128];
unsigned short esp8266_cnt = 0, esp8266_cntPre = 0;


//==========================================================
//	函数名称:	ESP8266_Clear
//
//	函数功能:	清空缓存
//
//	入口参数:	无
//
//	返回参数:	无
//
//	说明:		
//==========================================================
void ESP8266_Clear(void)
{

	memset(esp8266_buf, 0, sizeof(esp8266_buf));
	esp8266_cnt = 0;

}
//==========================================================
//	函数名称:	ESP8266_WaitRecive
//
//	函数功能:	等待接收完成
//
//	入口参数:	无
//
//	返回参数:	REV_OK-接收完成		REV_WAIT-接收超时未完成
//
//	说明:		循环调用检测是否接收完成
//==========================================================
_Bool ESP8266_WaitRecive(void)
{

	if(esp8266_cnt == 0) 							//如果接收计数为0 则说明没有处于接收数据中,所以直接跳出,结束函数
		return REV_WAIT;
		
	if(esp8266_cnt == esp8266_cntPre)				//如果上一次的值和这次相同,则说明接收完毕
	{
		esp8266_cnt = 0;							//清0接收计数
			
		return REV_OK;								//返回接收完成标志
	}
		
	esp8266_cntPre = esp8266_cnt;					//置为相同
	
	return REV_WAIT;								//返回接收未完成标志

}

//==========================================================
//	函数名称:	ESP8266_SendCmd
//
//	函数功能:	发送命令
//
//	入口参数:	cmd:命令
//				res:需要检查的返回指令
//
//	返回参数:	0-成功	1-失败
//
//	说明:		
//==========================================================
_Bool ESP8266_SendCmd(char *cmd, char *res)
{
	
	unsigned char timeOut = 200;

	Usart_SendString(USART2, (unsigned char *)cmd, strlen((const char *)cmd));
	
	while(timeOut--)
	{
		if(ESP8266_WaitRecive() == REV_OK)							//如果收到数据
		{
			if(strstr((const char *)esp8266_buf, res) != NULL)		//如果检索到关键词
			{
				ESP8266_Clear();									//清空缓存
				
				return 0;
			}
		}
		
		delay_ms(10);
	}
	
	return 1;

}

//==========================================================
//	函数名称:	ESP8266_SendData
//
//	函数功能:	发送数据
//
//	入口参数:	data:数据
//				len:长度
//
//	返回参数:	无
//
//	说明:		
//==========================================================
void ESP8266_SendData(unsigned char *data, unsigned short len)
{

	char cmdBuf[32];
	
	ESP8266_Clear();								//清空接收缓存
	sprintf(cmdBuf, "AT+CIPSEND=%d\r\n", len);		//发送命令
	if(!ESP8266_SendCmd(cmdBuf, ">"))				//收到‘>’时可以发送数据
	{
		Usart_SendString(USART2, data, len);		//发送设备连接请求数据
	}

}

//==========================================================
//	函数名称:	ESP8266_GetIPD
//
//	函数功能:	获取平台返回的数据
//
//	入口参数:	等待的时间(乘以10ms)
//
//	返回参数:	平台返回的原始数据
//
//	说明:		不同网络设备返回的格式不同,需要去调试
//				如ESP8266的返回格式为	"+IPD,x:yyy"	x代表数据长度,yyy是数据内容
//==========================================================
unsigned char *ESP8266_GetIPD(unsigned short timeOut)
{

	char *ptrIPD = NULL;
	
	do
	{
		if(ESP8266_WaitRecive() == REV_OK)								//如果接收完成
		{
			ptrIPD = strstr((char *)esp8266_buf, "IPD,");				//搜索“IPD”头
			if(ptrIPD == NULL)											//如果没找到,可能是IPD头的延迟,还是需要等待一会,但不会超过设定的时间
			{
				//UsartPrintf(USART_DEBUG, "\"IPD\" not found\r\n");
			}
			else
			{
				ptrIPD = strchr(ptrIPD, ':');							//找到':'
				if(ptrIPD != NULL)
				{
					ptrIPD++;
					return (unsigned char *)(ptrIPD);
				}
				else
					return NULL;
				
			}
		}
		
		delay_ms(5);													//延时等待
		timeOut--;
	} while(timeOut>0);
	
	return NULL;														//超时还未找到,返回空指针

}

//==========================================================
//	函数名称:	ESP8266_Init
//
//	函数功能:	初始化ESP8266
//
//	入口参数:	无
//
//	返回参数:	无
//
//	说明:		
//==========================================================
void ESP8266_Init(void)
{
	
	GPIO_InitTypeDef GPIO_Initure;
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);

	//ESP8266复位引脚
	GPIO_Initure.GPIO_Mode = GPIO_Mode_Out_PP;
	GPIO_Initure.GPIO_Pin = GPIO_Pin_14;					//GPIOC14-复位
	GPIO_Initure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOC, &GPIO_Initure);
	
	GPIO_WriteBit(GPIOC, GPIO_Pin_14, Bit_RESET);
	delay_ms(250);
	GPIO_WriteBit(GPIOC, GPIO_Pin_14, Bit_SET);
	delay_ms(500);
	
	ESP8266_Clear();
	
	UsartPrintf(USART_DEBUG, "0. AT\r\n");
	while(ESP8266_SendCmd("AT\r\n", "OK"))
	delay_ms(500);
	
	
	UsartPrintf(USART_DEBUG, "1. RST\r\n");
	ESP8266_SendCmd("AT+RST\r\n", "");
	delay_ms(500);
	ESP8266_SendCmd("AT+CIPCLOSE\r\n", "");
	delay_ms(500);
	
	UsartPrintf(USART_DEBUG, "2. CWMODE\r\n");
	while(ESP8266_SendCmd("AT+CWMODE=1\r\n", "OK"))
		delay_ms(500);
	
	UsartPrintf(USART_DEBUG, "3. AT+CWDHCP\r\n");
	while(ESP8266_SendCmd("AT+CWDHCP=1,1\r\n", "OK"))
		delay_ms(500);
	
	UsartPrintf(USART_DEBUG, "4. CWJAP\r\n");
	while(ESP8266_SendCmd(ESP8266_WIFI_INFO, "OK"))//GOT IP
		delay_ms(500);
	
	UsartPrintf(USART_DEBUG, "5. CIPSTART\r\n");
	while(ESP8266_SendCmd(ESP8266_ONENET_INFO, "CONNECT"))
		delay_ms(500);
	
	UsartPrintf(USART_DEBUG, "6. ESP8266 Init OK\r\n");

}

//==========================================================
//	函数名称:	USART2_IRQHandler
//
//	函数功能:	串口2收发中断
//
//	入口参数:	无
//
//	返回参数:	无
//
//	说明:		
//==========================================================
void USART2_IRQHandler(void)
{

	if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET) //接收中断
	{
		if(esp8266_cnt >= sizeof(esp8266_buf))	esp8266_cnt = 0; //防止串口被刷爆
		esp8266_buf[esp8266_cnt++] = USART2->DR;
		
		USART_ClearFlag(USART2, USART_FLAG_RXNE);
	}

}

        其与部分较为简单,暂不列出,有问题欢迎评论交流!

上位机小程序设计

        便携式气体检测微信小程序是一款智能软件,可实时监测和显示气体浓度,并为用户的操作提供界面,该软件应用于各种环境监测、工业生产和公共安全,可以帮助用户及时发现和处理潜在的气体安全风险。

功能与特点

  1. 实时监测:通过Wi-Fi与便携式气体检测系统通信,获取实时气体浓度数据,并通过图表的方式向用户展示。
  2. 数据展示:上位机显示气体浓度数据,支持折线图显示,使用户了解气体数据变化趋势。
  3. 用户界面:提供简洁明了的主界面、图表界面、数据界面等,方便用户操作和查找数据。
  4. 数据存储:将数据存入本地数据库,防止数据丢失,保证数据安全。
  5. 报警功能:当检测到气体浓度超过设定阈值时,上位机会发出警报消息,通知用户采取相应处理措施。
  6. 设备管理:支持对连接的便携式气体检测仪进行管理,包括设备报警装置自检等操作。

界面设计

        1.主界面:显示实时气体浓度数据、设备连接状态、气体类型、报警状态等信息,方便用户快速了解设备运行情况和气体浓度变化。

        2. 图表界面:有数据表和气体浓度曲线图供用户选择,其中气体浓度曲线图展示了气体变化情况,方便用户直观地查看气体浓度数据的变化趋势。

         3.设置界面:提供设备参数设置、报警阈值设置等操作界面,方便用户根据需要进行个性化配置。

开发环境与工具

1. 开发语言:使用Java编程语言进行开发,具有良好的性能和可维护性。

2. 开发工具:使用微信开发者工具进行开发,提高开发效率和代码质量。

3. 数据库工具:使用MySQL数据库工具进行数据存储和管理,保证数据的安全性和可靠性。

 最终实现效果 !         

<iframe allowfullscreen="true" data-mediaembed="bilibili" frameborder="0" id="lL4PrRN2-1720535304180" src="https://player.bilibili.com/player.html?aid=112756984384294"></iframe>

基于STM32的气体检测系统设计--含上位机小程序

标签:机小,ESP8266,--,气体,ppm,STM32,include,esp8266
From: https://blog.csdn.net/gu_fan_/article/details/140306287

相关文章

  • 从零学习大模型——使用GLM-4-9B-Chat + BGE-M3 + langchain + chroma建立的本地RAG应
    第一篇介绍了如何配置最基本的环境并下载了GLM-4-9B-Chat到本地,接下来我们试着将GLM-4-9B-Chat接入LangChain。LangChain 是一个基于大型语言模型(LLM)开发应用程序的框架。LangChain简化了LLM应用程序生命周期的每个阶段:开发:使用LangChain的开源构建模块和组件构建应用程序......
  • 初创芯片公司非常疯狂,将CPU性能提高100倍
    初创芯片公司非常疯狂,将CPU性能提高100倍[http://mp.weixin.qq.com/s?__biz=Mzg2NDgzNTQ4MA**&mid=2247741576&idx=5&sn=733a2dffecbfd99e41e97e93e204b2cb&chksm=ce6e327ff919bb691bf4e3ed418f27d816846b1c577477d5d7f063c103e01d9d4cbeca47195b&mpshare=1&scen......
  • 做外卖霸王餐前景咋样,好不好做
    微客云霸王餐系统当今互联网经济蓬勃发展,外卖行业已成为日常生活不可或缺的一部分,其市场规模和用户基础逐年攀升。随着消费者对便捷性和优惠力度需求的增加,外卖霸王餐作为一种新兴的营销模式,正逐渐成为商家和第三方平台眼中的香饽饽。本文将深入探讨外卖霸王餐的前景,分析其市场潜......
  • Windows定时器-timeSetEvent
     接口:MMRESULTtimeSetEvent(UINTuDelay,//以毫秒指定事件的周期UINTuResolution,//以毫秒指定延时的精度,缺省值为1msLPTIMECALLBACKlpTimeProc,//指向回调函数的指针WORDdwUser,//用户定义的回调数据,传递给回调函数......
  • 模块
    1.第三方模块Python内部提供的模块有限,所以在平时在开发的过程中,经常会使用第三方模块。而第三方模块必须要先安装才能可以使用,使用第三方模块的行为就是去用别人写好并开源出来的py代码,这样自己拿来就用,不必重复造轮子了。下面介绍常见的3种安装第三方模块的方式。1.1pip【最......
  • POI导出案例
    /** *POI方式导出 *@paramlist数据 *@paramexportFields绑定列数组,0:表头,1:数据key *@paramresponse */ publicstaticvoidexportExcel(Listlist,String[]exportFields,HttpServletResponseresponse){ Workbookwb=newXSSFWorkbook(); /......
  • 从零学习大模型——使用GLM-4-9B + BGE-M3 + langchain + chroma建立的本地RAG应用(一)
    本项目基于DataWhaleChina的self-llm教程与llm-universe及Langchain官方文档等资料开发,旨在实现全部本地化的RAG应用。本项目使用AutoDL的云服务器进行开发。在 AutoDL 平台中租一个3090等24G显存的显卡机器,如下图所示镜像选择 PyTorch-->2.1.0-->3.10(ubuntu22.04)-->12.......
  • stm32时钟源
    stm32时钟源目录stm32时钟源MCU提供5种时钟源(1) HSE高速外部时钟(2) HSI高速内部时钟(3) LSE低速外部时钟(4) LSI低速内部时钟(5) PLL倍频锁相环定时器是挂载在总线下,而不同的总线的频率是不同的,而总线的频率是由时钟提供,而时钟的提供者又各不相同,所以必须要提前了解时钟源的......
  • Easyadmin创建新的目录菜单步骤,Curd创建方法介绍
    应用easyadmin创建文件目录,可以使用复制原有的目录,也可以使用CURD方法创建,一般为了防止复制后,修改出错或漏改,建议使用CURD方法,此方法不仅快捷方便,也因存在设计数据库,便于后面开发。1.创建CURD命令行,创建数据表。  2.打开Composer,注意版本,此处使用php7.4,  3.执行Compose......
  • api文字识别智能录入、身份证识别、接口识别​
    OCR技术和由此带来的文字识别自动化程度不断增加,不少人预计该技术将对相当一部分的行业、工作产生影响,其中有一部分是颠覆性的。比如文字录入的工作,现在不少企业为自己的系统、产品集成了OCR技术核心,不仅能够减少人工录入的压力,正确率及录入速度等都得到了大幅的提升。可......