標(biāo)題: 求一個(gè)單片機(jī)串口發(fā)送ADC值的代碼 [打印本頁(yè)]

作者: samxon    時(shí)間: 2024-12-2 22:13
標(biāo)題: 求一個(gè)單片機(jī)串口發(fā)送ADC值的代碼
51單片機(jī)STC15W408

如題,謝謝分享。

作者: 3176912825    時(shí)間: 2024-12-2 22:30
這段代碼首先初始化ADC和串口,然后在主循環(huán)中不斷讀取ADC值,并通過(guò)串口發(fā)送。請(qǐng)注意,根據(jù)您的具體硬件配置和需求,您可能需要調(diào)整ADC通道和串口設(shè)置。
作者: 3176912825    時(shí)間: 2024-12-2 22:39
3176912825 發(fā)表于 2024-12-2 22:30
這段代碼首先初始化ADC和串口,然后在主循環(huán)中不斷讀取ADC值,并通過(guò)串口發(fā)送。請(qǐng)注意,根據(jù)您的具體硬件配 ...

#include <STC15W.h>
#include <intrins.h>

// 定義ADC通道,這里以通道1為例
#define ADC_CHANNEL 1

// 延時(shí)函數(shù),用于ADC穩(wěn)定
void Delay20ms() {
    unsigned char i, j, k;
    i = 1;
    j = 234;
    k = 113;
    do {
        do {
            while (--k);
        } while (--j);
    } while (--i);
}

// 初始化ADC
void InitADC() {
    P1ASF |= (1 << ADC_CHANNEL); // 設(shè)置P1.1為ADC輸入
    ADC_RES = 0; // 清除ADC結(jié)果寄存器
    ADC_CONTR = ADC_POWER | ADC_SPEEDLL; // 使能ADC,設(shè)置速度為最快
    Delay20ms(); // 等待ADC穩(wěn)定
}

// 獲取ADC結(jié)果
unsigned int GetADCResult() {
    unsigned int ADC_10BIT_RES;
    ADC_CONTR = ADC_POWER | ADC_SPEEDLL | ADC_CHANNEL | ADC_START; // 啟動(dòng)ADC轉(zhuǎn)換
    _nop_(); // 等待4個(gè)機(jī)器周期
    _nop_();
    _nop_();
    _nop_();
    while (!(ADC_CONTR & ADC_FLAG)); // 等待轉(zhuǎn)換完成
    ADC_CONTR &= ~ADC_FLAG; // 清除完成標(biāo)志
    ADC_10BIT_RES = ADC_RES; // 獲取高8位結(jié)果
    ADC_10BIT_RES <<= 2;
    ADC_10BIT_RES += (0x03 & ADC_RESL); // 獲取低2位結(jié)果
    return ADC_10BIT_RES; // 返回10位ADC結(jié)果
}

// 初始化串口
void InitSerialPort() {
    SCON = 0x50; // 設(shè)置串口為模式1
    AUXR |= 0x01; // 選擇定時(shí)器2作為波特率發(fā)生器
    AUXR |= 0x04; // 定時(shí)器2工作在8位自動(dòng)重裝模式
    T2L = 0xE0; // 設(shè)置定時(shí)器2的重裝值,波特率9600
    T2H = 0xFE;
    AUXR |= 0x10; // 啟動(dòng)定時(shí)器2
    REN = 1; // 允許接收
    SM0 = 0; // 設(shè)置串口為模式1
    SM1 = 1;
}

// 發(fā)送一個(gè)字節(jié)
void SendByte(unsigned char byte) {
    SBUF = byte; // 將數(shù)據(jù)放入到發(fā)送緩沖寄存器
    while (!TI); // 等待發(fā)送完成
    TI = 0; // 清除發(fā)送完成標(biāo)志
}

// 發(fā)送字符串
void SendString(unsigned char *str) {
    while (*str) {
        SendByte(*str++); // 發(fā)送當(dāng)前字符,并指向下一個(gè)字符
    }
}

// 主函數(shù)
void main() {
    unsigned int adcValue;
    InitADC(); // 初始化ADC
    InitSerialPort(); // 初始化串口

    while (1) {
        adcValue = GetADCResult(); // 獲取ADC值
        SendString("ADC Value: "); // 發(fā)送字符串
        SendByte(adcValue / 1000 + '0'); // 發(fā)送ADC值的千位
        SendByte((adcValue % 1000) / 100 + '0'); // 發(fā)送ADC值的百位
        SendByte((adcValue % 100) / 10 + '0'); // 發(fā)送ADC值的十位
        SendByte(adcValue % 10 + '0'); // 發(fā)送ADC值的個(gè)位
        SendByte('\r\n'); // 發(fā)送換行符
        Delay20ms(); // 延時(shí),控制發(fā)送頻率
    }
}

作者: samxon    時(shí)間: 2024-12-2 23:37
3176912825 發(fā)表于 2024-12-2 22:39
#include
#include

謝謝你的精彩代碼,我想發(fā)送的是ADC_10BIT_RES這個(gè)結(jié)果
作者: jjy1039    時(shí)間: 2024-12-3 08:34
samxon 發(fā)表于 2024-12-2 23:37
謝謝你的精彩代碼,我想發(fā)送的是ADC_10BIT_RES這個(gè)結(jié)果

你是說(shuō)直接發(fā)送ADC的值嗎
那以下兩個(gè)函數(shù)直接改成
// 獲取ADC結(jié)果
unsigned char GetADCResult() {
    unsigned int ADC_10BIT_RES;
    ADC_CONTR = ADC_POWER | ADC_SPEEDLL | ADC_CHANNEL | ADC_START; // 啟動(dòng)ADC轉(zhuǎn)換
    _nop_(); // 等待4個(gè)機(jī)器周期
    _nop_();
    _nop_();
    _nop_();
    while (!(ADC_CONTR & ADC_FLAG)); // 等待轉(zhuǎn)換完成
    ADC_CONTR &= ~ADC_FLAG; // 清除完成標(biāo)志
    ADC_10BIT_RES = ADC_RES; // 獲取高8位結(jié)果
//    ADC_10BIT_RES <<= 2;
//    ADC_10BIT_RES += (0x03 & ADC_RESL); // 獲取低2位結(jié)果
    return ADC_10BIT_RES; // 返回10位ADC結(jié)果
}

// 主函數(shù)
void main() {
    unsigned int adcValue;
    InitADC(); // 初始化ADC
    InitSerialPort(); // 初始化串口

    while (1) {
         SendByte(GetADCResult()); // 獲取ADC值并發(fā)送高8位
        SendByte(ADC_RESL);//低2位
        Delay20ms(); // 延時(shí),控制發(fā)送頻率
    }
}


作者: coody_sz    時(shí)間: 2024-12-3 12:29
STC官方ADC例子基本都是用串口發(fā)送的。
作者: samxon    時(shí)間: 2024-12-4 17:12
發(fā)表于 2024-12-3 08:34
你是說(shuō)直接發(fā)送ADC的值嗎
那以下兩個(gè)函數(shù)直接改成
// 獲取ADC結(jié)果

問(wèn)題解決,可以正常發(fā)送顯示。
新的問(wèn)題,STC15W如何調(diào)整ADC的采樣速度,
我用單通道采集電壓穩(wěn)定,如果在兩個(gè)通道之間切換采集,采集的電壓很不穩(wěn)定,懷疑是采樣速度太快的問(wèn)題。有合適方法解決嗎,我用的是ADC中斷式采集。

謝謝大家。

123.JPG (128.89 KB, 下載次數(shù): 0)

123.JPG

作者: QQ_OA2CEB    時(shí)間: 2024-12-4 21:20
#include "stm32f1xx_hal.h"  UART_HandleTypeDef huart1; ADC_HandleTypeDef hadc1;  void SystemClock_Config(void) {     // 配置系統(tǒng)時(shí)鐘 }  void MX_GPIO_Init(void) {     // 初始化GPIO }  void MX_ADC1_Init(void) {     // 初始化ADC,設(shè)置分辨率為12位,采樣時(shí)間為239.5個(gè)周期     hadc1.Instance = ADC1;     hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;     hadc1.Init.ContinuousConvMode = DISABLE;     hadc1.Init.DiscontinuousConvMode = DISABLE;     hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;     hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;     hadc1.Init.NbrOfConversion = 1;     hadc1.Init.DMAContinuousRequests = DISABLE;     hadc1.Init.EOCSelection = ADC_EOC_SINGLE_CONV;     HAL_ADC_Init(&hadc1); }  void MX_USART1_UART_Init(void) {     // 初始化USART1,設(shè)置波特率為9600bps     huart1.Instance = USART1;     huart1.Init.BaudRate = 9600;     huart1.Init.WordLength = UART_WORDLENGTH_8B;     huart1.Init.StopBits = UART_STOPBITS_1;     huart1.Init.Parity = UART_PARITY_NONE;     huart1.Init.Mode = UART_MODE_TX_RX;     huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;     huart1.Init.OverSampling = UART_OVERSAMPLING_NONE;     HAL_UART_Init(&huart1); }  int main(void) {     HAL_Init();     SystemClock_Config();     MX_GPIO_Init();     MX_ADC1_Init();     MX_USART1_UART_Init();     while (1) {         HAL_ADC_Start(&hadc1); // 啟動(dòng)ADC轉(zhuǎn)換         if (HAL_ADC_PollForConversion(&hadc1, HAL_MAX_DELAY) == HAL_OK) { // 等待轉(zhuǎn)換完成             uint32_t adcValue = HAL_ADC_GetValue(&hadc1); // 獲取ADC值             char adcStr; // 用于存儲(chǔ)轉(zhuǎn)換結(jié)果的字符串表示形式,包括終止符'\0'             sprintf(adcStr, "%d", adcValue); // 將ADC值轉(zhuǎn)換為字符串形式             HAL_UART_Transmit(&huart1, (uint8_t*)adcStr, strlen(adcStr), HAL_MAX_DELAY); // 通過(guò)串口發(fā)送ADC值         }     } }




歡迎光臨 (http://www.torrancerestoration.com/bbs/) Powered by Discuz! X3.1