C#串口通信

本文主要是讲述如何通过C# .net1.1平台调用API操作串口,.net2.0已经内置串口操作类!

通常,在C#中实现串口通信,我们有四种方法:

第一:通过MSCOMM控件这是最简单的,最方便的方法。可功能上很难做到控制自如,同时这个控件并不是系统本身所带,所以还得注册。可以访问
http://www.devhood.com/tutorials/tutorial_details.aspx?tutorial_id=320
一个外国人写的教程
第二:微软在.NET新推出了一个串口控件,基于.NET的P/Invoke调用方法实现,详细的可以访问微软网站
Serial Comm
Use P/Invoke to Develop a .NET Base Class Library for Serial Device Communications

http://msdn.microsoft.com/msdnmag/issues/02/10/netserialcomm/
第三:就是用第三方控件啦,可一般都要付费的,不太合实际,何况楼主不喜欢,不作考虑
第四:自己用API写串口通信,这样难度高点,但对于我们来说,可以方便实现自己想要的各种功能。

我们采用第四种方法来实现串口通信,用现成的已经封装好的类库,常见两个串口操作类是JustinIO和SerialStreamReader。介绍JustinIO的使用方法:

打开串口:

函数原型:public void Open()

说明:打开事先设置好的端口

示例:

[csharp] view plain copy
  1. using JustinIO;   
  2.   
  3. static JustinIO.CommPort ss_port = new JustinIO.CommPort();   
  4. ss_port.PortNum = COM1; //端口号   
  5. ss_port.BaudRate = 19200; //串口通信波特率   
  6. ss_port.ByteSize = 8; //数据位   
  7. ss_port.Parity = 0; //奇偶校验   
  8. ss_port.StopBits = 1;//停止位   
  9. ss_port.ReadTimeout = 1000; //读超时   
  10. try   
  11. {   
  12. if (ss_port.Opened)   
  13. {   
  14. ss_port.Close();   
  15. ss_port.Open(); //打开串口   
  16. }   
  17. else   
  18. {   
  19. ss_port.Open();//打开串口   
  20. }   
  21. return true;   
  22. }   
  23. catch(Exception e)   
  24. {   
  25. MessageBox.Show("错误:" + e.Message);   
  26. return false;   
  27. }   



写串口:

函数原型:public void Write(byte[] WriteBytes)

WriteBytes 就是你的写入的字节,注意,字符串要转换成字节数组才能进行通信

示例:

ss_port.Write(Encoding.ASCII.GetBytes("AT+CGMI\r")); //获取手机品牌

读串口:

函数原型:public byte[] Read(int NumBytes)

NumBytes 读入缓存数,注意读取来的是字节数组,要实际应用中要进行字符转换

示例:

string response = Encoding.ASCII.GetString(ss_port.Read(128)); //读取128个字节缓存

关闭串口:

函数原型:ss_port.Close()

示例:

ss_port.Close();

整合代码:

