久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索

stc12c5a60s2單片機,ad模式 p1口采集問題

查看數: 6342 | 評論數: 18 | 收藏 0
關燈 | 提示:支持鍵盤翻頁<-左 右->
    組圖打開中,請稍候......
發布時間: 2018-5-15 17:57

正文摘要:

stc12c5a60s2單片機,ad模式 p1口采集,當輸入的模擬電壓小于3v時,被p1口果斷拉倒3.2v以上,無法采集,怎么解決,在線等,急!!!!!!!!!!

回復

ID:693991 發表于 2020-2-18 09:26
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
是這樣,
ID:332500 發表于 2018-5-17 21:00
軟件濾波--比如采5次, 去掉最大值和最小值, 其余3個取平均.

評分

參與人數 1黑幣 +20 收起 理由
admin + 20 回帖助人的獎勵!

查看全部評分

ID:213173 發表于 2018-5-17 20:54
本帖最后由 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采樣通道
                }
        }
}
ID:330999 發表于 2018-5-17 19:53
模式設置不對,看下寄存器配置
ID:309126 發表于 2018-5-17 18:41
wulin 發表于 2018-5-17 15:52
我改動的主要部分就在上面回帖里,其他改動與你程序功能無關。單片機使用的是IAP15W4K58S4,主要是LCD驅 ...

那要如何進行軟件濾波呢,能說下嘛
ID:213173 發表于 2018-5-17 15:52
本帖最后由 wulin 于 2018-5-17 16:24 編輯
哈哈007 發表于 2018-5-16 17:15
就是我修改完之后,AD沒有接輸入進來,它的值會一直跳動,跳動范圍在0.99~1.3,怎么修改才能讓他的值等于零 ...

我改動的主要部分就在上面回帖里,其他改動與你程序功能無關。單片機使用的是IAP15W4K58S4,主要是LCD驅動程序和延時程序。因為我的開發板線路與你有差異,IAP15W4K58S4與stc12c5a60s2都是1T單片機,ADC部分相同。出現數字亂是因為端口設為高阻后不能懸空和程序中沒有軟件濾波。可以隨意接分壓電阻試試。
ID:309126 發表于 2018-5-16 17:02
wulin 發表于 2018-5-16 10:41
主程序這樣寫的顯示效果
void main()
{

我改了好像出了問題 你能把所有的程序復制一下給我嗎 拜托了
ID:213173 發表于 2018-5-16 10:41
哈哈007 發表于 2018-5-15 23:20
我把P1ASF = P1 | 0x3f;     都改為0Xff了都還是不行

主程序這樣寫的顯示效果
  1. void main()
  2. {
  3.         uchar i;
  4.         InitADC();                 //初始化ADC特殊功能寄存器        
  5.         lcd_system_reset();        //LCD1602 初始化
  6.         for (i=0;i<16;i++)         //顯示框架
  7.         {
  8.                 lcd_char_write(i,0,Lcd_Dis1_table[i]);//從void LcdDisp(uchar j, uint num)里移到這里
  9.                 lcd_char_write(i,1,Lcd_Dis2_table[i]); //從void LcdDisp(uchar j, uint num)里移到這里
  10.         }
  11.         while (1)
  12.         {         
  13.                 for(i=0;i<6;i++)
  14.                 {
  15.                         LcdDisp(i, GetADCResult(i));
  16.                         Delay(1000);
  17.                 }
  18.         }  
  19. }
復制代碼




ID:213173 發表于 2018-5-16 07:01
哈哈007 發表于 2018-5-15 21:59
現在我就不知道到底設為高阻模式,還是設為開漏模式,還有是不是AD的信號可以直接送到對應的IO口,還是需要 ...

相應端口設為高阻模式和ADC功能,模擬信號可以直接送到對應的端口。最好每條通道多次取樣,軟件濾波,消除顯示的數據跳動。你的主循環代碼是錯誤的。下面是添加和改動的部分代碼,供參考。
  1. /***** 定義與ADC相關的特殊功能寄存器 *****/                       
  2. sfr  ADC_CONTR =  0xBC;                 //ADC控制寄存器                    
  3. sfr  ADC_RES   =  0xBD;                 //ADC hight 8-bit result register
  4. sfr  ADC_RESL  =  0xBE;                 //ADC low 2-bit result register
  5. sfr  P1ASF     =  0x9D;                 //P1口功能控制寄存器P1ASF
  6. sfr  P1M0      =  0x92;                  //I/O模式寄存器
  7. sfr  P1M1      =  0x91;                  //I/O模式寄存器
  8. /*-------------------------------- 主函數 --------------------------------*/                                             
  9. void main()
  10. {
  11.         uchar i;
  12.         P1M1 = 0x3f;                                         //P1.0~P1.5高阻 0011 1111
  13.         P1M0 = 0x00;                                         //P1.0~P1.5高阻 0000 0000
  14.         InitADC();                 //初始化ADC特殊功能寄存器        
  15.         lcd_system_reset();        //LCD1602 初始化
  16. //        lcd_bad_check();           //LCD1602 壞點檢查

  17.         for (i=0;i<16;i++)         //顯示框架
  18.         {
  19.                 lcd_char_write(i,0,Lcd_Dis1_table[i]);//顯示第一行框架
  20.                 lcd_char_write(i,1,Lcd_Dis2_table[i]);//顯示第二行框架
  21.         }
  22.         while (1)
  23.         {         
  24.                 for(i=0;i<6;i++)
  25.                 {
  26.                         //獲取 i 通道轉換數據
  27.                         //數據處理
  28.                         //1602顯示輸入電壓值更新
  29.                         //Delay(1000);
  30.                 }
  31.         }  
  32. }
  33. /*---------------------------- 初始化ADC特殊功能寄存器 -------------------*/
  34. void InitADC( )
  35. {
  36. //        P1ASF = P1 | 0x3f; //Set  P1.0 - P1.5 as analog input port
  37.         P1ASF = 0x3f;        //0011 1111
  38.         ADC_RES  = 0;   //Clear previous result
  39.         ADC_RESL = 0;
  40.         ADC_CONTR = ADC_POWER | ADC_SPEEDLL ;
  41.         Delay(20);                      //ADC power-on delay and Start A/D conversion
  42. }
復制代碼
ID:309126 發表于 2018-5-15 23:20
393163967 發表于 2018-5-15 22:08
你先把P1.0設置成模擬口線,而不是IO口線

我把P1ASF = P1 | 0x3f;     都改為0Xff了都還是不行

評分

參與人數 1黑幣 +20 收起 理由
admin + 20 回帖助人的獎勵!

查看全部評分

ID:330846 發表于 2018-5-15 22:08
你先把P1.0設置成模擬口線,而不是IO口線

評分

參與人數 1黑幣 +20 收起 理由
admin + 20 回帖助人的獎勵!

查看全部評分

ID:309126 發表于 2018-5-15 21:59
現在我就不知道到底設為高阻模式,還是設為開漏模式,還有是不是AD的信號可以直接送到對應的IO口,還是需要接什么電路?
ID:309126 發表于 2018-5-15 21:56
#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++)
            {;}
    }   
}  





