51单片机普通IO口模拟串行口之查询方式

发布者:EtherealBeauty最新更新时间:2012-08-04 来源: 61ic关键字:51单片机  普通IO口  模拟串行口  查询方式 手机看文章 扫描二维码
随时随地手机看文章

论坛新老朋友们。祝大家新年快乐。在新的一年开始的时候,给大家一点小小的玩意。工程师经常碰到需要多个串口通信的时候,而低端单片机大多只有一个串行口,甚至没有串口。这时候无论是选择高端芯片,还是更改系统设计都是比较麻烦的事。我把以前搞的用普通I/O口模拟串行口通讯的程序拿出来,供大家参考,希望各位兄弟轻点拍砖。基本原理:我们模拟的是串行口方式1.就是最普通的方式。一个起始位、8个数据位、一个停止位。模拟串行口最关键的就是要计算出每个位的时间。以波特率9600为例,每秒发9600个位,每个位就是1/9600秒,约104个微秒。我们需要做一个精确的延时,延时时间+对IO口置位的时间=104微秒。起始位是低状态,再延时一个位的时间。停止位是高状态,也是一个位的时间。数据位是8个位,发送时低位先发出去,接收时先接低位。了解这些以后,做个IO模拟串口的程序,就是很容易的事。我们开始。先上简单原理图:就一个MAX232芯片,没什么好说的,一看就明白。使用单片机普通I/O口,232数据输入端使用51单片机P3.2口(外部中断1口,接到普通口上也可以,模拟中断方式的串行口会有用。呵呵)。数据输出为P0.4(随便哪个口都行)。
点击浏览下一页


下面这个程序,您只需吧P0.4 和P3.2 当成串口直接使用即可,经过测试完全没有问题.
2、底层函数代码如下:

sbit TXD1 = P0^4;	//定义模拟输出脚
sbit RXD1 = P3^2;   //定义模拟输入脚

bdata unsigned char SBUF1;   //定义一个位操作变量
sbit SBUF1_bit0 = SBUF1^0;
sbit SBUF1_bit1 = SBUF1^1;
sbit SBUF1_bit2 = SBUF1^2;
sbit SBUF1_bit3 = SBUF1^3;
sbit SBUF1_bit4 = SBUF1^4;
sbit SBUF1_bit5 = SBUF1^5;
sbit SBUF1_bit6 = SBUF1^6;
sbit SBUF1_bit7 = SBUF1^7;

void delay_bps() {unsigned char i; for (i = 0; i < 29; i++); _nop_();_nop_();} //波特率9600 模拟一个9600波特率

unsigned char getchar2()	//模拟接收一个字节数据
{
	while (RXD1);
    _nop_();_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();
    delay_bps();
    SBUF1_bit0 = RXD1; //0       
    delay_bps();
    SBUF1_bit1 = RXD1; //1       
    delay_bps();
    SBUF1_bit2 = RXD1; //2       
    delay_bps();
    SBUF1_bit3 = RXD1; //3       
    delay_bps();
    SBUF1_bit4 = RXD1; //4       
    delay_bps();
    SBUF1_bit5 = RXD1; //5       
    delay_bps();
    SBUF1_bit6 = RXD1; //6       
    delay_bps();
    SBUF1_bit7 = RXD1; //7       
	delay_bps();
    return(SBUF1) ;      //返回读取的数据
}

void putchar2(unsigned char input)    //模拟发送一个字节数据
{
    SBUF1 = input;
    TXD1 = 0;  //起始位
    delay_bps();    
    TXD1 = SBUF1_bit0;  //0
    delay_bps();    
    TXD1 = SBUF1_bit1;  //1
    delay_bps();    
    TXD1 = SBUF1_bit2;  //2
    delay_bps();    
    TXD1 = SBUF1_bit3;  //3
    delay_bps();    
    TXD1 = SBUF1_bit4;  //4
    delay_bps();    
    TXD1 = SBUF1_bit5;  //5
    delay_bps();    
    TXD1 = SBUF1_bit6;  //6
    delay_bps();    
    TXD1 = SBUF1_bit7;  //7
    delay_bps();    
    TXD1 = 1;       //停止位
    delay_bps();    
}

