| 
 
| 本帖最后由 xppx1987 于 2012-9-10 15:24 编辑 
 本人前两天发过数码管动态扫描的帖子,问题还是没有解决,显示问题:后面一个四位一体的数码管的第二个和第三个管子均不能正常显示(怀疑74HC164送段选出问题)!今天附上完整仿真图和程序,望各位指出错误!注:1、需要说明的是两个四位一体的共阴极数码管是共用的位选端口,占用C端口的低四位!2、连接数码管的位选端口的三极管的C级若去掉上拉电阻则不能正常显示,请问什么原因?小弟看见有的电路图没有加上拉电阻也行。
 #include<pic.h>
 #include<math.h>
 #include<stdio.h>
 
 __CONFIG(0xe0f1); //芯片配置字,看门狗关,上电延时开,掉电检测关,低压编程关,加密,4M晶体HS振荡 //__CONFIG(HS&WDTDIS&LVPDIS);
 #define data164 RA3
 #define clk164 RA5
 #define uchar unsigned char
 #define uint unsigned int
 
 #define  _nop_()  asm("nop")
 uchar count_10ms=0;
 float wendu,shidu;
 uint temp=123,humi=456;
 uchar const bit_tab[]={0x80,0x40,0x20,0x01,0x08,0x04,0x02,0x01,0};//位选表,用来选择哪一只数码管进行显示
 uchar const seg_data[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};//段选表,共阴数码显示码值
 uchar disbuf[6];  //定义显示缓冲单元
 
 /******函数定义**************/
 void sendbyte1(char byte);
 void sendbyte2(char byte);
 void conv(int tmp_s,int humi_s);
 void display();
 
 /********端口设置函数********/
 void port_init( )
 {
 TRISA=0x07;
 TRISB=0x06;
 TRISC =0x00;  //端口C输出,位选
 ADCON1=0x06;  //定义RA为IO端口
 ANSEL=0x00;
 ANSELH=0x00;
 }
 
 /*****************显示函数***********************/
 void sendbyte1(char byte)
 {
 unsigned char  c;
 for(c=0;c<8;c++)
 {
 if(byte&0x80)
 data164=1;
 else data164=0;
 clk164=0;
 _nop_();
 clk164=1;
 byte<<=1;
 }
 }
 
 void sendbyte2(char byte)
 {
 unsigned char  c;
 for(c=0;c<16;c++)
 {
 if(byte&0x80)
 data164=1;
 else data164=0;
 clk164=0;
 _nop_();
 clk164=1;
 byte<<=1;
 }
 }
 
 void conv(int tmp_s,int humi_s)
 {
 disbuf[0]=tmp_s/100;
 disbuf[1]=tmp_s%100/10;
 disbuf[2]=tmp_s%10;
 disbuf[3]=humi_s/100;
 disbuf[4]=humi_s%100/10;
 disbuf[5]=humi_s%10;
 }
 void display()
 {
 static uchar count=0;
 PORTC=bit_tab[8];
 switch(count)
 {
 case 0:sendbyte1(seg_data[disbuf[0]]);PORTC=bit_tab[4];break;
 case 1:sendbyte1(seg_data[disbuf[1]]);PORTC=bit_tab[5];break;
 case 2:sendbyte1(seg_data[disbuf[2]]);PORTC=bit_tab[6];break;
 case 3:sendbyte2(seg_data[disbuf[3]]);PORTC=bit_tab[4];break;
 case 4:sendbyte2(seg_data[disbuf[4]]);PORTC=bit_tab[5];break;
 case 5:sendbyte2(seg_data[disbuf[5]]);PORTC=bit_tab[6];break;
 }
 count++;
 if(count==6) count=0;
 }
 /*********定时器0/定时器1初始化函数********/
 void  timer_init()
 {
 GIE=1;     //开总中断
 PEIE=1;    //开外围功能模块中断
 T0CS=0;    //定时器0对内部时钟计数
 PSA=0;    //分频器分配给TRM0
 PS0=1;
 PS1=1;
 PS2=1;     //定时器0分频比为1:256
 T0IE=1;    //允许TMR0溢出中断
 TMR0=248;    //TMR0赋初值,定时2ms
 T1CKPS0=1;T1CKPS1=1;  //定时器1分频比为1:8
 TMR1CS=0;    //定时器1设置为定时功能
 TMR1IE=1;    //使能定时器1中断
 TMR1ON=1;    //启动定时器1
 TMR1H=0xfb;    //置计数值高位,定时时间为10ms
 TMR1L=0x1e;    //置计数值低位
 }
 /********中断服务程序********/
 void interrupt ISR(void)
 {
 if(TMR0IF==1)
 {
 TMR0IF=0;   //清TMR1溢出中断标志位
 TMR0=248;                       //TMR0赋初值,定时2ms
 display();   //调显示函数
 }
 if(TMR1IF==1)
 {
 TMR1IF=0;   //清TMR1溢出中断标志位
 TMR1H=0xfb;   //重置计数值,定时时间为10ms
 TMR1L=0x1e;   //重置计数值
 count_10ms++;  //10ms计数器加1
 if(count_10ms>=100)
 {
 count_10ms = 0;   //计数100次后恰好为1s,此时10ms计数器清零
 }
 }
 }
 
 /*************************主函数****************************/
 void main( )
 {
 port_init();
 timer_init();           //调定时器T0、T1初始化函数
 while(1)
 {
 conv(temp,humi); //调走时转换函数
 }
 }
 
 | 
 
×本帖子中包含更多资源您需要 登录 才可以下载或查看,没有账号?注册 
  |