[csharp] view plain copy
  1. using System;   
  2. using System.Runtime.InteropServices;   
  3.   
  4. namespace JustinIO {   
  5. class CommPort {   
  6.   
  7. public int PortNum;   
  8. public int BaudRate;   
  9. public byte ByteSize;   
  10. public byte Parity; // 0-4=no,odd,even,mark,space   
  11. public byte StopBits; // 0,1,2 = 1, 1.5, 2   
  12. public int ReadTimeout;   
  13.   
  14. //comm port win32 file handle   
  15. private int hComm = -1;   
  16.   
  17. public bool Opened = false;   
  18.   
  19. //win32 api constants   
  20. private const uint GENERIC_READ = 0x80000000;   
  21. private const uint GENERIC_WRITE = 0x40000000;   
  22. private const int OPEN_EXISTING = 3;   
  23. private const int INVALID_HANDLE_VALUE = -1;   
  24.   
  25. [StructLayout(LayoutKind.Sequential)]   
  26. public struct DCB {   
  27. //taken from c struct in platform sdk   
  28. public int DCBlength; // sizeof(DCB)   
  29. public int BaudRate; // current baud rate   
  30. /* these are the c struct bit fields, bit twiddle flag to set  
  31. public int fBinary; // binary mode, no EOF check  
  32. public int fParity; // enable parity checking  
  33. public int fOutxCtsFlow; // CTS output flow control  
  34. public int fOutxDsrFlow; // DSR output flow control  
  35. public int fDtrControl; // DTR flow control type  
  36. public int fDsrSensitivity; // DSR sensitivity  
  37. public int fTXContinueOnXoff; // XOFF continues Tx  
  38. public int fOutX; // XON/XOFF out flow control  
  39. public int fInX; // XON/XOFF in flow control  
  40. public int fErrorChar; // enable error replacement  
  41. public int fNull; // enable null stripping  
  42. public int fRtsControl; // RTS flow control  
  43. public int fAbortOnError; // abort on error  
  44. public int fDummy2; // reserved  
  45. */   
  46. public uint flags;   
  47. public ushort wReserved; // not currently used   
  48. public ushort XonLim; // transmit XON threshold   
  49. public ushort XoffLim; // transmit XOFF threshold   
  50. public byte ByteSize; // number of bits/byte, 4-8   
  51. public byte Parity; // 0-4=no,odd,even,mark,space   
  52. public byte StopBits; // 0,1,2 = 1, 1.5, 2   
  53. public char XonChar; // Tx and Rx XON character   
  54. public char XoffChar; // Tx and Rx XOFF character   
  55. public char ErrorChar; // error replacement character   
  56. public char EofChar; // end of input character   
  57. public char EvtChar; // received event character   
  58. public ushort wReserved1; // reserved; do not use   
  59. }   
  60.   
  61. [StructLayout(LayoutKind.Sequential)]   
  62. private struct COMMTIMEOUTS {   
  63. public int ReadIntervalTimeout;   
  64. public int ReadTotalTimeoutMultiplier;   
  65. public int ReadTotalTimeoutConstant;   
  66. public int WriteTotalTimeoutMultiplier;   
  67. public int WriteTotalTimeoutConstant;   
  68. }   
  69.   
  70. [StructLayout(LayoutKind.Sequential)]   
  71. private struct OVERLAPPED {   
  72. public int Internal;   
  73. public int InternalHigh;   
  74. public int Offset;   
  75. public int OffsetHigh;   
  76. public int hEvent;   
  77. }   
  78.   
  79. [DllImport("kernel32.dll")]   
  80. private static extern int CreateFile(   
  81. string lpFileName, // file name   
  82. uint dwDesiredAccess, // access mode   
  83. int dwShareMode, // share mode   
  84. int lpSecurityAttributes, // SD   
  85. int dwCreationDisposition, // how to create   
  86. int dwFlagsAndAttributes, // file attributes   
  87. int hTemplateFile // handle to template file   
  88. );   
  89. [DllImport("kernel32.dll")]   
  90. private static extern bool GetCommState(   
  91. int hFile, // handle to communications device   
  92. ref DCB lpDCB // device-control block   
  93. );   
  94. [DllImport("kernel32.dll")]   
  95. private static extern bool BuildCommDCB(   
  96. string lpDef, // device-control string   
  97. ref DCB lpDCB // device-control block   
  98. );   
  99. [DllImport("kernel32.dll")]   
  100. private static extern bool SetCommState(   
  101. int hFile, // handle to communications device   
  102. ref DCB lpDCB // device-control block   
  103. );   
  104. [DllImport("kernel32.dll")]   
  105. private static extern bool GetCommTimeouts(   
  106. int hFile, // handle to comm device   
  107. ref COMMTIMEOUTS lpCommTimeouts // time-out values   
  108. );   
  109. [DllImport("kernel32.dll")]   
  110. private static extern bool SetCommTimeouts(   
  111. int hFile, // handle to comm device   
  112. ref COMMTIMEOUTS lpCommTimeouts // time-out values   
  113. );   
  114. [DllImport("kernel32.dll")]   
  115. private static extern bool ReadFile(   
  116. int hFile, // handle to file   
  117. byte[] lpBuffer, // data buffer   
  118. int nNumberOfBytesToRead, // number of bytes to read   
  119. ref int lpNumberOfBytesRead, // number of bytes read   
  120. ref OVERLAPPED lpOverlapped // overlapped buffer   
  121. );   
  122. [DllImport("kernel32.dll")]   
  123. private static extern bool WriteFile(   
  124. int hFile, // handle to file   
  125. byte[] lpBuffer, // data buffer   
  126. int nNumberOfBytesToWrite, // number of bytes to write   
  127. ref int lpNumberOfBytesWritten, // number of bytes written   
  128. ref OVERLAPPED lpOverlapped // overlapped buffer   
  129. );   
  130. [DllImport("kernel32.dll")]   
  131. private static extern bool CloseHandle(   
  132. int hObject // handle to object   
  133. );   
  134. [DllImport("kernel32.dll")]   
  135. private static extern uint GetLastError();   
  136.   
  137. public void Open() {   
  138.   
  139. DCB dcbCommPort = new DCB();   
  140. COMMTIMEOUTS ctoCommPort = new COMMTIMEOUTS();   
  141.   
  142.   
  143. // OPEN THE COMM PORT.   
  144.   
  145.   
  146. hComm = CreateFile("COM" + PortNum ,GENERIC_READ | GENERIC_WRITE,0, 0,OPEN_EXISTING,0,0);   
  147.   
  148. // IF THE PORT CANNOT BE OPENED, BAIL OUT.   
  149. if(hComm == INVALID_HANDLE_VALUE) {   
  150. throw(new ApplicationException("Comm Port Can Not Be Opened"));   
  151. }   
  152.   
  153. // SET THE COMM TIMEOUTS.   
  154.   
  155. GetCommTimeouts(hComm,ref ctoCommPort);   
  156. ctoCommPort.ReadTotalTimeoutConstant = ReadTimeout;   
  157. ctoCommPort.ReadTotalTimeoutMultiplier = 0;   
  158. ctoCommPort.WriteTotalTimeoutMultiplier = 0;   
  159. ctoCommPort.WriteTotalTimeoutConstant = 0;   
  160. SetCommTimeouts(hComm,ref ctoCommPort);   
  161.   
  162. // SET BAUD RATE, PARITY, WORD SIZE, AND STOP BITS.   
  163. GetCommState(hComm, ref dcbCommPort);   
  164. dcbCommPort.BaudRate=BaudRate;   
  165. dcbCommPort.flags=0;   
  166. //dcb.fBinary=1;   
  167. dcbCommPort.flags|=1;   
  168. if (Parity>0)   
  169. {   
  170. //dcb.fParity=1   
  171. dcbCommPort.flags|=2;   
  172. }   
  173. dcbCommPort.Parity=Parity;   
  174. dcbCommPort.ByteSize=ByteSize;   
  175. dcbCommPort.StopBits=StopBits;   
  176. if (!SetCommState(hComm, ref dcbCommPort))   
  177. {   
  178. //uint ErrorNum=GetLastError();   
  179. throw(new ApplicationException("Comm Port Can Not Be Opened"));   
  180. }   
  181. //unremark to see if setting took correctly   
  182. //DCB dcbCommPort2 = new DCB();   
  183. //GetCommState(hComm, ref dcbCommPort2);   
  184. Opened = true;   
  185.   
  186. }   
  187.   
  188. public void Close() {   
  189. if (hComm!=INVALID_HANDLE_VALUE) {   
  190. CloseHandle(hComm);   
  191. }   
  192. }   
  193. public byte[] Read(int NumBytes) {   
  194. byte[] BufBytes;   
  195. byte[] OutBytes;   
  196. BufBytes = new byte[NumBytes];   
  197. if (hComm!=INVALID_HANDLE_VALUE) {   
  198. OVERLAPPED ovlCommPort = new OVERLAPPED();   
  199. int BytesRead=0;   
  200. ReadFile(hComm,BufBytes,NumBytes,ref BytesRead,ref ovlCommPort);   
  201. OutBytes = new byte[BytesRead];   
  202. Array.Copy(BufBytes,OutBytes,BytesRead);   
  203. }   
  204. else {   
  205. throw(new ApplicationException("Comm Port Not Open"));   
  206. }   
  207. return OutBytes;   
  208. }   
  209.   
  210. public void Write(byte[] WriteBytes) {   
  211. if (hComm!=INVALID_HANDLE_VALUE) {   
  212. OVERLAPPED ovlCommPort = new OVERLAPPED();   
  213. int BytesWritten = 0;   
  214. WriteFile(hComm,WriteBytes,WriteBytes.Length,ref BytesWritten,ref ovlCommPort);   
  215. }   
  216. else {   
  217. throw(new ApplicationException("Comm Port Not Open"));   
  218. }   
  219. }   
  220. }   
  221.   
  222. }   
  223. }   



 

C#中使用SerialPort类实现串口通信

在.NET Framework 2.0中提供了SerialPort类,该类主要实现串口数据通信等。本文章将本人在学习过程中从网络上搜集到的相关信息写出来供大家参考。

下面主要介绍该类的主要属性(表1)和方法(表.2)。

如果需要了解更多的信息请登录http://msdn.microsoft.com/zh-cn/library/system.io.ports.serialport(VS.80).aspx查看。

相关文章

《使用System.IO.Ports读取COM口数据》

http://www.devasp.net/net/articles/display/727.html

表1                                                      SerialPort类的常用属性

名  称

说  明

BaseStream

获取 SerialPort 对象的基础 Stream 对象

BaudRate

获取或设置串行波特率

BreakState

获取或设置中断信号状态

BytesToRead

获取接收缓冲区中数据的字节数

BytesToWrite

获取发送缓冲区中数据的字节数

CDHolding

获取端口的载波检测行的状态

CtsHolding

获取“可以发送”行的状态

DataBits

获取或设置每个字节的标准数据位长度

DiscardNull

获取或设置一个值,该值指示 Null 字节在端口和接收缓冲区之间传输时是否被忽略

DsrHolding

获取数据设置就绪 (DSR) 信号的状态

DtrEnable

获取或设置一个值,该值在串行通信过程中启用数据终端就绪 (DTR) 信号

Encoding

获取或设置传输前后文本转换的字节编码

Handshake

获取或设置串行端口数据传输的握手协议

IsOpen

获取一个值,该值指示 SerialPort 对象的打开或关闭状态

NewLine

获取或设置用于解释 ReadLine( )和WriteLine( )方法调用结束的值

Parity

获取或设置奇偶校验检查协议

续表

名  称

说  明

ParityReplace

获取或设置一个字节,该字节在发生奇偶校验错误时替换数据流中的无效字节

PortName

获取或设置通信端口,包括但不限于所有可用的 COM 端口

ReadBufferSize

获取或设置 SerialPort 输入缓冲区的大小

ReadTimeout

获取或设置读取操作未完成时发生超时之前的毫秒数

ReceivedBytesThreshold

获取或设置 DataReceived 事件发生前内部输入缓冲区中的字节数

RtsEnable

获取或设置一个值,该值指示在串行通信中是否启用请求发送 (RTS) 信号

StopBits

获取或设置每个字节的标准停止位数

WriteBufferSize

获取或设置串行端口输出缓冲区的大小

WriteTimeout

获取或设置写入操作未完成时发生超时之前的毫秒数

表2                                                     SerialPort类的常用方法

方 法 名 称

说  明

Close

关闭端口连接,将 IsOpen 属性设置为False,并释放内部 Stream 对象

Open

打开一个新的串行端口连接

Read

从 SerialPort 输入缓冲区中读取

ReadByte

从 SerialPort 输入缓冲区中同步读取一个字节

ReadChar

从 SerialPort 输入缓冲区中同步读取一个字符

ReadLine

一直读取到输入缓冲区中的 NewLine 值

ReadTo

一直读取到输入缓冲区中指定 value 的字符串

Write

已重载。将数据写入串行端口输出缓冲区

WriteLine

将指定的字符串和 NewLine 值写入输出缓冲区

使用SerialPort类的方法:

方法一:

首先要添加

using System.IO;
using System.IO.Ports;

1...在类的内部定义SerialPort com;

2...打开串口

            com = new SerialPort();
            com.BaudRate = 115200;
            com.PortName = "COM1";
            com.DataBits = 8;
            com.Open();//打开串口

3...发送数据

            Byte[] TxData ={1,2,3,4,5,6,7,8 };
            com.Write(TxData, 0, 8);

4...接收数据

     4.1使用事件接收

     this.com.DataReceived += new System.IO.Ports.SerialDataReceivedEventHandler(this.OnDataReceived);

private void OnDataReceived(object sender, SerialDataReceivedEventArgs e)

    4.2使用线程接收

     接收数据启动一个线程,使其接收。

在类的内部定义

        Thread _readThread;
        bool _keepReading;

打开串口后启动线程

            _keepReading = true;
            _readThread = new Thread(ReadPort);
            _readThread.Start();

线程函数

[csharp] view plain copy
  1. private void ReadPort()  
  2. {  
  3.     while (_keepReading)  
  4.     {  
  5.         if (com.IsOpen)  
  6.         {  
  7.             byte[] readBuffer = new byte[com.ReadBufferSize + 1];  
  8.             try  
  9.             {  
  10.                 // If there are bytes available on the serial port,  
  11.                 // Read returns up to "count" bytes, but will not block (wait)  
  12.                 // for the remaining bytes. If there are no bytes available  
  13.                 // on the serial port, Read will block until at least one byte  
  14.                 // is available on the port, up until the ReadTimeout milliseconds  
  15.                 // have elapsed, at which time a TimeoutException will be thrown.  
  16.                 int count = com.Read(readBuffer, 0, com.ReadBufferSize);  
  17.                 String SerialIn = System.Text.Encoding.ASCII.GetString(readBuffer, 0, count);  
  18.                 if (count != 0)  
  19.                     //byteToHexStr(readBuffer);  
  20.                     ThreadFunction(byteToHexStr(readBuffer,count));  
  21.             }  
  22.             catch (TimeoutException) { }  
  23.         }  
  24.         else  
  25.         {  
  26.             TimeSpan waitTime = new TimeSpan(0, 0, 0, 0, 50);  
  27.             Thread.Sleep(waitTime);  
  28.         }  
  29.     }  
  30. }  


方法二:使用C#自带的SerialPor控件。

1...在“工具箱”的“组件”中选择SerialPor控件添加。

2...设置串口并打开

serialPort1.PortName = "COM1";

serialPort1.BaudRate = 9600;

serialPort1.Open();

3...写入数据可以使用Write或者下面的函数

serialPort1.WriteLine(str);

4...添加数据接收的事件

private void serialPort1_DataReceived(object sender, SerialDataReceivedEventArgs e)

使用中的一些常见问题

C#中SerialPort类中DataReceived事件GUI实时处理方法(来自wanglei_wan@yahoo.com.cn 的看法)
MSDN:从 SerialPort 对象接收数据时,将在辅助线程上引发 DataReceived 事件。由于此事件在辅助线程而非主线程上引发,因此尝试修改主线程中的一些元素(如 UI 元素)时会引发线程异常。如果有必要修改主 Form 或 Control 中的元素,必须使用 Invoke 回发更改请求,这将在正确的线程上执行.进而要想将辅助线程中所读到的数据显示到主线程的Form控件上时,只有通过Invoke方法来实现
下面是代码实例:

[csharp] view plain copy
  1. private void serialPort1_DataReceived(object sender, SerialDataReceivedEventArgs e)  
  2. {  
  3.    int SDateTemp = this.serialPort1.ReadByte();  
  4.    //读取串口中一个字节的数据  
  5.    this.tB_ReceiveDate.Invoke(     
  6.     //在拥有此控件的基础窗口句柄的线程上执行委托Invoke(Delegate)  
  7.     //即在textBox_ReceiveDate控件的父窗口form中执行委托.  
  8.     new MethodInvoker(              
  9.     /*表示一个委托,该委托可执行托管代码中声明为 void 且不接受任何参数的任何方法。 在对控件的 Invoke    方法进行调用时或需要一个简单委托又不想自己定义时可以使用该委托。*/  
  10.     delegate{                   
  11.     /*匿名方法,C#2.0的新功能,这是一种允许程序员将一段完整代码区块当成参数传递的程序代码编写技术,通过此种方法可  以直接使用委托来设计事件响应程序以下就是你要在主线程上实现的功能但是有一点要注意,这里不适宜处理过多的方法,因为C#消息机制是消息流水线响应机制,如果这里在主线程上处理语句的时间过长会导致主UI线程阻塞,停止响应或响应不顺畅,这时你的主form界面会延迟或卡死      */                     
  12.     this.tB_ReceiveDate.AppendText(SDateTemp.ToString());//输出到主窗口文本控件  
  13.     this.tB_ReceiveDate.Text += " ";}  
  14.     )  
  15.     );  
  16. }  

如何知道当前电脑有哪个串口

在窗体上添加一个comboBox控件。

然后使用comboBox1.Items.AddRange(System.IO.Ports.SerialPort.GetPortNames());  或者

string[] portList = System.IO.Ports.SerialPort.GetPortNames();
            for (int i = 0; i < portList.Length; ++i)
            {
                string name = portList[i];
                comboBox1.Items.Add(name);
            }