3、实现串行通讯。在主程序文件中直接调用上面的getchar2()和putchar2()函数,配合电脑的串行口,即可实现串行通讯功能
点击浏览下一页
4、请参考完整程序文件,不过此串行通讯为程序查询方式,如果程序中有中断程序,很可能会造成接收数据丢失。在一会继续发一个帖子,把利用中断方式串行通讯程序也发来给大家看看。注意问题:1、波特率是可以有误差,但每个位的误差,不能大于3%2、中断可能会改变延时的时间。如果你的中断里的程序较长,应该在模拟串口接收和发送时禁止中断。3、接收时要延时1.5个的位时间(一个起始位+半个数据位)。使数据位的采样点尽量放在数据位的中间。[page]

完整程序工程源代码:点击下载
 
主程序:

#include 
#include "delay.h"
#include "sub4094.c"
#include 
sbit spk = P2^5;    //定义蜂鸣器使用的I/O口P2.5
sbit LED = P2^7;
#include "subuart2.c"


void main (void)
{
unsigned char first,zjgs,order,zhen_xh,jym,end;  //定义起始字、字节个数、命令码、帧序号、校验码、结束字
unsigned char i;       //定义1个随机变量
unsigned char sum;     //定义单片机计算用的校验码
unsigned char LED_contrl;  //指示灯控制字
unsigned contrl_1,contrl_2; //移位变量
//unsigned int delay_counter;
P5=0xEF;   //使能流水灯,屏蔽数码管
P4=0x00;      //流水灯全部点亮	
update4094(); //刷新流水灯状态
delay_ms(300);
P4=0xFF;      //流水灯全部熄灭
update4094(); //刷新流水灯状态
while(1)
{
	first=getchar2();   //读取6个数据进行处理。 
	zjgs=getchar2();
	order=getchar2();
	zhen_xh=getchar2();
	jym=getchar2();
	end=getchar2();
	if(0xfa != first) goto end;
	sum=zjgs+order+zhen_xh;
	if(sum != jym) 
	{
		putchar2(0xfa);   //起始字
		putchar2(0x07);    //字节个数
		sum=0x07;
		putchar2(order);   //接收到的命令码
		sum+=order;
		putchar2(zhen_xh); //接收到的帧序号
        putchar2(0x00);    //命令校验错误标志位
		sum+=zhen_xh;
		putchar2(sum);     //校验码
		putchar2(0xfb);
		//蜂鸣器发出报警声音,指示灯闪烁
		for(i=0;i<8;i++)  
		{
		    LED=~LED;     //取反指示灯
			spk=~spk;     //取反蜂鸣器
			delay_ms(200); 
		}
		goto end;
	} 
	if(0xfb != end) goto end;
	switch(order)
	{
		case 1: //将收到的命令返回给串行口
				LED=0;
				putchar2(first);      //起始字  
				putchar2(zjgs);       //字节个数 
				putchar2(order);      //命令码
				putchar2(zhen_xh);    //帧序号
				putchar2(jym);        //校验码
				putchar2(end);      //结束字
				delay_ms(50);
				LED=1;
				//流水灯效果 循环右移
				P4=0xff;                       //熄灭所有指示灯
				update4094();
   				LED_contrl=0x01;               //初始化指示灯控制字节
 				delay_ms(50);                 //延时300MS
 				for(i=0;i<8;i++)
        		{
            	    P4=~LED_contrl;            //点亮控制字节相应指示灯
            		update4094();
					delay_ms(50);
					LED_contrl<<=1;
        		}
				P4=0xff;                       //熄灭所有指示灯
				update4094();
				break;
		case 2: //将收到的命令返回给串行口
				putchar2(first);      //起始字  
				putchar2(zjgs);       //字节个数 
				putchar2(order);      //命令码
				putchar2(zhen_xh);    //帧序号
				putchar2(jym);        //校验码
				putchar2(end);      //结束字
				//流水灯效果 从左到右逐个点亮
				P4=0xff;         //熄灭所有指示灯
    			update4094();
				LED_contrl=0xff;   //初始化指示灯控制字节
 				delay_ms(50);
  				for(i=0;i<8;i++)
     			{
   				    LED_contrl<<=1;
		    		P4=LED_contrl;
					update4094();
					delay_ms(50);
                }
				break;
		case 3: //将收到的命令返回给串行口
				putchar2(first);      //起始字  
				putchar2(zjgs);       //字节个数 
				putchar2(order);      //命令码
				putchar2(zhen_xh);    //帧序号
				putchar2(jym);        //校验码
				putchar2(end);      //结束字
				//流水灯效果  循环对撞
				P4=0xff;          //熄灭所有指示灯
				update4094();
				contrl_1=0x02;   //初始化移位变量1
 				contrl_2=0x80;   //初始化移位变量2
 				delay_ms(50);
  				for(i=0;i<8;i++)
    			{
		            LED_contrl=contrl_1|contrl_2;
  					P4=~LED_contrl; //点亮控制字节相应指示灯
     				update4094();
					delay_ms(50);
  					contrl_1<<=1;  //移位变量1左移1位
   					contrl_2>>=1;  //移位变量2右移1位
     			} 
                P4=0xff;                       //熄灭所有指示灯
				update4094();
                break;
		default:break;
	}
	end:;

}
}
关键字:51单片机  普通IO口  模拟串行口  查询方式 引用地址:51单片机普通IO口模拟串行口之查询方式