這是我用的程序,不知道為什么,就是總會被拉高上去
ID:287575 發表于 2018-5-15 21:22
你先把P1.0設置成模擬口線,而不是IO口線
ID:155507 發表于 2018-5-15 21:11
你應該設置P1口為開漏模式

sfr P1ASF  =   0x9D;           //Pl口模擬功能控制寄存器

P1ASF = 0xff;                   //設置P1口為開漏模式,使用AD功能
ID:213173 發表于 2018-5-15 20:55
哈哈007 發表于 2018-5-15 18:05
我需要采樣的電壓時直接通過電阻分壓送到p1.0口的,假設我要讀取的電壓為0.454V,可是P1.0口的電壓會把它拉 ...

設置P1M0、 P1M1寄存器用作ADC的口為高阻模式,P1ASF寄存器設置相應端口作為模擬輸入,否則不能正常測試模擬信號。
ID:309126 發表于 2018-5-15 18:05
我需要采樣的電壓時直接通過電阻分壓送到p1.0口的,假設我要讀取的電壓為0.454V,可是P1.0口的電壓會把它拉上去根本就讀不了

小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 成人黄色a | 国产一级网站 | 日本一区二区三区免费观看 | 国产精品永久免费 | 一区二区三区四区免费视频 | 午夜精品一区二区三区在线 | 久久久久国产一区二区 | 日韩精品影院 | 国产亚洲精品久久午夜玫瑰园 | 日韩欧美视频在线 | 精品欧美一区二区三区久久久 | 九久久| 色橹橹欧美在线观看视频高清 | 国产成人jvid在线播放 | 国产精品夜夜春夜夜爽久久电影 | 欧美日韩免费在线 | 免费在线观看一区二区三区 | 91青娱乐在线 | 日韩在线观看网站 | 99精品国产一区二区三区 | 综合久久久久 | 亚洲一区在线免费观看 | 国产精品美女久久久久久免费 | 在线91| 日韩中文字幕在线免费 | 国产一二三区精品视频 | 国产中文字幕在线观看 | 97视频在线免费 | 国产日韩欧美 | 亚洲精品视频免费观看 | 一级国产精品一级国产精品片 | 亚洲 精品 综合 精品 自拍 | 精精久久 | 国产精品不卡 | 欧美一级大片免费观看 | 国产精品大片 | 91干b| 成人在线观看免费 | 亚洲一区二区三区久久 | 四虎永久免费黄色影片 | 一级片在线观看 |