具体请参考http://msdn.microsoft.com/zh-tw/library/system.io.ports.serialport.getportnames.aspx

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
更新了下。。。 ************************************************************************ 函数名称:SP_InitComm 函数功能:初始化异步窗口操作模块 参数.一:lpComName In/Out:In 可空:N 意思:速率 返回值类型:要读取的串口号码 意思:是否初始化成功 备注: *************************************************************************/ extern "C" __declspec(dllexport) BOOL SP_InitComm(LPCTSTR lpComName); //初始化串口操作 /************************************************************************ 函数名称:SP_CloseComm 函数功能:关闭串口 返回值类型:无 意思: 备注:不在使用串口的时候可以关闭它。不然会造成下次使用出现问题 *************************************************************************/ extern "C" __declspec(dllexport) void SP_CloseComm(); /************************************************************************ 函数名称:SP_ReadComm 函数功能:读取串口数据 参数.一:tszDest In/Out:In/Out 可空:N 意思:要保存到的缓冲区,提供一个缓冲区 参数.二:dwWriteLen In/Out:In/Out 可空:N 意思:读取的预计大小,实际写出大小 返回值类型:逻辑型 意思:是否成功读取 备注: *************************************************************************/ extern "C" __declspec(dllexport) BOOL SP_ReadComm(TCHAR *pTszBuff,DWORD *pdwLen); /************************************************************************ 函数名称:SP_WriteComm 函数功能:异步发送串口数据 参数.一:tszData In/Out:In 可空:N 意思:要发送的数据字符串指针 参数.二:dwWriteLen In/Out:Out 可空:N 意思:实际发送的长度,可空 返回值类型:逻辑型 意思:是否发送成功 备注:如果错误你需要获取错误码来知道为什么错误 *************************************************************************/ extern "C" __declspec(dllexport) BOOL SP_WriteComm(TCHAR *pTszBuff,DWORD *pdwLen); /************************************************************************ 函数名称:SP_GetComCount 函数功能:获取当前系统有多少COM端口 参数.一:ComName In/Out:Out 可空:N 意思:Com字符串名称 返回值类型:int 意思:当前COM端口个数 备注:ComName会返回搜索到的最后一个COM端口名称,然后你可以根据序号来操作 *************************************************************************/ extern "C" __declspec(dllexport) DWORD SP_GetComCount(TCHAR *tszComNumber); //获取错误码 extern "C" __declspec(dllexport) DWORD SP_COM_GetLastError();
重叠IO模型之OverLapped完成例程模型WSACompletionRoutineServer VS2010 基础入门 客户端与服务器端 客户端向服务器端发送数据 可接收多个客户端 #include #include #pragma comment (lib, "ws2_32.lib") #define PORT 8088 #define MSG_SIZE 1024 SOCKET g_sConnect; bool g_bConnect = false; typedef struct { WSAOVERLAPPED overLap; WSABUF wsaBuf; char chMsg[MSG_SIZE]; DWORD nRecvNum; DWORD nFlags; SOCKET sClient; }PRE_IO_OPERATION_DATA, *LP_PER_IO_OPERATION_DATA; void CALLBACK CompletionRoutine(DWORD dwError, DWORD dwTrans, LPWSAOVERLAPPED lpOverlap, DWORD nFlags); DWORD WINAPI workThread(LPVOID lp) { LP_PER_IO_OPERATION_DATA lpData; while(TRUE) { if (g_bConnect) // 有新的连接 { // 为lpData分配空间并初始化 lpData = (LP_PER_IO_OPERATION_DATA)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(PRE_IO_OPERATION_DATA)); lpData->wsaBuf.len = MSG_SIZE; lpData->wsaBuf.buf = lpData->chMsg; lpData->sClient = g_sConnect; WSARecv(lpData->sClient, &lpData->wsaBuf, 1, &lpData->nRecvNum, &lpData->nFlags, &lpData->overLap, CompletionRoutine); g_bConnect = false; // 处理完毕 } SleepEx(1000, TRUE); } return 0; } // 系统在WSARecv收到信息后,自动调用此函数,并传入参数--回调函数 void CALLBACK CompletionRoutine(DWORD dwError, DWORD dwTrans, LPWSAOVERLAPPED lpOverlap, DWORD nFlags) { LP_PER_IO_OPERATION_DATA lpData = (LP_PER_IO_OPERATION_DATA)lpOverlap; if (0 != dwError) // 接收失败 { printf("Socket %d Close!\n", lpData->sClient); closesocket(lpData->sClient); HeapFree(GetProcessHeap(), 0, lpData); } else // 接收成功 { lpData->chMsg[dwTrans] = '\0'; send(lpData->sClient, lpData->chMsg, dwTrans, 0); printf("Socket:%d MSG: %s \n", lpData->sClient, lpData->chMsg); memset(&lpData->overLap, 0, sizeof(WSAOVERLAPPED)); lpData->wsaBuf.len = MSG_SIZE; lpData->wsaBuf.buf = lpData->chMsg; // 继续接收来自客户端的数据 实现 WSARecv与CompletionRoutine循环 WSARecv(lpData->sClient, &lpData->wsaBuf,1, &lpData->nRecvNum, &lpData->nFlags, &lpData->overLap, CompletionRoutine); } } int main() { WSADATA wsaData; WSAStartup(0x0202, &wsaData); SOCKET sListen; sListen = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); sockaddr_in addrListen; addrListen.sin_family = AF_INET; addrListen.sin_port = htons(PORT); addrListen.sin_addr.S_un.S_addr = htonl(ADDR_ANY); int nErrorCode = 0; nErrorCode = bind(sListen, (sockaddr*)&addrListen, sizeof(sockaddr)); nErrorCode = listen(sListen, 5); DWORD nThreadID; CreateThread(NULL, 0, workThread, NULL, 0, &nThreadID); sockaddr_in addrConnect; int nAddrLen = sizeof(sockaddr_in); printf("Server Started!\n"); while(TRUE) { g_sConnect= accept(sListen, (sockaddr*)&addrConnect, &nAddrLen); if (INVALID_SOCKET == g_sConnect) { return -1; } g_bConnect = true; // 连接成功 printf("Accept Client :%s -- PORT:%d\n", inet_ntoa(addrConnect.sin_addr), htons(addrConnect.sin_port)); } return 0; }

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值