A. 單片機矩陣按鍵的模擬,求高手!!!!
這是我做的鍵掃描程序,你可以參考一下,你那個程序當然可以模擬出來,但是,實際電路是不可能這樣接的。
void delay(uchar a)
{
uchar i,j;
for(i=0;i<a;i++)
for(j=0;j<125;j++);
}
uchar kbscan(void)
{
unsigned char sccode,recode;
P3=0x0f; //發0掃描,列線輸入
if ((P3 & 0x0f) != 0x0f) //有鍵按下
{
delay(20); //延時去抖動
if ((P3&0x0f)!= 0x0f)
{
sccode = 0xef; //逐行掃描初值
while((sccode&0x01)!=0)
{
P3=sccode;
if((P3&0x0f)!=0x0f)
{
recode=(P3&0x0f)|0xf0;
while((P3&0x0f)!=0x0f);//等待鍵抬起
return((~sccode)+(~recode));
}
else
sccode=(sccode<<1)|0x01;
}
}
}
return 0; //無鍵按下,返回0
}
void getkey(void)
{
unsigned char key;
key=kbscan();
if(key==0)
{
return;
}
switch(key)
{
case 0x11:keyval=7;break;
case 0x12:keyval=4;break;
case 0x14:keyval=1;break;
case 0x18:keyval=10;break;
case 0x21:keyval=8;break;
case 0x22:keyval=5;break;
case 0x24:keyval=2;break;
case 0x28:keyval=0;break;
case 0x41:keyval=9;break;
case 0x42:keyval=6;break;
case 0x44:keyval=3;break;
case 0x48:keyval=11;break;
case 0x81:keyval=12;break;
case 0x82:keyval=13;break;
case 0x84:keyval=14;break;
case 0x88:keyval=15;break;
default:keyval=0xff;break;
}
}
B. proteus模擬電路圖,51單片機按鍵計數
按照你題目,用了2個2位顯示,實際有4位合一起的。
k3:切換計數模式/預置模式。
計數模式:LED顯示計時數字,從0開始計時,直到預置最大值。
預置模式:LED顯示當前預置最大值,按k1,k2可對預置值+-操作,長按k1,k2大約2秒,會進入自動加減預置值。直到再次點擊k1,k2,k3任意一鍵停止自動。
k4:在計數模式下使用,每按下一次顯示的數字加一(會在正常計時同時額外+1)。
當計數達到預置最大值,會停止計數,LEN閃爍(實際就是交替顯示間隔邊長),蜂鳴器響。
按鍵時長、LED動態顯示間隔、閃爍間隔、計數速度,均可直接修改常量,需要自己改,我備注寫的很詳細。
電路基本按照你上圖,略有修改。
#include <reg52.h>
#define uint unsigned int
#define uchar unsigned char
#define an P0
#define on 0
#define off 1
#define SSSPEED 35 //LED交替閃爍間隔時間
#define JSPEED 5000//計數模式,速度默認數值(5000*200us=1S) 值越小計數越快
#define PREESTIME 500//按鈕長按時間判定,預設500(大約2秒),需要自改,值越大,長按時間越長
sbit fm=P3^3;
sbit wei1=P3^4;
sbit wei2=P3^5;
sbit wei3=P3^6;
sbit wei4=P3^7;
sbit k1=P1^4;
sbit k2=P1^5;
sbit k3=P1^6;
sbit k4=P1^7;
uint jsSpd=JSPEED;//計時速度,默認1s一次(5000*200us)
uint ssSpd=SSSPEED;//LED交替閃爍速度
//共陽極
int delay(uint xms);
void init();
void jspause();//計數器開啟/停止
void setnumYS();//設置預設數值
void numJsChange();//計數模式數字改變
void showLED();
int pressWait(uint btn);
uint g=0;
uint s=0;
uint b=0;
uint q=0;
uint count=0;
uint ispause=1;
uint numYS=0;//預設數值
uint numJS=0;//實際計時的數字
uint isMaxJs=0;//標識:計時達最大。 達最大1,否0
uint isk3press=0;//標識:k3按鈕是否被點擊。 點擊1,否0
uint ispress1=0;//標識:k1被長按
uint ispress2=0;//標識:k2被長按
uint isbtn4=0;//標識:k4被按下
uint btnName=0;//按鈕長按計時
void main()
{
init();
while(1)
{
if(ispause==1 && ispress1==1 && numYS<9999) //預置模式下,k1已長按,自動增
{
numYS++;
setnumYS();
}
if(ispause==1 && ispress2==1 && numYS>0) //預置模式下,k2已長按,自動減
{
numYS--;
setnumYS();
}
if(isMaxJs==0 && numJS>=numYS && ispause==0) //計時模式下達最大值
{
fm=on;
ssSpd=1000;//增加LED交替間隔,實現數字閃爍
isMaxJs=1;
EA=0;
setnumYS();
numJS=0;
}
if(k1==0 ||k2==0|| k3==0) //k1k2k3任意一個按鈕被按下,停止預置數自動增長
{
ispress1=0;
ispress2=0;
}
if(k1==0 && ispause==1)//預置模式下+
{
delay(10);
if(k1==0)
{
btnName=1;
if(pressWait(btnName))//判斷連按
{
while(k1==0);
ispress1=1;
}
else if(numYS<9999)
{
numYS++;
setnumYS();
}
}
}
if(k2==0 && ispause==1)//預置模式下-
{
delay(10);
if(k2==0)
{
btnName=2;
if(pressWait(btnName))//判斷連按
{
while(k2==0);
ispress2=1;
}
else if(numYS>0)
{
numYS--;
setnumYS();
}
}
}
if(k3==0)
{
delay(10);
if(k3==0)
{
while(k3==0);
fm=off;
jspause();
}
}
if(k4==0 && ispause==0)//計數模式下按下k4,k4的防抖寫在中斷中
{
delay(10);
if(k4==0)
{
while(k4==0);
isbtn4=1;
}
}
showLED();
}
}
void showLED()
{
uchar nums[10]={0xc0,0xf9,0xa4,0xB0,0x99,0x92,0x82,0xf8,0x80,0x98};
if(g>=0)
{
an=nums[g];
wei4=on;
delay(ssSpd);
wei4=off;
}
if(s>0 || (s==0 && b>0))
{
an=nums[s];
wei3=on;
delay(ssSpd);
wei3=off;
}
if(b>0 || (b==0 && q>0))
{
an=nums[b];
wei2=on;
delay(ssSpd);
wei2=off;
}
if(q>0)
{
an=nums[q];
wei1=on;
delay(ssSpd);
wei1=off;
}
}
void setnumYS()//設置預設數值
{
q=numYS/1000;
b=(numYS%1000)/100;
s=(numYS%100)/10;
g=numYS%10;
}
void jspause()
{
if(ispause==0 || isMaxJs==1)//關閉計時模式 / 啟動預置模式
{
EA=0;
isMaxJs=0;
ispause=1;
ssSpd=SSSPEED;
ispress1=0;
ispress2=0;
setnumYS();
}
else if(ispause==1) //啟動計時模式 / 關閉預置模式
{
ispause=0;
q=b=s=g=0;
numJS=0;
ssSpd=SSSPEED;
EA=1;
}
}
void init()
{
TMOD=0x02; //T0 工作模式2 自動裝填8位 200us
TH0=0x38;
TL0=0x38;
EA=0;
ET0=1;
TR0=1;
wei1=off;
wei2=off;
wei3=off;
wei4=off;
}
void numJsChange()//計數模式數字改變
{
if(g==9)
{
g=0;
if(s==9)
{
s=0;
if(b==9)
{
b=0;
if(q==9)
{
q=0;
}
else
q++;
}
else
b++;
}
else
s++;
}
else
g++;
}
void ct() interrupt 1 //一次中斷200us
{
if(count<jsSpd)
count++;
else
{
count=0;
numJsChange();
numJS++;
}
if(isbtn4==1)
{
isbtn4=0;
numJsChange();
numJS++;
}
}
int pressWait(uint btn)
{
uint i,j;
for(i=PREESTIME;i>0;i--)
for(j=110;j>0;j--)
{
if((k1==1 && btn==1)||(k2==1 && btn==2))
return 0;
}
return 1;
}
int delay(uint xms)
{
uint i,j;
for(i=xms;i>0;i--)
for(j=110;j>0;j--)
{
if(k1==0 || k2==0 ||k3==0)
return 1;
}
return 0;
}
C. MULTISIM怎麼模擬51單片機
1、⑴組建模擬電路
"" componet」對話框的「family」欄中先選取「805x」,="" 然後在「componet」欄下選取「8051」,最後單擊對話框右上角的「ok」按鈕。滑鼠箭頭將帶出一個「8052」模塊,在電子模擬平台上單擊滑鼠,將先彈出MCU向導對話框之一,第一欄是工作區路徑,第二欄是工作區取名,這里輸入「my mcu design1」,最後單擊下方的「Next」按鈕;
⑵再次彈出MCU向導對話框之二,共有4欄選項,第一欄為項目類型,一般選擇「Standard (標准) 」;第二欄選擇編程語言種類,單擊右側下拉箭頭,可以在「Assembly C匯編」和「C」語言之間選擇,這里選取「C」語言;則第三欄自動顯示「Hi-tech C51-Lite compiler」字樣; 第四欄可以輸入項目名稱,本文取名為「project1」,最後單擊下方的「Next」按鈕;
⑶再次彈出MCU向導對話框之三,有2個單選項:「Create emptyproject (創建空項目) 」和「Addsource file (添加源文件) 」, 這里選取「Add sourcefile」;下面欄中可以輸入後綴為「。 C」的源文件名,這里就用「LED001. c」最後單擊下方「Finish」按鈕完成。
⑷完成MCU向導對話框之後,這時能在電子模擬平台上顯示出原來調出單片機模塊U18051,右擊滑鼠退出元件操作。
⑸保存文件,在文件名欄輸入「led11」文件名,默認文件後綴「.ms10」。
⑹在電子模擬軟體Multisim10基本界面平台上,單擊主菜單「View /Design Toolbox」,在基本界面左側將打開「設計工具盒」窗口,單擊「led11」前的「 + 」號可以看到剛才命名的工作區「my mcudesign1」,在工作區下面是項目名「project1 」, 在項目名下面是源文件名「Led001. c」。
⑺編寫MCU源C語言程序(過程略);
⑻調出其他模擬元件,搭建單片機模擬電路,進行模擬;
2、51單片機用protues這個軟體模擬更好!裡面已經有現成的51模塊了。調用就可以用,電源、地、復位電路和時鍾等等都不用畫。只要把編好的程序下載到protues中的51單片機裡面就可以運行。外圍提供的器件可以自己搭建。非常好用。
3、可以參考《基於Multisim 10的51單片機模擬實戰教程:使用匯編和C語言》。
D. 51單片機實現多按鍵輸入
很簡單呀!!
先說30個按鍵吧,如果要加晶元,用7279就輕松搞定了,它還能實現多鍵按下的處理,不過那種處理很簡單,後鍵打斷前鍵的程序,或者不打斷,並不檢測幾個鍵組合的情況。如果不想加晶元,接到單片機上,可以採用交叉連接方法呀,單片機書上都有,網上一搜索一大片。
至於程序嗎,也很簡單,你想,交叉布局的話,當一個按鍵按下後,單片機的經緯線中,各有一根是通過按鍵導通的,當有兩個按鍵按下時,最少會有三根線導能,等等,這就需要你列個表,計算好,算出每種按鍵組合後,它反應到單片機的I/O口上會是什麼狀態,就可以計算出單片機讀到哪個數據時,就該是這個組合.
說的我都有點嫌羅嗦了,但確實很簡單.有問題的話再留言說吧
E. 51單片機鍵控四種流水燈,但是模擬時候按按鍵沒反應,求大佬解答
那key是用bit定義成了位變數,只能是0和1,可你程序卻讓key=1,2,3,4,這是錯誤的。很簡單的事,定義成8位無符號數變數,unsigned char key;
這就行了,為什麼用bit?
F. 51單片機按鍵模擬反應遲鈍,是怎麼回事
給你個建議
1、有顯示函數的
鍵按下?---含顯示子函數循環等鍵放開---放開鍵後處理--
顯示子函數即當延時函數用
2、51單片機常用低電平作有效信號
if(button4==0) //
G. 為51單片機設計一個上拉輸入和下拉輸入的按鍵電路並分析其工作原理
按鍵一般是上拉,單片機的IO通過電阻上拉高電平,按鍵正常高電平當按鍵按下短路,把IO變成低電平,單片機檢測到低電平表示有按鍵按下,按鍵下拉一般是把單片機IO通電阻接到GND,按鍵正常是低電平,當按鍵按下把IO拉高,單片機檢測到高電平表示有按鍵按下。
H. 怎麼用proteus來模擬51
用proteus來模擬51的步驟:
工具/原料:Proteus,keil c51
1、首先我們打開Proteus軟體,點擊左邊菜單欄的P按鈕,然後再搜索框里輸入80c51,選擇第一個就是80c51單片機;