上一篇:ATMEGA16读写24C02 C51程序
下一篇:LCD1602液晶屏+DS1302时钟芯片的电子时钟

推荐阅读最新更新时间:2024-03-16 13:04

51 单片机串行通讯中波特率的自动检测
本文介绍一种在 80C51 串行通讯应用中自动检测波特率的方法。按照经验,程序起动后所接收到的第1个字符用于测量波特率。 这种方法可以不用设定难于记忆的开关,还可以免去在有关应用中使用多种不同波特率的烦恼。人们可以设想:一种可靠地实现自动波特检测的方法是可能的,它无须严格限制可被确认的字符。问题是:在各种的条件下,如何可以在大量允许出现的字符中找出波特率的定时间隔。 显然,最快捷的方法是检测一个单独位时间(single bit time),以确定接收波特率应该是多少。可是,在 RS-232 模式下,许多 ASCII 字符并不能测量出一个单独位时间。对于大多数字符来说,只要波特率存在合理波动(这里的波特率是指标准波特率)
[单片机]
51 单片机串行通讯中波特率的自动检测
19-基于51单片机的倒计时系统
具体实现功能 系统由STC89C52单片机+按键电路+复位电路+晶振电路+LCD1602显示模块构成。 具体功能: (1)六位LED显示,从59分59秒99开始倒计时; (2)倒计时精度为0.01秒,能正确地进行倒计时; (3)复位后倒计时重新回到初始状态。 设计背景 近年来随着科技的飞速发展,单片机的应用正在不断的走向深入。本文阐述了基于单片机的数字电子秒表倒计时设计,倒计时精度达0.01s了,解决了传统的由于倒计时精度不够造成的误差和不公平性,是各种体育竞赛的必备设备之一。本设计的数字电子秒表倒计时系统采用8051单片机为中心器件,利用其定时器/计数器定时和记数的原理,结合显示电路、LED数码管以及外部中断
[单片机]
19-基于<font color='red'>51单片机</font>的倒计时系统
51单片机做的拉幕式数码显示课程设计
简介:用AT89S51单片机的P0.0/AD0-P0.7/AD7端口接数码管的a-h端,8位数码管的S1-S8通过74LS138译码器的Y0-Y7来控制选通每个数码管的位选端。AT89S51单片机的P1.0-P1.2控制74LS138的A,B,C端子。在8位数码管上从右向左循环显示“12345678”。能够比较平滑地看到拉幕的效果。 1. 电路原理图 2. 系统板上硬件连线 (1. 把“单片机系统”区域中的P0.0/AD0-P0.7/AD7用8芯排线连接到“动态数码显示”区域中的a-h端口上; (2. 把“三八译码模块”区域中的Y0-Y7用8芯排线连接到“动态数码显示”区域中的S1-S8端口上; (3. 把“单
[单片机]
用<font color='red'>51单片机</font>做的拉幕式数码显示课程设计
51单片机的串行通信口原理解析
MCS-51单片机内部有一个全双工的串行通信口,即串行接收和发送缓冲器(SBUF),这两个在物理上独立的接收发送器,既可以接收数据也可以发送数据。但接收缓冲器只能读出不能写入,而发送缓冲器则只能写入不能读出,它们的地址为99H。这个通信口既可以用于网络通信,亦可实现串行异步通信,还可以构成同步移位寄存器使用。如果在传行口的输入输出引脚上加上电平转换器,就可方便地构成标准的RS-232接口。下面我们分别介绍。 [1]。 基本概念 数据通信的传输方式 常用于数据通信的传输方式有单工、半双工、全双工和多工方式。 单工方式:数据仅按一个固定方向传送。因而这种传输方式的用途有限,常用于串行口的打印数据传输与简单系统间的数据采集。 半双工
[单片机]
<font color='red'>51单片机</font>的串行通信口原理解析
MCS-51单片机的程序设计方法
  程序设计是用计算机解决实际问题过程中的一个环节。用计算机解决实际问题的过程包括了建立数学模型、选择计算方法、设计程序以及上机调试、运行程序并分析计算结果两个阶段。在分析一个具体的问题后,我们可以得出输入和输出量之间的关系,并用数学式进行描述,这个过程称之为建立数学模型。有了数学模型还必须选择适合于计算机实现的具体计算方法,然后再设计程序。程序设计实际上是对具体算法的实施。设计的程序还需在一定的调试环境中进行调试,此时应输入数据,并检查输出数据是否正确,为保证解题的正确性,应设计合理的输入数据,以检查程序可能出现的错误。   MCS-51单片机的程序设计通常分为下面3个步骤:   1.制作程序流程图   对提出的算法找出最合理、最
[单片机]
1-基于51单片机的病床呼叫系统仿真
具体实现功能 具体功能: (1)当病人紧急呼叫时,会产生声、光提示,并显示病人编号(房间号和床号); (2)根据病人病情设置优先级,当有多人呼叫时,病情严重者优先; (3)医护人员处理完当前最高级别呼叫后,系统按优先级别显示其它呼叫病人的号码; (4)可设定呼叫倒计时,总床位数为24个。 注意:本设计以仿真为主!! 单片机介绍 51单片是一种低功耗、高性能CMOS8位微控制器,具有 8K 在系统可编程Flash 存储器。在单芯片上,拥有灵巧的8 位CPU 和在系统可编程Flash,使得STC89C51为众多嵌入式控制应用系统提供高灵活、超有效的解决方案。具有以下标准功能:8k字节Flash,512字节RAM
[单片机]
1-基于<font color='red'>51单片机</font>的病床呼叫系统仿真
51单片机的主要组成结构分析详解
随着微电子技术的快速发展,以ARM为主的32位MCU(微控制器)已普及开来,8位MCU已被很多人认为将被淘汰,更何况其中的老古董MCS-51系列单片机。但从目前的形势来看,8位MCU还牢牢占据着工业控制领域的主导地位,一个原因是8位MCU的开发成本比较低,也有大量的成熟设计方案,还有一个原因是历史的延续,新的产品还处在不断变化中,老产品则是经数十年的淘汰而留存下来的精品。 MCS-51系列单片机正是如此,自1980年由Intel推出后,获得很大成功,并不断改进而形成系列,成为最普遍使用的单片机内核和指令系统。后来,ATMEL、NXP等多家著名半导体公司推出兼容和增强的51系列单片机,应用普遍,因此成为单片机教学的主要示例,熟悉其
[单片机]
一文详解80C51单片机的中断系统
中断系统是计算机或者单片机的主要功能部件。 有了中断系统,便可以使微处理器具备对外部的异步事件进行处理的能力。 当微处理器的CPU正在执行程序的过程中,如果外部硬件或者内部组件有紧急的请求(如通信,断点,发生重大故障等),中断系统就可以将当前的程序暂停,优先处理这些中断请求。 这种处理方式,对整个系统的稳定性,健壮性至关重要,同时也能大大提高处理器的效率,使得系统的应用更加灵活多变。 中断的概念 中断是指单片机在执行程序的过程中,当出现异常情况或特殊请求时,单片机停止当前程序的运行,转向对这些异常情况或特殊请求进行处理,当处理结束后再返回原程序的间断处,继续执行原程序,这一现象称为中断。 中断是单片机实时处理内部或外部事件的一
[单片机]
一文详解80C<font color='red'>51单片机</font>的中断系统
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved