用C语言编写串口程序
- 格式:doc
- 大小:29.50 KB
- 文档页数:6
51单片机的串口通信程序(C语言) 51单片机的串口通信程序(C语言)在嵌入式系统中,串口通信是一种常见的数据传输方式,也是单片机与外部设备进行通信的重要手段之一。
本文将介绍使用C语言编写51单片机的串口通信程序。
1. 硬件准备在开始编写串口通信程序之前,需要准备好相应的硬件设备。
首先,我们需要一块51单片机开发板,内置了串口通信功能。
另外,我们还需要连接一个与单片机通信的外部设备,例如计算机或其他单片机。
2. 引入头文件在C语言中,我们需要引入相应的头文件来使用串口通信相关的函数。
在51单片机中,我们需要引入reg51.h头文件,以便使用单片机的寄存器操作相关函数。
同时,我们还需要引入头文件来定义串口通信的相关寄存器。
3. 配置串口参数在使用串口通信之前,我们需要配置串口的参数,例如波特率、数据位、停止位等。
这些参数的配置需要根据实际需要进行调整。
在51单片机中,我们可以通过写入相应的寄存器来配置串口参数。
4. 初始化串口在配置完串口参数之后,我们需要初始化串口,以便开始进行数据的发送和接收。
初始化串口的过程包括打开串口、设置中断等。
5. 数据发送在串口通信中,数据的发送通常分为两种方式:阻塞发送和非阻塞发送。
阻塞发送是指程序在发送完数据之后才会继续执行下面的代码,而非阻塞发送是指程序在发送数据的同时可以继续执行其他代码。
6. 数据接收数据的接收与数据的发送类似,同样有阻塞接收和非阻塞接收两种方式。
在接收数据时,需要不断地检测是否有数据到达,并及时进行处理。
7. 中断处理在串口通信中,中断是一种常见的处理方式。
通过使用中断,可以及时地响应串口数据的到达或者发送完成等事件,提高程序的处理效率。
8. 串口通信实例下面是一个简单的串口通信实例,用于在51单片机与计算机之间进行数据的传输。
```c#include <reg51.h>#include <stdio.h>#define BAUDRATE 9600#define FOSC 11059200void UART_init(){TMOD = 0x20; // 设置定时器1为模式2SCON = 0x50; // 设置串口为模式1,允许接收TH1 = 256 - FOSC / 12 / 32 / BAUDRATE; // 计算波特率定时器重载值TR1 = 1; // 启动定时器1EA = 1; // 允许中断ES = 1; // 允许串口中断}void UART_send_byte(unsigned char byte){SBUF = byte;while (!TI); // 等待发送完成TI = 0; // 清除发送完成标志位}unsigned char UART_receive_byte(){while (!RI); // 等待接收完成RI = 0; // 清除接收完成标志位return SBUF;}void UART_send_string(char *s){while (*s){UART_send_byte(*s);s++;}}void main(){UART_init();UART_send_string("Hello, World!"); while (1){unsigned char data = UART_receive_byte();// 对接收到的数据进行处理}}```总结:通过以上步骤,我们可以编写出简单的51单片机串口通信程序。
void main(){delayms(100);init(); //初始化系统delayms(100);init_wdt(); //初始化看门狗while(1){while(!RI_0) //是否收到数据{clr_wdt();}RI_0=0; //清除接收中断标志buffer=S0BUF;if(buffer==0x5a) //检测祯头0start0=1;if(buffer==0x54) //检测祯头1start1=1;if(buffer==0x5a) //检测祯尾0end0=1;if(buffer==0xfe) //检测祯尾1end1=1;if((start0==1)&(start1==1)){buff[i]=buffer; //从祯头1开始存储数据i++;}if((end0==1)&(end1==1)) //是否已经接收祯尾{count=i; //数据长度为count个i=1;if((buff[2]==0x03)&(count==107)) //是否422指令 {buff[0]=0x5a; //重填祯头0buff[count-4]=0; //校验和清零for(k=2;k<(count-4);k++) //计算校验和{buff[count-4]+=buff[k];}for(k=0;k<count;k++) //从祯头1开始循环发送{S0BUF=buff[k];while(!TI_0); //等待发送完成TI_0=0; //清除发送中断标志}reset();}else if((buff[2]==0x05)&(count==7)) //是否AD测试指令{sendad();reset();}else if((buff[2]==0x18)&(count==7)) //是否发送时序信号指令{sendpaulse();reset();}else //如果接收错误,则恢复各标志位为初始状态以便下次接收 {reset();}}}}void reset(){start0=0; //祯头祯尾标志位清零start1=0;end0=0;end1=0;for(k=0;k<count;k++){buff[k]=0x00; //缓冲区清零}count=0; //计数清零}。
c语言怎么写串口通信编程 -回复在C语言中进行串口通信编程,你需要了解串口的基本原理以及相关的函数和库。
串口通信是一种通过串行数据传输进行通信的方式,常用于嵌入式系统中与外部设备进行数据交互。
本文将以步骤的形式来介绍如何在C语言中进行串口通信编程。
步骤一:了解串口的基本原理在开始串口通信编程之前,你需要了解串口的基本原理。
串口是通过发送和接收数据位的序列来进行通信的。
串口通信需要考虑的一些参数包括波特率(即数据传输速率)、数据位数、校验位和停止位等。
波特率指的是每秒钟传输的位数,可以是常用的9600、19200、38400等。
步骤二:选择合适的串口库在C语言中,你可以选择使用合适的串口库来简化串口通信的编程工作。
常见的串口库包括Windows系统中的WinAPI、Linux系统中的termios库等。
选择库的时候,需根据你所使用的操作系统和开发环境进行选择。
步骤三:打开串口在开始使用串口进行通信之前,需要先打开串口。
使用串口库的函数,可以根据需要选择打开特定的串口,一般通过指定串口名称或者端口号来进行打开。
打开串口的函数可能返回一个文件描述符或者句柄,用于后续的读写操作。
步骤四:配置串口参数打开串口之后,需要进行串口参数的配置。
这包括波特率、数据位数、校验位和停止位等参数的设置。
一般通过调用相应的函数,将需要设置的参数传递给串口库,以完成参数的配置。
步骤五:读取串口数据配置完串口参数后,你可以开始读取串口数据。
通过调用读取函数,你可以从串口接收缓冲区中获取数据,并进一步进行处理。
读取函数可能会阻塞程序执行,直到有数据可读取为止。
步骤六:发送串口数据与读取串口数据相对应的是发送串口数据。
通过调用相应的发送函数,你可以将你要发送的数据写入串口发送缓冲区,等待发送。
发送函数可能会阻塞程序执行,直到数据成功发送。
步骤七:关闭串口在程序结束时,需要关闭已经打开的串口。
通过调用相应的函数,可以完成串口的关闭。
用c语言实现串口读写程序一、前言串口通信是一种常见的通信方式,它可以实现单片机与计算机之间的数据传输。
在嵌入式系统中,使用串口通信可以方便地进行调试和数据传输。
本文将介绍如何使用C语言实现串口读写程序。
二、硬件准备在进行串口通信之前,需要准备好相应的硬件设备。
一般来说,需要一台计算机和一个串口转USB模块(或者直接使用带有串口接口的计算机)。
同时,在单片机端也需要连接一个串口模块。
三、C语言编程实现1. 打开串口在C语言中,可以通过打开文件的方式来打开串口设备。
下面是一个示例代码:```#include <stdio.h>#include <fcntl.h>#include <termios.h>int open_serial_port(const char *device_path, int baud_rate) {int fd;struct termios options;fd = open(device_path, O_RDWR | O_NOCTTY | O_NDELAY); if (fd == -1) {perror("open_serial_port: Unable to open device");return -1;}fcntl(fd, F_SETFL, 0);tcgetattr(fd, &options);cfsetispeed(&options, baud_rate);cfsetospeed(&options, baud_rate);options.c_cflag |= (CLOCAL | CREAD);options.c_cflag &= ~PARENB;options.c_cflag &= ~CSTOPB;options.c_cflag &= ~CSIZE;options.c_cflag |= CS8;options.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG);options.c_oflag &= ~OPOST;tcsetattr(fd, TCSANOW, &options);return fd;}```在上述代码中,open_serial_port函数用来打开串口设备,并设置相应的参数。
串口通信之用C语言编写串口程序在当今,流行的编程软件种类繁多,它们编程方便、易于维护,但是在与硬件直接打交道和编制系统软件时却束手无策,于是C语言就有了用武之地。
C语言作为汇编语言与高级语言之间的一种过渡语言,兼有汇编语言的高效和高级语言的方便。
在通讯中,为了保证行运安全可靠,标准的串行口必须具有许多握手信号和状态信息。
这是因为通讯的各个计算机CPU速度不一样(这会导致“错帧”)以及发送机发送数据速度比接收机接收速度快(这会导致“过冲”)。
为解决这个问题,我们采用一个简单的握手信号,即发送机每次仅发送半个字节(低4位)的数据,而另外半个字节(高4位)则用来传送信息。
我们可以对信息位(高4位)进行如下简单的编码:0H:发送的是新的半个字节数据1H:重新发送上次传送错误的数据2H:文件名结束3H:文件结束这样,每当发送机发送一个字节以后,就等待接受机发回送信号,这回送信号就是发送机发送过来的那个字节。
发送机接收到回送信号后,把它与刚发送的字节相比较,如果相同,就发送新的半个字节,否则就重新发送。
新数据与旧数据通过信息位来区分。
下面就是用C 语言编写控制串行口的程序。
#include "dos.h"#include "stdlib.h"#include "stdio.h"#define PORT 0void SendFile(char *fname); /* 发送文件*/void Send(int s); /*发送一个字节*/void SendFileName(char *fname); /*发送文件名*/void ReceiveFile(); /*接收文件*/void GetFileName(char *f); /*接收文件名*/void InitPort(int port,unsigned char para); /*初始化端口*/ void SendPort(int port,char c); /*端口发送*/int ReadPort(int port); /*读端口字节*/int CheckState(int port); /*检查端口状态*/int Receive(int port,int *G); /*接收一个字节*/main(int argc,char *argv[]){if(argc<2){printf("Please input R(receive) or S(sent) parametre:"); exit(1);}InitPort(PORT,231);if(*argv[1]==''''S'''') /*检查选择的有效性*/SendFile(argv[2]);else if(*argv[1]==''''R'''')ReceiveFile();else{printf("Error parament.Please input again.");exit(1);}}void SendFile(char *fname){FILE *fp;int ch,s;if((fp=fopen(fname,"rb"))==NULL){printf("Can''''t open the file.\n");exit(1);}SendFileName(fname);do{ch=(int)getc(fp);if(ferror(fp)){printf("Error reading file.\n");break;}s=ch%16; /*取文件中一个字节的低4位*/ Send(s);s=ch/16; /*取文件中一个字节的高4位*/ Send(s);}while(!feof(fp));s=46; /*发送文件结束信息*/Send(s);Send(s);fclose(fp);}void Send(s)int s;{int G;SendPort(PORT,s);G=ReadPort(PORT); /*等待握手信号*/ if(s!=G)s=s+16;do{SendPort(PORT,s);G=ReadPort(PORT);/*等待握手信号*/}while(s!=G);}void SendFileName(fname)char *fname;{int s,ch;printf("Now transmit the file.Please wait..."); while(*fname){ch=(int)fname++;s=ch%16; /*取文件名中一个字节的低4位*/ Send(s);s=ch/16;Send(s); /*取文件名中一个字节的低4位*/}s=32; /*发送文件名结束标志*/Send(s);Send(s);}void ReceiveFile(){FILE *fp;char ch;int G1,G2,G3;char fname[15];GetFileName(fname);printf("Receiving file %s.\n",fname); remove(fname);if((fp=fopen(fname,"wb"))==NULL){printf("Can''''t open output file.\n");exit(1);}/*循环为检测每次接受的数据是否为新数据,如果不是,*//*则用此次接收的数据覆盖上次接收的数据*/G1=ReadPort(PORT);G2=Receive(PORT,&G1);do{G3=Receive(PORT,&G2);ch=(char)(G1%16+G2*16);/*恢复分开的数据,组合高4位和低4位*/putc(ch,fp);if(ferror(fp)){printf("\nError writing file.");exit(1);}G2=Receive(PORT,&G3);G1=G3;}while(G1/16!=48);printf("\nTransmit finished.");fclose(fp);}int Receive(port,G)int port,*G;{int GM;SendPort(port,*G);GM=ReadPort(port);if(GM/16==0)return GM;else if(GM/16==1){do{*G=GM;SendPort(port,GM);GM=ReadPort(port);}while(GM/16==1);}return GM;}void GetFileName(char *f){int G1,G2,G3;char ch;G1=ReadPort(PORT);G2=ReadPort(PORT);do{G3=Receive(PORT,&G3);ch=(char)(G1%16+G2/16);*f=ch;*f++;G2=Receive(PORT,&G3);G1=G3;}while(G1/16!=32);printf("File name transmit finished.\n"); }void InitPort(port,para)int port;unsigned char para;{union REGS reg;reg.x.dx=port;reg.h.ah=0;reg.h.al=para;int86(0x14,?,?);}void SendPort(port,c)int port;char c;{union REGS reg;reg.x.dx=port;reg.h.al=c;reg.h.ah=1;int86(0x14,?,?);if(reg.h.ah&128){printf("\nSend mistakes!");exit(1);}}int ReadPort(port)int port;{union REGS reg;while(!(CheckState(port)&256)){if(kbhit()){/*如端口长期无数据可人为终止等待*/ printf("Press any key to exit.");getch();exit(1);}}reg.x.dx=port;reg.h.ah=2;int86(0x14,?,?);if(reg.h.ah&128){printf("\nRead mistake!");exit(1);}return reg.h.al;}int CheckState(port)int port;{union REGS reg;reg.x.dx=port;reg.h.ah=3;int86(0x14,?,?);return reg.x.ax;}在当今,流行的编程软件种类繁多,它们编程方便、易于维护,但是在与硬件直接打交道和编制系统软件时却束手无策,于是C语言就有了用武之地。
串口通信c 语言程序#include "c8051f020.h"#include <intrins.h>/********函数声明********/void SYSCLK_Init (void); //*单片机时钟初始化void PORT_Init (void); // 初始化IO口void Uart0_Init(void); //串口0 初始化//**** 变量及I/O口定义*********unsigned char data aaa[5]_at_ 0x30;unsigned char data BBB[5]_at_ 0x40;unsigned char data EEE _at_ 0x50;int k;//*********单片机时钟初始化***********//void SYSCLK_Init (void){int i; // delay counterOSCXCN = 0x65; // start external oscillator with// 18.432MHz crystalfor (i=0; i < 256; i++) ; // Wait for osc. to start upwhile (!(OSCXCN & 0x80)) ; // Wait for crystal osc. to settleOSCICN = 0x88; // select external oscillator as SYSCLK// source and enable missing clock// detector}//***************//初始化IO口*************************//void PORT_Init (void){XBR0 = 0x07; // Enable SMBus, SPI0, and UART0XBR1= 0x02; //交叉开关允许XBR2 = 0x40; // Enable crossbar and weak P74OUT=0x04 ;pull-ups}//***************串口0 初始化**************//void Uart0_Init(void){TMOD = 0x20; // TMOD: 定时器1, 模式2, 8位重载SCON0 = 0x40; // SCON0: 模式1, 8位UART,PCON = 0x80; // SMOD00 = 1TH1= 0x70;TL1= 0x70;CKCON=0x10; // T1时钟为系统时钟12分频TR1=1; //计数器1开始计时, 打开串口Uart0 }//************主程序****************************//void main (void){WDTCN = 0xde;WDTCN = 0xad; //关看门狗SYSCLK_Init (); //初始化时钟PORT_Init (); //初始化IO口Uart0_Init(); //串口0 初始化aaa[0]=0x011;aaa[1]=0x022;aaa[2]=0x033;aaa[3]=0x044;aaa[4]=0x055;aaa[5]=0x066;P5=0x40;//485发送for(k=0;k<=5;k++){SBUF0=aaa[k];while(TI0!=1);TI0=0;}P5=0x0;//485接收REN0=1;for(k=0;k<=5;k++){while(RI0!=1);RI0=0;BBB[k]=SBUF0 ;}while(1) ;}。
VC实现串口通信项目源码以下是一个简单的VC实现串口通信项目的源码:```cpp#include <windows.h>#include <iostream>using namespace std;//定义串口类class SerialPortpublic:SerialPort(const char* portName)//打开串口hSerial = CreateFile(portName, GENERIC_READ , GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);//检查串口是否成功打开if (hSerial == INVALID_HANDLE_VALUE)if (GetLastError( == ERROR_FILE_NOT_FOUND)cerr << "Error: " << portName << " 不存在" << endl;} elsecerr << "Error: 打开串口失败" << endl;}}//配置串口DCB dcbSerialParams = {0};dcbSerialParams.DCBlength = sizeof(dcbSerialParams);cerr << "Error: 获取串口配置信息失败" << endl;}dcbSerialParams.BaudRate = CBR_9600; // 设置波特率为9600dcbSerialParams.ByteSize = 8; // 设置数据位为8位dcbSerialParams.StopBits = ONESTOPBIT; // 设置停止位为1个dcbSerialParams.Parity = NOPARITY; // 设置奇偶校验方式为无校验cerr << "Error: 设置串口配置信息失败" << endl;}//配置读写操作的超时时间cerr << "Error: 设置串口超时时间失败" << endl;}}//读取串口数据int readData(char* buffer, int bufferSize)DWORD bytesRead;if (!ReadFile(hSerial, buffer, bufferSize, &bytesRead, NULL)) cerr << "Error: 读取串口数据失败" << endl;return -1;}return bytesRead;}//发送串口数据bool writeData(const char* buffer, int bufferSize)DWORD bytesWritten;if (!WriteFile(hSerial, buffer, bufferSize, &bytesWritten, NULL))cerr << "Error: 发送串口数据失败" << endl;return false;}return true;}//关闭串口void closePorCloseHandle(hSerial);}private:HANDLE hSerial;};int mai// 创建SerialPort对象,指定串口名称(根据实际情况修改)SerialPort serialPort("COM1");//读取串口数据char buffer[256];int bytesRead = serialPort.readData(buffer, sizeof(buffer)); if (bytesRead == -1)cerr << "Error: 读取串口数据失败" << endl;} elsecout << "读取到的数据: " << buffer << endl;}//发送串口数据const char* data = "Hello, Serial Port!";if (!serialPort.writeData(data, strlen(data)))cerr << "Error: 发送串口数据失败" << endl;} elsecout << "成功发送串口数据" << endl;}//关闭串口serialPort.closePort(;return 0;```上述代码实现了一个简单的串口通信项目,包括串口初始化、读取串口数据、发送串口数据和关闭串口的功能。
基于C语言的RS232串口通信的设计RS232是一种常见的串行通信接口,广泛用于计算机与外部设备之间的数据传输。
RS232串口通信涉及到多个方面的设计,包括串口参数设置、数据的发送与接收等。
本文将以C语言为基础,介绍如何设计一个基于RS232串口通信的程序。
接下来,我们需要编写数据发送和接收的函数。
数据的发送包括两个步骤:打开串口和发送数据。
首先,我们需要打开串口,并设置好相应的参数。
在C语言中,可以通过打开文件的方式来打开串口设备文件。
例如,可以使用`fopen(`函数打开串口设备文件:```FILE* serial_port = fopen("/dev/ttyS0", "w");if(serial_port == NULL)printf("Failed to open the serial port.\n");return -1;```然后,我们可以使用`fprintf(`函数将数据写入串口设备文件,实现数据的发送:```fprintf(serial_port, "Hello, RS232!\n");```数据的接收与数据的发送类似,也包括两个步骤:打开串口和接收数据。
我们仍然可以使用`fopen(`函数打开串口设备文件,并设置好相应的参数。
然后,可以使用`fgets(`函数从串口设备文件中读取数据:```char buffer[1024];fgets(buffer, sizeof(buffer), serial_port);printf("Received Data: %s", buffer);```需要注意的是,当数据到达串口时,我们需要设置好超时时间,以免数据接收阻塞程序执行。
在C语言中,可以通过设置串口设备文件的文件描述符来设置超时时间。
最后,我们需要在程序中循环调用数据发送和接收的函数,实现数据的循环传输。
1.添加C++类 CCESeries下面是CCESeries.h 文件的内容:#pragma once//定义串口接收数据函数类型//这是一个回调函数,执行该函数时,表示串口接收到了数据。
typedef void (CALLBACK* ONSERIESREAD)(void * pOwner /*父对象指针*/ ,BYTE* buf /*接收到的缓冲区*/,DWORD dwBufLen /*接收到的缓冲区长度*/);class CCESeries{public:CCESeries(void); //CCESeries类的构造函数~CCESeries(void); //CCESeries类的析构函数public://打开串口BOOL OpenPort(void* pOwner,/*指向父指针*/UINT portNo = 1, /*串口号*/UINT baud = 9600, /*波特率*/UINT parity = NOPARITY, /*奇偶校验*/UINT databits = 8, /*数据位*/UINT stopbits = 0 /*停止位*/);//关闭串口void ClosePort();//同步写入数据BOOL WriteSyncPort(const BYTE*buf , DWORD bufLen);//设置串口读取、写入超时BOOL SetSeriesTimeouts(COMMTIMEOUTS CommTimeOuts);//得到串口是否打开BOOL GetComOpened();private://串口读线程函数,该函数被定义成私有静态。
static DWORD WINAPI ReadThreadFunc(LPVOID lparam);private://关闭读线程,当使用完串口后,便调用这个函数退出串口数据接收线程。
void CloseReadThread();private://已打开的串口句柄// HANDLE m_hComm;//读线程句柄HANDLE m_hReadThread;//读线程ID标识DWORD m_dwReadThreadID;//读线程退出事件HANDLE m_hReadCloseEvent;BOOL m_bOpened; //串口是否打开void * m_pOwner; //指定父对象指针public:ONSERIESREAD m_OnSeriesRead; //串口读取回调函数HANDLE m_hComm;};2.下面是CCESeries.cpp文件的内容:#include "StdAfx.h"#include "CESeries.h"//构造函数,将m_hComm初始化为无效的句柄CCESeries::CCESeries(){//初始化内部变量m_hComm = INVALID_HANDLE_VALUE;m_OnSeriesRead = NULL;m_bOpened = 0;}//析构函数,检测如果串口是打开的,则关闭串口CCESeries::~CCESeries(){if (m_bOpened){//关闭串口ClosePort();}}//串口读线程函数,该线程用于异步接收串口数据。
用C语言编写串口程序在当今,流行的编程软件种类繁多,它们编程方便、易于维护,但是在与硬件直接打交道和编制系统软件时却束手无策,于是C语言就有了用武之地。
C语言作为汇编语言与高级语言之间的一种过渡语言,兼有汇编语言的高效和高级语言的方便。
在通讯中,为了保证行运安全可靠,标准的串行口必须具有许多握手信号和状态信息。
这是因为通讯的各个计算机CPU速度不一样(这会导致“错帧”)以及发送机发送数据速度比接收机接收速度快(这会导致“过冲”)。
为解决这个问题,我们采用一个简单的握手信号,即发送机每次仅发送半个字节(低4位)的数据,而另外半个字节(高4位)则用来传送信息。
我们可以对信息位(高4位)进行如下简单的编码:0H:发送的是新的半个字节数据1H:重新发送上次传送错误的数据2H:文件名结束3H:文件结束这样,每当发送机发送一个字节以后,就等待接受机发回送信号,这回送信号就是发送机发送过来的那个字节。
发送机接收到回送信号后,把它与刚发送的字节相比较,如果相同,就发送新的半个字节,否则就重新发送。
新数据与旧数据通过信息位来区分。
下面就是用C语言编写控制串行口的程序。
#include "dos.h"#include "stdlib.h"#include "stdio.h"#define PORT 0void SendFile(char *fname); /* 发送文件*/void Send(int s); /*发送一个字节*/void SendFileName(char *fname); /*发送文件名*/void ReceiveFile(); /*接收文件*/void GetFileName(char *f); /*接收文件名*/void InitPort(int port,unsigned char para); /*初始化端口*/void SendPort(int port,char c); /*端口发送*/int ReadPort(int port); /*读端口字节*/int CheckState(int port); /*检查端口状态*/int Receive(int port,int *G); /*接收一个字节*/main(int argc,char *argv[]){if(argc<2){printf("Please input R(receive) or S(sent) parametre:");exit(1);}InitPort(PORT,231);if(*argv[1]==''''S'''') /*检查选择的有效性*/ SendFile(argv[2]);else if(*argv[1]==''''R'''')ReceiveFile();else{printf("Error parament.Please input again."); exit(1);}}void SendFile(char *fname){FILE *fp;int ch,s;if((fp=fopen(fname,"rb"))==NULL){printf("Can''''t open the file.\n");exit(1);}SendFileName(fname);do{ch=(int)getc(fp);if(ferror(fp)){printf("Error reading file.\n");break;}s=ch%16; /*取文件中一个字节的低4位*/ Send(s);s=ch/16; /*取文件中一个字节的高4位*/ Send(s);}while(!feof(fp));s=46; /*发送文件结束信息*/Send(s);Send(s);fclose(fp);}void Send(s)int s;{int G;SendPort(PORT,s);G=ReadPort(PORT); /*等待握手信号*/if(s!=G)s=s+16;do{SendPort(PORT,s);G=ReadPort(PORT);/*等待握手信号*/}while(s!=G);}void SendFileName(fname)char *fname;{int s,ch;printf("Now transmit the file.Please wait...");while(*fname){ch=(int)fname++;s=ch%16; /*取文件名中一个字节的低4位*/Send(s);s=ch/16;Send(s); /*取文件名中一个字节的低4位*/}s=32; /*发送文件名结束标志*/Send(s);Send(s);}void ReceiveFile(){FILE *fp;char ch;int G1,G2,G3;char fname[15];GetFileName(fname);printf("Receiving file %s.\n",fname);remove(fname);if((fp=fopen(fname,"wb"))==NULL){printf("Can''''t open output file.\n");exit(1);}/*循环为检测每次接受的数据是否为新数据,如果不是,*//*则用此次接收的数据覆盖上次接收的数据*/G1=ReadPort(PORT);G2=Receive(PORT,&G1);do{G3=Receive(PORT,&G2);ch=(char)(G1%16+G2*16);/*恢复分开的数据,组合高4位和低4位*/ putc(ch,fp);if(ferror(fp)){printf("\nError writing file.");exit(1);}G2=Receive(PORT,&G3);G1=G3;}while(G1/16!=48);printf("\nTransmit finished.");fclose(fp);}int Receive(port,G)int port,*G;{int GM;SendPort(port,*G);GM=ReadPort(port);if(GM/16==0)return GM;else if(GM/16==1){do{*G=GM;SendPort(port,GM);GM=ReadPort(port);}while(GM/16==1);}return GM;}void GetFileName(char *f){int G1,G2,G3;char ch;G1=ReadPort(PORT);G2=ReadPort(PORT);do{G3=Receive(PORT,&G3);ch=(char)(G1%16+G2/16);*f=ch;*f++;G2=Receive(PORT,&G3);G1=G3;}while(G1/16!=32);printf("File name transmit finished.\n"); }void InitPort(port,para)int port;unsigned char para;{union REGS reg;reg.x.dx=port;reg.h.ah=0;reg.h.al=para;int86(0x14,®,®);}void SendPort(port,c)int port;char c;{union REGS reg;reg.x.dx=port;reg.h.al=c;reg.h.ah=1;int86(0x14,®,®);if(reg.h.ah&128){printf("\nSend mistakes!");exit(1);}}int ReadPort(port)int port;{union REGS reg;while(!(CheckState(port)&256)){if(kbhit()){/*如端口长期无数据可人为终止等待*/ printf("Press any key to exit.");getch();exit(1);}}reg.x.dx=port;reg.h.ah=2;int86(0x14,®,®);if(reg.h.ah&128){printf("\nRead mistake!");exit(1);}return reg.h.al;}int CheckState(port)int port;{union REGS reg;reg.x.dx=port;reg.h.ah=3;int86(0x14,®,®);return reg.x.ax;}以上程序可传送各种格式的文件,也有一定的自动纠错能力,但对于异常情况的处理能力比较弱,读者可以自己改进。
由于篇幅限制,对于中断14H的功能、入口参数及返回参数的意义请读者自己查有关资料。
**********************************附录:现在大多数串行口都遵循RS-232标准,以下是最常用的RS-232信号:名称针号含义RTS4Request to send(请求发送)CTS5Clear to send(清除发送)DSR6Data set ready(数据设备准备好)DTR20 Data terminal ready(数据终端准备好)TXD2Transmit data(发送数据)RXD3Receive data(接收数据)GRD7Ground(接地)。