极海APM32F003系列MCU在满足高性能与稳定性的同时,也具有小体积、小封装,低成本等优势,助力客户以更经济的成本获取更加复杂、创新的产品功能。由于其主要针对一些对成本比较敏感的应用,外设资源以常用为主,所以APM32F003系列MCU并没有集成内置的温度传感器的功能。在众多应用领域中,有时候也会遇到一些应用场景需要MCU的资源并不多,封装尽可能小,但是需要检测MCU工作环境的温度。这样的应用如果选择一颗集成温度传感器的MCU,势必会增加应用成本,并且造成资源浪费。此时,如果一颗低成本、高性能的MCU,再加一个热敏电阻测温,能同时满足低成本和测温的需求,这不妨是最佳的选择。
下面就给大家介绍一下使用APM32F003F6P6和一个热敏电阻实现测温的应用。
NTC介绍
NTC是负温度系数的简写,全称是Negative temperature coefficient,是一种随温度上升时,电阻值呈指数关系减小的热敏电阻,常用作温度传感器。这里有一个式子表示负温度系数的电阻值:RT=R0*exp(B (1/T-1/T0))
RT为周围温度为T (K) 时的电阻值,R0是周围温度为T0 (K) 时的电阻值,注意这里的温度是开尔文温度。B为常数,它也是材料常数,一般在25摄氏度下测得。
温度测量电路
如下图所示,温度测量常用电阻串联分压,ADC采样的方式进行。本例中APM32F6P6的供电电压VCC为5V,为了提高测温精度,串联电阻采用了15KΩ和4.3KΩ。
温度转换
通过ADC得到采样值后,如果转换得到温度呢?这里有两种计算方法,一种是公式法,一种是查表法,下面将一一介绍。
首先第一步是找到自己的热敏电阻的数据手册,查找B值和25℃的电阻值,本例中使用的热敏电阻型号是SDNT2012X473F4150FTF,25℃的阻值47KΩ,B值为4150,其温度-阻值特性曲线如下图所示。
然后我们在程序中定义好这些参数:
- float ntc_series_resistance = 15000.0; //NTC的串联电阻
- float ntc_b_value = 4150.0; //B值
- float ntc_r25 = 47000.0; //25℃时的电阻
- float kelvins_zero = 273.15; //绝对零度
- uint16_t system_power_voltage = 5000; //系统参考电压是5V
- float T25 = 298.15; //25 = kelvins_zero + 25
这里的计算逻辑就是先通过ADC采样得到NTC和4.3KΩ的电压推算出串联电阻15KΩ的分压,然后再计算出线路电流,再计通过欧姆定律从计算出热敏电阻和4.3KΩ之和的阻值,减去4.3KΩ后得到热敏电阻的阻值,最后通过公式计算出温度。务必注意,这些计算,需要引入math.h。
- uint16_t adc_value;
- uint32_t ntc_voltage;
- float ntc_current;
- float ntc_resistance;
- float temperature;
-
- adc_value = Get_Adc(); //获取ADC采样值
- ntc_voltage = adc_value * 5000 / 4095; //采样值转换为电压(mV)
- ntc_current = ((system_power_voltage- ntc_voltage) / ntc_series_resistance); //计算NTC的电流(mA)
- ntc_resistance = (ntc_voltage / ntc_current) - 4300; //计算NTC电阻值(Ω)
- temperature = (ntc_b_value * T25) / (T25 * (log(ntc_resistance) - log(ntc_r25)) + ntc_b_value);
- temperature -= kelvins_zero; //计算最终温度
-
- return temperature;
一般而言,厂家会提供NTC电阻的温度与电阻表,如下图所示,基于这个表,我们只需要计算出当前热敏电阻的阻值Rt,然后查照此表,得到最接近的温度值即可。
程序中我们使用二分法查表,程序实现如下:
- /*!
- * [url=home.php?mod=space&uid=247401]@brief[/url] Find_Table 二分查找算法->查温度表
- *
- * @param 表地址、表长度、要查找的数据
- *
- * @retval 数据在表中的位置
- *
- * [url=home.php?mod=space&uid=536309]@NOTE[/url] 查找数据在表中对应的位置 表中数据从大到小
- */
- uint8_t Find_Table(float *table, uint8_t table_len, float data)
- {
- uint8_t st, ed, m;
- uint8_t i = 0;
- st = 0;
- ed = table_len - 1;
- if (data >= table[st]) return st;
- else if (data <= table[ed]) return ed;
- while (st < ed) {
- m = (st + ed) / 2;
- if (data == table[m] ) break;
- if (data < table[m] && data > table[m+1]) {
- if ((table[m] - data) > (data - table[m+1])) {
- m = m + 1;
- }
- break;
- }
- if (data > table[m]) {
- ed = m;
- } else {
- st = m;
- }
- if (i++ > table_len) break;
- }
- if (st > ed ) return 0;
- return m;
- }
首先我们需要通过ADC采样值计算后得到热敏电阻值,然后通过二分法查表函数找到当前热敏电阻值在表中的位置,从而得到对应的温度值,程序实现如下:
- int8_t Get_Temperature_Value(void)
- {
- uint16_t any_value;
- uint16_t ntc_voltage;
- float ntc_res;
- int8_t temp;
-
- any_value = Get_Adc(); //获取ADC采样值
- ntc_voltage = any_value * 5000 / 4095; //采样值转换为电压(mV)
- ntc_res = ((ntc_voltage * 15000) / (5000 - ntc_voltage)) - 4300; //计算NTC电阻值(Ω)
- ntc_res = ntc_res / 1000;//计算NTC电阻值(KΩ)
-
- any_value = Find_Table(table_res, 152, ntc_res); //二分法查表
- //通过热敏电阻在表中的位置得到对应的温度值
- if (any_value < 151) {
- temp = any_value - 30;
- } else {
- temp = 125;
- }
-
- return temp;
- }
以上就是通过超低成本MCU APM32F003F6P6和热敏电阻实现温度测量的例程分享,欢迎大家评论区讨论交流。
|