#define TM0_FLAG P1_2 //设传输标志位

//计数器及中断初始化

void S2INI(void)

{

TMOD |=0x02; //计数器0,方式2

TH0=0xA0; //预值为256-96=140,十六进制A0

TL0=TH0;

TR0=0; //在发送或

接收才开始使用

TF0=0;

ET0=1; //允许定时

器0中断

EA=1; //中断允许

总开关

}

//接收一个字符

uchar RByte()

{

uchar output=0;

uchar i=8;

TR0=1; //启动Timer0

TL0=TH0;

WaitTF0(); //等过起始

//发送8位数据位

while(i--)

{

Output 》》=1;

if(RXD) Output |=0x80; //先收低位

WaitTF0(); //位间延时

}

while(!TM0_FLAG) if(RXD) break;

TR0=0; //停止

Timer0

return Output;

}

//中断1处理程序

void IntTimer0() interrupt 1

{

TM0_FLAG=1; //设置标志位。

}

//查询传输标志位

void WaitTF0( void )

{

while(!TM0_FLAG);

TM0_FLAG=0; //清标志位

}

中断法也是我推荐的方法,和计数法大同小异。发送程序参考计数法,相信是件很容易的事。

另外还需注明的是本文所说的串口就是通常的三线制异步通信串口(UART),只用RXD、TXD、GND。

附:51 IO口模拟串口通讯C源程序(定时器计数法)

#i nclude

sbit BT_SND =P1^0;

sbit BT_REC =P1^1;

/**********************************************

IO 口模拟232通讯程序

使用两种方式的C程序 占用定时器0

**********************************************/

#define MODE_QUICK

#define F_TM F0

#define TIMER0_ENABLE TL0=TH0; TR0=1;

#define TIMER0_DISABLE TR0=0;

sbit ACC0= ACC^0;

sbit ACC1= ACC^1;

sbit ACC2= ACC^2;

sbit ACC3= ACC^3;

sbit ACC4= ACC^4;

sbit ACC5= ACC^5;

sbit ACC6= ACC^6;

sbit ACC7= ACC^7;

void IntTimer0() interrupt 1

{

F_TM=1;

}

//发送一个字符

void PSendChar(unsigned char inch)

{

#ifdef MODE_QUICK

ACC=inch;

F_TM=0;

BT_SND=0; //start bit

TIMER0_ENABLE; //启动

while(!F_TM);

BT_SND=ACC0; //先送出低位

F_TM=0;

while(!F_TM);

BT_SND=ACC1;

F_TM=0;

while(!F_TM);

BT_SND=ACC2;

F_TM=0;

while(!F_TM);

BT_SND=ACC3;

F_TM=0;

while(!F_TM);

BT_SND=ACC4;

F_TM=0;

while(!F_TM);

BT_SND=ACC5;

F_TM=0;

while(!F_TM);

BT_SND=ACC6;

F_TM=0;

while(!F_TM);

BT_SND=ACC7;

F_TM=0;

while(!F_TM);

BT_SND=1;

F_TM=0;

while(!F_TM);

TIMER0_DISABLE; //停止timer

#else

unsigned char ii;

ii=0;

F_TM=0;

BT_SND=0; //start bit