![]() |
發布時間: 2018-5-15 17:57
正文摘要:stc12c5a60s2單片機,ad模式 p1口采集,當輸入的模擬電壓小于3v時,被p1口果斷拉倒3.2v以上,無法采集,怎么解決,在線等,急!!!!!!!!!! |
2.#define WAIT_SALE 0 3.#define NO_WATER 1 4.#define INPUT_MONEY 2 5.#define SALING 3 6.#define CHANGE 4 7.#define CANCLE 5 8.#define NO_DATA 0 怎么會 #define WAIT_SALE 0 #define NO_DATA 0 是這樣, |
本帖最后由 wulin 于 2018-5-18 06:17 編輯 哈哈007 發表于 2018-5-17 18:41 這是一個實際應用中的簡單的平均法濾波程序,2ms采樣一次,兩個通道交替運作。 /*-------------------------- ADC采樣濾波子程序 ---------------------------*/ void LB_ADC() { static uint PJZ_ADC1=0; //累加10次采樣的變量1 static uint PJZ_ADC2=0; //累加10次采樣的變量2 static uchar count1=0; //計數器變量 static uchar count2=0; //計數器變量 static bit CYTD=0; //采樣通道變量 if(CYTD==0) //ADC通道1采樣 { PJZ_ADC1 += GetADCResult(5);//累加采樣值 count1++; //計數器變量自+1 if(count1>=10) //如果計數到10次采樣 { count1=0; //計數器清0 WDCY1=((PJZ_ADC1+5)/10);//獲取平均值,含四舍五入 PJZ_ADC1=0; //采樣變量清0 CYTD=1; //改為第2采樣通道 } } if(CYTD==1) //ADC通道2采樣 { PJZ_ADC2 += GetADCResult(6);//累加采樣值 count2++; //計數器變量自+1 if(count2>=10) //如果計數到10次采樣 { count2=0; //計數器清0 WDCY2=((PJZ_ADC2+5)/10);//獲取平均值,含四舍五入 PJZ_ADC2=0; //采樣變量清0 CYTD=0; //改為第1采樣通道 } } } |
模式設置不對,看下寄存器配置 |
wulin 發表于 2018-5-17 15:52 那要如何進行軟件濾波呢,能說下嘛 ![]() ![]() |
本帖最后由 wulin 于 2018-5-17 16:24 編輯 哈哈007 發表于 2018-5-16 17:15 我改動的主要部分就在上面回帖里,其他改動與你程序功能無關。單片機使用的是IAP15W4K58S4,主要是LCD驅動程序和延時程序。因為我的開發板線路與你有差異,IAP15W4K58S4與stc12c5a60s2都是1T單片機,ADC部分相同。出現數字亂是因為端口設為高阻后不能懸空和程序中沒有軟件濾波。可以隨意接分壓電阻試試。 |
wulin 發表于 2018-5-16 10:41 我改了好像出了問題 你能把所有的程序復制一下給我嗎 拜托了 ![]() |
哈哈007 發表于 2018-5-15 23:20 主程序這樣寫的顯示效果
![]() |
哈哈007 發表于 2018-5-15 21:59 相應端口設為高阻模式和ADC功能,模擬信號可以直接送到對應的端口。最好每條通道多次取樣,軟件濾波,消除顯示的數據跳動。你的主循環代碼是錯誤的。下面是添加和改動的部分代碼,供參考。
|
現在我就不知道到底設為高阻模式,還是設為開漏模式,還有是不是AD的信號可以直接送到對應的IO口,還是需要接什么電路? |
#include "reg52.h" #include "intrins.h" typedef unsigned char uchar; typedef unsigned int uint; #define _Nop() _nop_() /*------------------------以下為LCD1602顯示模塊定義-----------------------*/ unsigned char data_char_table[]= {"0123456789ABCDEF"}; //LCD數據 unsigned char Lcd_Dis1_table[] = {"Position:No. "}; //第一行顯示框架 unsigned char pos_char_table[] = {" D "}; // 顯示位置 unsigned char Lcd_Dis2_table[] = {"Voltage : V"}; //第二行顯示框架 unsigned char num_char_table[] = {" 9A.CD V"}; // 顯示位置 sbit lcd_rs_port = P2^4; //定義LCD控制端口,根據硬件調整 sbit lcd_rw_port = P2^3; sbit lcd_en_port = P2^2; #define lcd_data_port P0 void lcd_delay(uchar ms); //LCD1602 延時 void lcd_busy_wait(); //LCD1602 忙等待 void lcd_command_write(uint command); //LCD1602 命令字寫入 void lcd_system_reset(); //LCD1602 初始化 void lcd_char_write(uint x_pos,y_pos,lcd_dat); //LCD1602 字符寫入 void lcd_bad_check(); //LCD1602 壞點檢查 void Num_to_Disp(uchar i, uint Num); //顯示數據處理 void LcdDisp(uchar j, uint num); //液晶顯示函數 /*------------------------以下為ADC相應寄存器初始化及端口定義-------------*/ /***** 定義與ADC相關的特殊功能寄存器 *****/ sfr ADC_CONTR = 0xBC; //ADC控制寄存器 sfr ADC_RES = 0xBD; //ADC hight 8-bit result register sfr ADC_RESL = 0xBE; //ADC low 2-bit result register sfr P1ASF = 0x9D; //P1口功能控制寄存器P1ASF /************定義相應操作位***************/ #define ADC_POWER 0x80 //ADC電源控制位,0:關閉,1:打開 #define ADC_FLAG 0x10 //ADC結束標志位 #define ADC_START 0x08 //ADC啟動控制位 #define ADC_SPEEDLL 0x00 //540 clocks___________選擇轉換速度 /*------------------------以下為相關函數聲明------------------------------*/ void InitADC(); //ADC初始化 uint GetADCResult(uchar ch); void Delay(uint n); //延時程序 void delay_1ms(uchar x); /*-------------------------------- 主函數 --------------------------------*/ void main() { uchar i; lcd_system_reset(); //LCD1602 初始化 lcd_bad_check(); //LCD1602 壞點檢查 InitADC(); //初始化ADC特殊功能寄存器 while (1) { i = 2; while(i < 7) { LcdDisp(i, GetADCResult(i)); //液晶1602顯示輸入電壓值(P1.0 - P1.7) Delay(1000); //i++; } } } /*-------------------------------- ADC 取值 ------------------------------*/ uint GetADCResult(uchar ch) { ADC_CONTR = ADC_POWER | ADC_SPEEDLL | ch | ADC_START; _nop_(); //Must wait before inquiry _nop_(); _nop_(); _nop_(); while (!(ADC_CONTR & ADC_FLAG)); //Wait complete flag ADC_CONTR &= ADC_FLAG; //Close ADC return (ADC_RES*4 + ADC_RESL); //Return ADC result } /*---------------------------- 初始化ADC特殊功能寄存器 -------------------*/ void InitADC( ) { P1ASF = P1 | 0x3f; //Set P1.0 - P1.5 as analog input port ADC_RES = 0; //Clear previous result ADC_RESL = 0; ADC_CONTR = ADC_POWER | ADC_SPEEDLL ; Delay(20); //ADC power-on delay and Start A/D conversion } /*---------------------------- LCD1602相應函數 ---------------------------*/ ///////////////以下為LCD顯示數據處理///////////////// void Num_to_Disp(uchar i, uint Num) { float NUM; int xx, yy, zz; NUM =1.1* (Num * 5/ 1024.0); //計算公式:10-bit A/D Conversion Result = 1024 x (Vin / Vcc) xx = (int)NUM; yy = (int)((NUM - (float)(xx)) * 10); zz = (int)((NUM - (float)(xx)) * 100)%10; num_char_table[9] = data_char_table[xx / 10]; //電壓值十位 num_char_table[10]= data_char_table[xx % 10]; //電壓值個位 num_char_table[12]= data_char_table[yy]; //電壓值小數點后一位 num_char_table[13]= data_char_table[zz]; //電壓值小數點后兩位? pos_char_table[13]= data_char_table[i]; //當前ADC接口 } //////////////////以下為LCD顯示//////////////////// void LcdDisp(uchar j, uint num) { uint i=0; for (i=0;i<16;i++) { lcd_char_write(i,0,Lcd_Dis1_table[i]); lcd_char_write(i,1,Lcd_Dis2_table[i]); //顯示框架 } Num_to_Disp(j, num); lcd_char_write(13,0,pos_char_table[13]); for(i = 9; i < 14; i++) { lcd_char_write(i,1,num_char_table[i]); //顯示電壓 } delay_1ms(100); } //////////////以下是LCD1602驅動程序//////////////// void lcd_delay(uchar ms) /*LCD1602 延時*/ { uchar j; while(ms--){ for(j=0;j<250;j++) {;} } } void lcd_busy_wait() /*LCD1602 忙等待*/ { lcd_rs_port = 0; lcd_rw_port = 1; lcd_en_port = 1; lcd_data_port = 0xff; while (lcd_data_port&0x80); lcd_en_port = 0; } void lcd_command_write(uint command) /*LCD1602 命令字寫入*/ { lcd_busy_wait(); lcd_rs_port = 0; lcd_rw_port = 0; lcd_en_port = 0; lcd_data_port = command; lcd_en_port = 1; lcd_en_port = 0; } void lcd_system_reset() /*LCD1602 初始化*/ { lcd_delay(20); lcd_command_write(0x38); lcd_delay(100); lcd_command_write(0x38); lcd_delay(50); lcd_command_write(0x38); lcd_delay(10); lcd_command_write(0x08); lcd_command_write(0x01); lcd_command_write(0x06); lcd_command_write(0x0c); } void lcd_char_write(uint x_pos,y_pos,lcd_dat) /*LCD1602 字符寫入*/ { x_pos &= 0x0f; /* X位置范圍 0~15 */ y_pos &= 0x01; /* Y位置范圍 0~ 1 */ if(y_pos==1) x_pos += 0x40; x_pos += 0x80; lcd_command_write(x_pos); lcd_busy_wait(); lcd_rs_port = 1; lcd_rw_port = 0; lcd_en_port = 0; lcd_data_port = lcd_dat; lcd_en_port = 1; lcd_en_port = 0; } void lcd_bad_check() /*LCD1602 壞點檢查*/ { char i,j; for(i=0;i<2;i++){ for(j=0;j<16;j++) { lcd_char_write(j,i,0xff); } } lcd_delay(200); lcd_delay(200); lcd_delay(200); lcd_delay(100); lcd_delay(200); lcd_command_write(0x01); /* clear lcd disp */ } //////////////////以上是LCD1602驅動程序//////////////// /*----------------------------- 延時程序 ---------------------------*/ void Delay(uint n) { uint x; while (n--) { x = 500; while (x--); } } /*1MS為單位的延時程序*/ void delay_1ms(uchar x) { uchar j; while(x--) { for(j=0;j<125;j++) {;} } } 這是我用的程序,不知道為什么,就是總會被拉高上去 |
你先把P1.0設置成模擬口線,而不是IO口線 |
你應該設置P1口為開漏模式 sfr P1ASF = 0x9D; //Pl口模擬功能控制寄存器 P1ASF = 0xff; //設置P1口為開漏模式,使用AD功能 |
哈哈007 發表于 2018-5-15 18:05 設置P1M0、 P1M1寄存器用作ADC的口為高阻模式,P1ASF寄存器設置相應端口作為模擬輸入,否則不能正常測試模擬信號。 |
我需要采樣的電壓時直接通過電阻分壓送到p1.0口的,假設我要讀取的電壓為0.454V,可是P1.0口的電壓會把它拉上去根本就讀不了 |