① 如何使用51單片機利用紅外線發射接收對管或紅外接收一體化頭實現側車流量(計數)
樓上是對的,其實程序不復雜,但電路是比較復雜的,你了解下紅外波瓣寬度是多少,監測距離多遠豎皮,車間間距多少,要不然做出來沒意義的,當兩車離得近時分不出來,當車體較襲纖者遠或者紅外反射系數低時識別不到,這里除了拍薯電路處理外也可用光學處理,光學你就天天去擦鏡頭上的灰塵吧,電路處理得想辦法增大發射功率,壓縮垂直波瓣寬度,程序里的演算法還要濾波,等等,看了這些你覺得簡單嗎?還矩陣?
② 急求一紅外管計數器,51單片機控制的C程序
電路都沒看到就要程序?這個不難的 一個統計進來的人數 一個統計出去的人數 當人數是0的時候就給個控制信號關燈
③ 51單片機計數器怎麼用
51單片機是基礎入門的一個單片機,還是應用最廣泛的。
51單片機的定時/計數器的概念
單片機中,脈沖計數與時間之間的關系十分密切,每輸入一個脈沖,計數器的值就會自動累加1,只要相鄰兩個計數脈沖之間的時間間隔相等,則計數值就代表了時間的流逝,因此,單片機中的定時器和計數器其實是同一個物理的電子元件,只不過計數器記錄的是單片機外部發生的事情(接受的是外部脈沖),而定時器則是由單片機自身提供的一個非常穩定的計數器,這個穩定的計數器就是單片機上連接的晶振部件;MCS-51單片機的晶振經過12分頻之後提供給單片機穩定脈沖;晶振的頻率是非常准確的,所以單片機的計數脈沖之間的時間間隔也是非常准確的。
51單片機的定時/計數器的工作原理
加1計數器輸入的計數脈沖有兩個來源,一個是由系統的時鍾振盪器輸出脈沖經12分頻後送來;一個是T0或T1引腳輸入的外部脈沖源。
作為定時器使用時,定時器計數8051單片機片內振盪器輸出經過12分頻後的脈沖個數,即:每個機器周期使定時器T0/T1的寄存器值自動累加1,直到溢出,溢出後繼續從0開始循環計數;所以,定時器的解析度是時鍾振盪頻率的1/12;
作為計數器使用時,通過引腳T0(P3.4)或T1(P3.5)對外部脈沖信號進行計數,當輸入的外部脈沖信號發生從1到0的負跳變時,計數器的值就自動加1由於檢測一個從1到0的下降沿需要2個機器周期,因此要求被采樣的電平至少要維持一個機器周期。當晶振頻率為12MHz時,最高計數頻率不超過1/2MHz,即計數脈沖的周期要大於2微秒。;計數器的最高頻率一般是時鍾振盪頻率的1/24;
由此可知,不論是定時器還是計數器工作方式,定時器T0和T1均不佔用CPU的時間,除非定時器/計數器T0和T1溢出,才可能引起CPU中斷,轉而去執行中斷處理程序。所以說,定時器/計數器是單片機中效率高而工作靈活的部件。
④ 關於51單片機的紅外計數實驗問題,寫完程序後數碼管數值不會改變。求大神指點。。。。。。
以STC89C52RC單片機為例,部分程序如下:
void initTimer1(void) //初始化定時器1
{
AUXR |= 0x40; //定時器1工作於1T模式
TMOD |= 0x60; //設置定時器1運行模式2 (8-bit 自動裝載)
TL1 = TH1 = 0xff; //初始值
TR1 = 1; //定時器1開始運行
EA = 1; //開總中斷
ET1 = 1; //使能定時器1中斷
}
//--------------------------------------------
void T1int() interrupt 3 //中斷
{
ucCount++; //計數
flag = 1;
}
//------------------------------------------
這個表格定義一般用code:
unsigned char code num[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x98};//0----9
前面聲明變數增加ucCount和flag:
unsigned char ucCount = 0;
bit flag = 0;
主程序:
viod main(viod)
{
aa=1;
bb=1;
initTimer1();
m=0;
n=0;
while(1)
{
if(flag == 1)
{
flag = 0;
if(uiCount > 99) uiCount = 0;
n = uiCount / 10;
m = uiCount % 10;
}
cc=num[m];
aa=0;
delay_nms();
aa=1;
cc=num[m];
bb=0;
delay_nms();
bb=1;
}
以上程序沒有完整整理,但思路請參考。
PS:定義變數不要用諸如「c「這樣的簡單字母,因為容易和單片機內部寄存器混淆。
⑤ 基於51單片機的紅外接收模塊的c編程
#include<reg52.h> //包含單片機寄存器的頭文件
#include<intrins.h> //包含_nop_()函數定義的頭文件
sbit IR=P3^2; //將IR位定義為P3.2引腳
sbit RS=P2^0; //寄存器選擇位,將RS位定義為P2.0引腳
sbit RW=P2^1; //讀寫選擇位,將RW位定義為P2.1引腳
sbit E=P2^2; //使能信號位,將E位定義為P2.2引腳
sbit BF=P0^7; //忙碌標志位,,將BF位定義為P0.7引腳
sbit BEEP = P3^6; //蜂鳴器控制埠P36
unsigned char flag;
unsigned char code string[ ]= {"1602IR-CODE TEST"};
unsigned char a[4]; //儲存用戶碼、用戶反碼與鍵數據碼、鍵數據反碼
unsigned int LowTime,HighTime; //儲存高、低電平的寬度
/*****************************************************
函數功能:延時1ms
***************************************************/
void delay1ms()
{
unsigned char i,j;
for(i=0;i<10;i++)
for(j=0;j<33;j++)
;
}
/*****************************************************
函數功能:延時若干毫秒
入口參數:n
***************************************************/
void delay(unsigned char n)
{
unsigned char i;
for(i=0;i<n;i++)
delay1ms();
}
/*********************************************************/
void beep() //蜂鳴器響一聲函數
{
unsigned char i;
for (i=0;i<100;i++)
{
delay1ms();
BEEP=!BEEP; //BEEP取反
}
BEEP=1; //關閉蜂鳴器
delay(250); //延時
}
/*****************************************************
函數功能:判斷液晶模塊的忙碌狀態
返回值:result。result=1,忙碌;result=0,不忙
***************************************************/
unsigned char BusyTest(void)
{
bit result;
RS=0; //根據規定,RS為低電平,RW為高電平時,可以讀狀態
RW=1;
E=1; //E=1,才允許讀寫
_nop_(); //空操作
_nop_();
_nop_();
_nop_(); //空操作四個機器周期,給硬體反應時間
result=BF; //將忙碌標志電平賦給result
E=0;
return result;
}
/*****************************************************
函數功能:將模式設置指令或顯示地址寫入液晶模塊
入口參數:dictate
***************************************************/
void WriteInstruction (unsigned char dictate)
{
while(BusyTest()==1); //如果忙就等待
RS=0; //根據規定,RS和R/W同時為低電平時,可以寫入指令
RW=0;
E=0; //E置低電平(根據表8-6,寫指令時,E為高脈沖,
// 就是讓E從0到1發生正跳變,所以應先置"0"
_nop_();
_nop_(); //空操作兩個機器周期,給硬體反應時間
P0=dictate; //將數據送入P0口,即寫入指令或地址
_nop_();
_nop_();
_nop_();
_nop_(); //空操作四個機器周期,給硬體反應時間
E=1; //E置高電平
_nop_();
_nop_();
_nop_();
_nop_(); //空操作四個機器周期,給硬體反應時間
E=0; //當E由高電平跳變成低電平時,液晶模塊開始執行命令
}
/*****************************************************
函數功能:指定字元顯示的實際地址
入口參數:x
***************************************************/
void WriteAddress(unsigned char x)
{
WriteInstruction(x|0x80); //顯示位置的確定方法規定為"80H+地址碼x"
}
/*****************************************************
函數功能:將數據(字元的標准ASCII碼)寫入液晶模塊
入口參數:y(為字元常量)
***************************************************/
void WriteData(unsigned char y)
{
while(BusyTest()==1);
RS=1; //RS為高電平,RW為低電平時,可以寫入數據
RW=0;
E=0; //E置低電平(根據表8-6,寫指令時,E為高脈沖,
// 就是讓E從0到1發生正跳變,所以應先置"0"
P0=y; //將數據送入P0口,即將數據寫入液晶模塊
_nop_();
_nop_();
_nop_();
_nop_(); //空操作四個機器周期,給硬體反應時間
E=1; //E置高電平
_nop_();
_nop_();
_nop_();
_nop_(); //空操作四個機器周期,給硬體反應時間
E=0; //當E由高電平跳變成低電平時,液晶模塊開始執行命令
}
/*****************************************************
函數功能:對LCD的顯示模式進行初始化設置
***************************************************/
void LcdInitiate(void)
{
delay(15); //延時15ms,首次寫指令時應給LCD一段較長的反應時間
WriteInstruction(0x38); //顯示模式設置:16×2顯示,5×7點陣,8位數據介面
delay(5); //延時5ms
WriteInstruction(0x38);
delay(5);
WriteInstruction(0x38);
delay(5);
WriteInstruction(0x0C); //顯示模式設置:顯示開,有游標,游標閃爍
delay(5);
WriteInstruction(0x06); //顯示模式設置:游標右移,字元不移
delay(5);
WriteInstruction(0x01); //清屏幕指令,將以前的顯示內容清除
delay(5);
}
/************************************************************
函數功能:對4個位元組的用戶碼和鍵數據碼進行解碼
說明:解碼正確,返回1,否則返回0
出口參數:dat
*************************************************************/
bit DeCode(void)
{
unsigned char i,j;
unsigned char temp; //儲存解碼出的數據
for(i=0;i<4;i++) //連續讀取4個用戶碼和鍵數據碼
{
for(j=0;j<8;j++) //每個碼有8位數字
{
temp=temp>>1; //temp中的各數據位右移一位,因為先讀出的是高位數據
TH0=0; //定時器清0
TL0=0; //定時器清0
TR0=1; //開啟定時器T0
while(IR==0) //如果是低電平就等待
; //低電平計時
TR0=0; //關閉定時器T0
LowTime=TH0*256+TL0; //保存低電平寬度
TH0=0; //定時器清0
TL0=0; //定時器清0
TR0=1; //開啟定時器T0
while(IR==1) //如果是高電平就等待
;
TR0=0; //關閉定時器T0
HighTime=TH0*256+TL0; //保存高電平寬度
if((LowTime<370)||(LowTime>640))
return 0; //如果低電平長度不在合理范圍,則認為出錯,停止解碼
if((HighTime>420)&&(HighTime<620)) //如果高電平時間在560微秒左右,即計數560/1.085=516次
temp=temp&0x7f; //(520-100=420, 520+100=620),則該位是0
if((HighTime>1300)&&(HighTime<1800)) //如果高電平時間在1680微秒左右,即計數1680/1.085=1548次
temp=temp|0x80; //(1550-250=1300,1550+250=1800),則該位是1
}
a[i]=temp; //將解碼出的位元組值儲存在a[i]
}
if(a[2]=~a[3]) //驗證鍵數據碼和其反碼是否相等,一般情況下不必驗證用戶碼
return 1; //解碼正確,返回1
}
/*------------------二進制碼轉換為壓縮型BCD碼,並顯示---------------*/
void two_2_bcd(unsigned char date)
{
unsigned char temp;
temp=date;
date&=0xf0;
date>>=4; //右移四位得到高四位碼
date&=0x0f; //與0x0f想與確保高四位為0
if(date<=0x09)
{
WriteData(0x30+date); //lcd顯示鍵值高四位
}
else
{
date=date-0x09;
WriteData(0x40+date);
}
date=temp;
date&=0x0f;
if(date<=0x09)
{
WriteData(0x30+date); //lcd顯示低四位值
}
else
{
date=date-0x09;
WriteData(0x40+date);
}
WriteData(0x48); //顯示字元'H'
}
/************************************************************
函數功能:1602LCD顯示
*************************************************************/
void Disp(void)
{
WriteAddress(0x40); // 設置顯示位置為第一行的第1個字
two_2_bcd(a[0]);
WriteData(0x20);
two_2_bcd(a[1]);
WriteData(0x20);
two_2_bcd(a[2]);
WriteData(0x20);
two_2_bcd(a[3]);
}
/************************************************************
函數功能:主函數
*************************************************************/
void main()
{
unsigned char i;
LcdInitiate(); //調用LCD初始化函數
delay(10);
WriteInstruction(0x01);//清顯示:清屏幕指令
WriteAddress(0x00); // 設置顯示位置為第一行的第1個字
i = 0;
while(string[i] != '\0') //'\0'是數組結束標志
{ // 顯示字元 www.RICHMCU.COM
WriteData(string[i]);
i++;
}
EA=1; //開啟總中斷
EX0=1; //開外中斷0
ET0=1; //定時器T0中斷允許
IT0=1; //外中斷的下降沿觸發
TMOD=0x01; //使用定時器T0的模式1
TR0=0; //定時器T0關閉
while(1); //等待紅外信號產生的中斷
}
/************************************************************
函數功能:紅外線觸發的外中斷處理函數
*************************************************************/
void Int0(void) interrupt 0
{
EX0=0; //關閉外中斷0,不再接收二次紅外信號的中斷,只解碼當前紅外信號
TH0=0; //定時器T0的高8位清0
TL0=0; //定時器T0的低8位清0
TR0=1; //開啟定時器T0
while(IR==0); //如果是低電平就等待,給引導碼低電平計時
TR0=0; //關閉定時器T0
LowTime=TH0*256+TL0; //保存低電平時間
TH0=0; //定時器T0的高8位清0
TL0=0; //定時器T0的低8位清0
TR0=1; //開啟定時器T0
while(IR==1); //如果是高電平就等待,給引導碼高電平計時
TR0=0; //關閉定時器T0
HighTime=TH0*256+TL0; //保存引導碼的高電平長度
if((LowTime>7800)&&(LowTime<8800)&&(HighTime>3600)&&(HighTime<4700))
{
//如果是引導碼,就開始解碼,否則放棄,引導碼的低電平計時
//次數=9000us/1.085=8294, 判斷區間:8300-500=7800,8300+500=8800.
if(DeCode()==1) // 執行遙控解碼功能
{
Disp();//調用1602LCD顯示函數
beep();//蜂鳴器響一聲 提示解碼成功
}
}
EX0=1; //開啟外中斷EX0
}
⑥ 關於C51單片機紅外線驅動LED發射和接收問題
最簡單的,發送端:按一下鍵發一個脈沖;接收端:進行脈沖計數,用led顯示。
發送端://12M晶體
#include "reg51.h"
#define KeyB P1^6
#define OutPin P1^7
bit Key1,Key2;
//timer init 20ms定時中斷
void initTimer(void)
{
TMOD=0x1;
TH0=0x3c;
TL0=0xb0;
}
//timer0/counter0 interrupt
void timer0(void) interrupt 1
{
TH0=0x3c;
TL0=0xb0;
//add your code here.
Key1=Key2;
Key2=KeyB;
}
//the main fun
void main(void)
{
initTimer();
TR0=1;
ET0=1;
EA=1;//開中斷
while(1)
{
OutPin=key1||key2;
}
}
接收端:
#include "reg51.h"
char Count
/薯冊/int0 interrupt
void int0(void) interrupt 0
{
//add your code here
Count++;
}
//the main fun
void main(void)
{
IT0=1; //INT0下降沿中斷
EX0=1; //允許塵中INT1中斷
EA=1;
Count=0;//初始化計數器
while(1)派手山
{
P1=~Count;//圖中的led應該是串電阻公共端接電源,P1口拉低亮,置高滅。
}
}
⑦ 關於51單片機紅外解碼程序,哪位大俠幫我看下
我以前做的一個項目,紅外遙控開關,解碼部分的code,供參考
6121碼,外部中斷0,at89s52
void int0() interrupt 0 //外部中斷1服務函數,紅外解碼程序
{
static uchar wei; //定義靜態變數
static uchar pp; //定義靜態變數
if(tt<56&&tt>50) {d2=0; tt=0;pp=0;wei=0;}//起始信號符合,將d2標記為0,各變數清零
if(tt>11)tt=0;
if(d2==0&&tt>=4)
{
buf[pp]>>=1;
if(tt>5) buf[pp]|=0x80; //如果時間大於780us ,則視為收到數據1
wei++;
if(wei==8)
{
pp++;
wei=0;
if(pp==4) { pp=0; d2=1;} //接收滿4個位元組,標志位清除
} //在 d2為0期間進入中斷8次,說明已經收到一個位元組數據,位元組號加1
}
tt=0;//每次進入中斷都清零
}
void timer1() interrupt 3 //紅外解碼計時
{
tt++;
}