目录
原创性申明" >原创性申明
引言" >引言
目标
代码
服务器端
客户端
首页 后端开发 C#.Net教程 C#高性能Socket服务器SocketAsyncEventArgs的实现(IOCP)

C#高性能Socket服务器SocketAsyncEventArgs的实现(IOCP)

Feb 27, 2017 am 10:59 AM

原创性申明

本文作者:小竹zz  博客地址:http://www.php.cn/转载请注明出处

引言

我一直在探寻一个高性能的Socket客户端代码。以前,我使用Socket类写了一些基于传统异步编程模型的代码(BeginSend、BeginReceive,等等)也看过很多博客的知识,在linux中有poll和epoll来实现,在windows下面
微软MSDN中也提供了SocketAsyncEventArgs这个类来实现IOCP 地址:http://www.php.cn/
NET Framework中的APM也称为Begin/End模式。这是因为会调用Begin方法来启动异步操作,然后返回一个IAsyncResult 对象。可以选择将一个代理作为参数提供给Begin方法,异步操作完成时会调用该方法。或者,一个线程可以等待 IAsyncResult.AsyncWaitHandle。当回调被调用或发出等待信号时,就会调用End方法来获取异步操作的结果。这种模式很灵活,使用相对简单,在 .NET Framework 中非常常见。
但是,您必须注意,如果进行大量异步套接字操作,是要付出代价的。针对每次操作,都必须创建一个IAsyncResult对象,而且该对象不能被重复使用。由于大量使用对象分配和垃圾收集,这会影响性能。为了解决这个问题,新版本提供了另一个使用套接字上执行异步I/O的方法模式。这种新模式并不要求为每个套接字操作分配操作上下文对象。

代码下载:http://www.php.cn/ 这里的代码优化了的

目标

在上面微软提供的例子我觉得不是很完整,没有具体一个流程,只是受到客户端消息后发送相同内容给客户端,初学者不容易看懂流程,因为我花了一天的时间来实现一个功能齐全的IOCP服务器,

效果如下


代码

首先是ICOPServer.cs 这个类是IOCP服务器的核心类,目前这个类是网络上比较全的代码,MSDN上面的例子都没有我的全

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

136

137

138

139

140

141

142

143

144

145

146

147

148

149

150

151

152

153

154

155

156

157

158

159

160

161

162

163

164

165

166

167

168

169

170

171

172

173

174

175

176

177

178

179

180

181

182

183

184

185

186

187

188

189

190

191

192

193

194

195

196

197

198

199

200

201

202

203

204

205

206

207

208

209

210

211

212

213

214

215

216

217

218

219

220

221

222

223

224

225

226

227

228

229

230

231

232

233

234

235

236

237

238

239

240

241

242

243

244

245

246

247

248

249

250

251

252

253

254

255

256

257

258

259

260

261

262

263

264

265

266

267

268

269

270

271

272

273

274

275

276

277

278

279

280

281

282

283

284

285

286

287

288

289

290

291

292

293

294

295

296

297

298

299

300

301

302

303

304

305

306

307

308

309

310

311

312

313

314

315

316

317

318

319

320

321

322

323

324

325

326

327

328

329

330

331

332

333

334

335

336

337

338

339

340

341

342

343

344

345

346

347

348

349

350

351

352

353

354

355

356

357

358

359

360

361

362

363

364

365

366

367

368

369

370

371

372

373

374

375

376

377

378

379

380

381

382

383

384

385

386

387

388

389

390

391

392

393

394

395

396

397

398

399

400

401

402

403

404

405

406

407

408

409

410

411

412

413

414

415

416

417

418

419

420

421

422

423

424

425

426

427

428

429

430

431

432

433

434

435

436

437

438

439

440

441

442

443

444

445

446

447

448

449

450

451

452

453

454

455

456

457

458

459

460

461

462

463

464

465

466

467

468

469

470

471

472

473

474

475

476

477

478

479

480

481

482

483

484

485

486

487

488

489

490

491

492

493

494

495

496

497

498

499

500

501

502

503

504

505

506

507

508

509

510

511

512

513

514

515

516

517

518

519

520

521

522

523

524

525

526

527

528

529

530

531

532

533

using System;

using System.Collections.Generic;

using System.Linq;

using System.Text;

using System.Net.Sockets;

using System.Net;

using System.Threading;

 

namespace ServerTest

{

    /// <summary>

    /// IOCP SOCKET服务器

    /// </summary>

    public class IOCPServer : IDisposable

    {

        const int opsToPreAlloc = 2;

        #region Fields

        /// <summary>

        /// 服务器程序允许的最大客户端连接数

        /// </summary>

        private int _maxClient;

 

        /// <summary>

        /// 监听Socket,用于接受客户端的连接请求

        /// </summary>

        private Socket _serverSock;

 

        /// <summary>

        /// 当前的连接的客户端数

        /// </summary>

        private int _clientCount;

 

        /// <summary>

        /// 用于每个I/O Socket操作的缓冲区大小

        /// </summary>

        private int _bufferSize = 1024;

 

        /// <summary>

        /// 信号量

        /// </summary>

        Semaphore _maxAcceptedClients;

 

        /// <summary>

        /// 缓冲区管理

        /// </summary>

        BufferManager _bufferManager;

 

        /// <summary>

        /// 对象池

        /// </summary>

        SocketAsyncEventArgsPool _objectPool;

 

        private bool disposed = false;

 

        #endregion

 

        #region Properties

 

        /// <summary>

        /// 服务器是否正在运行

        /// </summary>

        public bool IsRunning { get; private set; }

        /// <summary>

        /// 监听的IP地址

        /// </summary>

        public IPAddress Address { get; private set; }

        /// <summary>

        /// 监听的端口

        /// </summary>

        public int Port { get; private set; }

        /// <summary>

        /// 通信使用的编码

        /// </summary>

        public Encoding Encoding { get; set; }

 

        #endregion

 

        #region Ctors

 

        /// <summary>

        /// 异步IOCP SOCKET服务器

        /// </summary>

        /// <param name="listenPort">监听的端口</param>

        /// <param name="maxClient">最大的客户端数量</param>

        public IOCPServer(int listenPort,int maxClient)

            : this(IPAddress.Any, listenPort, maxClient)

        {

        }

 

        /// <summary>

        /// 异步Socket TCP服务器

        /// </summary>

        /// <param name="localEP">监听的终结点</param>

        /// <param name="maxClient">最大客户端数量</param>

        public IOCPServer(IPEndPoint localEP, int maxClient)

            : this(localEP.Address, localEP.Port,maxClient)

        {

        }

 

        /// <summary>

        /// 异步Socket TCP服务器

        /// </summary>

        /// <param name="localIPAddress">监听的IP地址</param>

        /// <param name="listenPort">监听的端口</param>

        /// <param name="maxClient">最大客户端数量</param>

        public IOCPServer(IPAddress localIPAddress, int listenPort, int maxClient)

        {

            this.Address = localIPAddress;

            this.Port = listenPort;

            this.Encoding = Encoding.Default;

 

            _maxClient = maxClient;

 

            _serverSock = new Socket(localIPAddress.AddressFamily, SocketType.Stream, ProtocolType.Tcp);

 

            _bufferManager = new BufferManager(_bufferSize * _maxClient * opsToPreAlloc,_bufferSize);

 

            _objectPool = new SocketAsyncEventArgsPool(_maxClient);

 

            _maxAcceptedClients = new Semaphore(_maxClient, _maxClient);

        }

 

        #endregion

 

 

        #region 初始化

 

        /// <summary>

        /// 初始化函数

        /// </summary>

        public void Init()

        {

            // Allocates one large byte buffer which all I/O operations use a piece of.  This gaurds

            // against memory fragmentation

            _bufferManager.InitBuffer();

 

            // preallocate pool of SocketAsyncEventArgs objects

            SocketAsyncEventArgs readWriteEventArg;

 

            for (int i = 0; i < _maxClient; i++)

            {

                //Pre-allocate a set of reusable SocketAsyncEventArgs

                readWriteEventArg = new SocketAsyncEventArgs();

                readWriteEventArg.Completed += new EventHandler<SocketAsyncEventArgs>(OnIOCompleted);

                readWriteEventArg.UserToken = null;

 

                // assign a byte buffer from the buffer pool to the SocketAsyncEventArg object

                _bufferManager.SetBuffer(readWriteEventArg);

 

                // add SocketAsyncEventArg to the pool

                _objectPool.Push(readWriteEventArg);

            }

 

        }

 

        #endregion

 

        #region Start

        /// <summary>

        /// 启动

        /// </summary>

        public void Start()

        {

            if (!IsRunning)

            {

                Init();

                IsRunning = true;

                IPEndPoint localEndPoint = new IPEndPoint(Address, Port);

                // 创建监听socket

                _serverSock = new Socket(localEndPoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp);

                //_serverSock.ReceiveBufferSize = _bufferSize;

                //_serverSock.SendBufferSize = _bufferSize;

                if (localEndPoint.AddressFamily == AddressFamily.InterNetworkV6)

                {

                    // 配置监听socket为 dual-mode (IPv4 & IPv6)

                    // 27 is equivalent to IPV6_V6ONLY socket option in the winsock snippet below,

                    _serverSock.SetSocketOption(SocketOptionLevel.IPv6, (SocketOptionName)27, false);

                    _serverSock.Bind(new IPEndPoint(IPAddress.IPv6Any, localEndPoint.Port));

                }

                else

                {

                    _serverSock.Bind(localEndPoint);

                }

                // 开始监听

                _serverSock.Listen(this._maxClient);

                // 在监听Socket上投递一个接受请求。

                StartAccept(null);

            }

        }

        #endregion

 

        #region Stop

 

        /// <summary>

        /// 停止服务

        /// </summary>

        public void Stop()

        {

            if (IsRunning)

            {

                IsRunning = false;

                _serverSock.Close();

                //TODO 关闭对所有客户端的连接

 

            }

        }

 

        #endregion

 

 

        #region Accept

 

        /// <summary>

        /// 从客户端开始接受一个连接操作

        /// </summary>

        private void StartAccept(SocketAsyncEventArgs asyniar)

        {

            if (asyniar == null)

            {

                asyniar = new SocketAsyncEventArgs();

                asyniar.Completed += new EventHandler<SocketAsyncEventArgs>(OnAcceptCompleted);

            }

            else

            {

                //socket must be cleared since the context object is being reused

                asyniar.AcceptSocket = null;

            }

            _maxAcceptedClients.WaitOne();

            if (!_serverSock.AcceptAsync(asyniar))

            {

                ProcessAccept(asyniar);

                //如果I/O挂起等待异步则触发AcceptAsyn_Asyn_Completed事件

                //此时I/O操作同步完成,不会触发Asyn_Completed事件,所以指定BeginAccept()方法

            }

        }

 

        /// <summary>

        /// accept 操作完成时回调函数

        /// </summary>

        /// <param name="sender">Object who raised the event.</param>

        /// <param name="e">SocketAsyncEventArg associated with the completed accept operation.</param>

        private void OnAcceptCompleted(object sender, SocketAsyncEventArgs e)

        {

            ProcessAccept(e);

        }

 

        /// <summary>

        /// 监听Socket接受处理

        /// </summary>

        /// <param name="e">SocketAsyncEventArg associated with the completed accept operation.</param>

        private void ProcessAccept(SocketAsyncEventArgs e)

        {

            if (e.SocketError == SocketError.Success)

            {

                Socket s = e.AcceptSocket;//和客户端关联的socket

                if (s.Connected)

                {

                    try

                    {

                         

                        Interlocked.Increment(ref _clientCount);//原子操作加1

                        SocketAsyncEventArgs asyniar = _objectPool.Pop();

                        asyniar.UserToken = s;

 

                        Log4Debug(String.Format("客户 {0} 连入, 共有 {1} 个连接。", s.RemoteEndPoint.ToString(), _clientCount));

                         

                        if (!s.ReceiveAsync(asyniar))//投递接收请求

                        {

                            ProcessReceive(asyniar);

                        }

                    }

                    catch (SocketException ex)

                    {

                        Log4Debug(String.Format("接收客户 {0} 数据出错, 异常信息: {1} 。", s.RemoteEndPoint, ex.ToString()));

                        //TODO 异常处理

                    }

                    //投递下一个接受请求

                    StartAccept(e);

                }

            }

        }

 

        #endregion

 

        #region 发送数据

 

        /// <summary>

        /// 异步的发送数据

        /// </summary>

        /// <param name="e"></param>

        /// <param name="data"></param>

        public void Send(SocketAsyncEventArgs e, byte[] data)

        {

            if (e.SocketError == SocketError.Success)

            {

                Socket s = e.AcceptSocket;//和客户端关联的socket

                if (s.Connected)

                {

                    Array.Copy(data, 0, e.Buffer, 0, data.Length);//设置发送数据

 

                    //e.SetBuffer(data, 0, data.Length); //设置发送数据

                    if (!s.SendAsync(e))//投递发送请求,这个函数有可能同步发送出去,这时返回false,并且不会引发SocketAsyncEventArgs.Completed事件

                    {

                        // 同步发送时处理发送完成事件

                        ProcessSend(e);

                    }

                    else

                    {

                        CloseClientSocket(e);

                    }

                }

            }

        }

 

        /// <summary>

        /// 同步的使用socket发送数据

        /// </summary>

        /// <param name="socket"></param>

        /// <param name="buffer"></param>

        /// <param name="offset"></param>

        /// <param name="size"></param>

        /// <param name="timeout"></param>

        public void Send(Socket socket, byte[] buffer, int offset, int size, int timeout)

        {

            socket.SendTimeout = 0;

            int startTickCount = Environment.TickCount;

            int sent = 0; // how many bytes is already sent

            do

            {

                if (Environment.TickCount > startTickCount + timeout)

                {

                    //throw new Exception("Timeout.");

                }

                try

                {

                    sent += socket.Send(buffer, offset + sent, size - sent, SocketFlags.None);

                }

                catch (SocketException ex)

                {

                    if (ex.SocketErrorCode == SocketError.WouldBlock ||

                    ex.SocketErrorCode == SocketError.IOPending ||

                    ex.SocketErrorCode == SocketError.NoBufferSpaceAvailable)

                    {

                        // socket buffer is probably full, wait and try again

                        Thread.Sleep(30);

                    }

                    else

                    {

                        throw ex; // any serious error occurr

                    }

                }

            } while (sent < size);

        }

 

 

        /// <summary>

        /// 发送完成时处理函数

        /// </summary>

        /// <param name="e">与发送完成操作相关联的SocketAsyncEventArg对象</param>

        private void ProcessSend(SocketAsyncEventArgs e)

        {

            if (e.SocketError == SocketError.Success)

            {

                Socket s = (Socket)e.UserToken;

 

                //TODO

            }

            else

            {

                CloseClientSocket(e);

            }

        }

 

        #endregion

 

        #region 接收数据

 

 

        /// <summary>

        ///接收完成时处理函数

        /// </summary>

        /// <param name="e">与接收完成操作相关联的SocketAsyncEventArg对象</param>

        private void ProcessReceive(SocketAsyncEventArgs e)

        {

            if (e.SocketError == SocketError.Success)//if (e.BytesTransferred > 0 && e.SocketError == SocketError.Success)

            {

                // 检查远程主机是否关闭连接

                if (e.BytesTransferred > 0)

                {

                    Socket s = (Socket)e.UserToken;

                    //判断所有需接收的数据是否已经完成

                    if (s.Available == 0)

                    {

                        //从侦听者获取接收到的消息。

                        //String received = Encoding.ASCII.GetString(e.Buffer, e.Offset, e.BytesTransferred);

                        //echo the data received back to the client

                        //e.SetBuffer(e.Offset, e.BytesTransferred);

 

                        byte[] data = new byte[e.BytesTransferred];

                        Array.Copy(e.Buffer, e.Offset, data, 0, data.Length);//从e.Buffer块中复制数据出来,保证它可重用

 

                        string info=Encoding.Default.GetString(data);

                        Log4Debug(String.Format("收到 {0} 数据为 {1}",s.RemoteEndPoint.ToString(),info));

                        //TODO 处理数据

 

                        //增加服务器接收的总字节数。

                    }

 

                    if (!s.ReceiveAsync(e))//为接收下一段数据,投递接收请求,这个函数有可能同步完成,这时返回false,并且不会引发SocketAsyncEventArgs.Completed事件

                    {

                        //同步接收时处理接收完成事件

                        ProcessReceive(e);

                    }

                }

            }

            else

            {

                CloseClientSocket(e);

            }

        }

 

        #endregion

 

        #region 回调函数

 

        /// <summary>

        /// 当Socket上的发送或接收请求被完成时,调用此函数

        /// </summary>

        /// <param name="sender">激发事件的对象</param>

        /// <param name="e">与发送或接收完成操作相关联的SocketAsyncEventArg对象</param>

        private void OnIOCompleted(object sender, SocketAsyncEventArgs e)

        {

            // Determine which type of operation just completed and call the associated handler.

            switch (e.LastOperation)

            {

                case SocketAsyncOperation.Accept:

                    ProcessAccept(e);

                    break;

                case SocketAsyncOperation.Receive:

                    ProcessReceive(e);

                    break;

                default:

                    throw new ArgumentException("The last operation completed on the socket was not a receive or send");

            }

        }

 

        #endregion

 

        #region Close

        /// <summary>

        /// 关闭socket连接

        /// </summary>

        /// <param name="e">SocketAsyncEventArg associated with the completed send/receive operation.</param>

        private void CloseClientSocket(SocketAsyncEventArgs e)

        {

            Log4Debug(String.Format("客户 {0} 断开连接!",((Socket)e.UserToken).RemoteEndPoint.ToString()));

            Socket s = e.UserToken as Socket;

            CloseClientSocket(s, e);

        }

 

        /// <summary>

        /// 关闭socket连接

        /// </summary>

        /// <param name="s"></param>

        /// <param name="e"></param>

        private void CloseClientSocket(Socket s, SocketAsyncEventArgs e)

        {

            try

            {

                s.Shutdown(SocketShutdown.Send);

            }

            catch (Exception)

            {

                // Throw if client has closed, so it is not necessary to catch.

            }

            finally

            {

                s.Close();

            }

            Interlocked.Decrement(ref _clientCount);

            _maxAcceptedClients.Release();

            _objectPool.Push(e);//SocketAsyncEventArg 对象被释放,压入可重用队列。

        }

        #endregion

 

        #region Dispose

        /// <summary>

        /// Performs application-defined tasks associated with freeing,

        /// releasing, or resetting unmanaged resources.

        /// </summary>

        public void Dispose()

        {

            Dispose(true);

            GC.SuppressFinalize(this);

        }

 

        /// <summary>

        /// Releases unmanaged and - optionally - managed resources

        /// </summary>

        /// <param name="disposing"><c>true</c> to release

        /// both managed and unmanaged resources; <c>false</c>

        /// to release only unmanaged resources.</param>

        protected virtual void Dispose(bool disposing)

        {

            if (!this.disposed)

            {

                if (disposing)

                {

                    try

                    {

                        Stop();

                        if (_serverSock != null)

                        {

                            _serverSock = null;

                        }

                    }

                    catch (SocketException ex)

                    {

                        //TODO 事件

                    }

                }

                disposed = true;

            }

        }

        #endregion

 

        public void Log4Debug(string msg)

        {

            Console.WriteLine("notice:"+msg);

        }

 

    }

}

登录后复制


BufferManager.cs 这个类是缓存管理类,是采用MSDN上面的例子一样的 地址: http://www.php.cn/

SocketAsyncEventArgsPool.cs 这个类也是来自MSDN的 地址:http://www.php.cn/

需要的话自己到MSDN网站上去取,我就不贴出来了


服务器端

1

2

3

4

5

6

7

8

static void Main(string[] args)

        {

 

            IOCPServer server = new IOCPServer(8088, 1024);

            server.Start();

            Console.WriteLine("服务器已启动....");

            System.Console.ReadLine();

        }

登录后复制

客户端

客户端代码也是很简单

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

static void Main(string[] args)

        {

            IPAddress remote=IPAddress.Parse("192.168.3.4");

            client c = new client(8088,remote);

 

            c.connect();

            Console.WriteLine("服务器连接成功!");

            while (true)

            {

                Console.Write("send>");

                string msg=Console.ReadLine();

                if (msg == "exit")

                    break;

                c.send(msg);

            }

            c.disconnect();

            Console.ReadLine();

        }

登录后复制

client.cs

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

public class client

    {

 

        public TcpClient _client;

 

        public int port;

 

        public IPAddress remote;

 

        public client(int port,IPAddress remote)

        {

 

            this.port = port;

            this.remote = remote;

        }

 

        public void connect()

        {

            this._client=new TcpClient();

            _client.Connect(remote, port);

        }

        public void disconnect()

        {

            _client.Close();

        }

        public void send(string msg)

        {

            byte[] data=Encoding.Default.GetBytes(msg);

            _client.GetStream().Write(data, 0, data.Length);

        }

    }

登录后复制


IOCPClient类,使用SocketAsyncEventArgs类建立一个Socket客户端。虽然MSDN说这个类特别设计给网络服务器应用,但也没有限制在客户端代码中使用APM。下面给出了IOCPClient类的样例代码:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

136

137

138

139

140

141

142

143

144

145

146

147

148

149

150

151

152

153

154

155

156

157

158

159

160

161

162

163

164

165

166

167

168

169

170

171

172

173

174

175

176

177

178

179

180

181

182

183

184

185

186

187

188

189

190

191

192

193

194

195

196

197

198

199

200

201

202

203

204

205

206

207

208

209

210

211

212

213

214

215

216

217

218

219

220

221

222

223

224

225

226

227

228

229

230

231

232

233

234

235

public class IOCPClient

   {

       /// <summary>

       /// 连接服务器的socket

       /// </summary>

       private Socket _clientSock;

 

       /// <summary>

       /// 用于服务器执行的互斥同步对象

       /// </summary>

       private static Mutex mutex = new Mutex();

       /// <summary>

       /// Socket连接标志

       /// </summary>

       private Boolean _connected = false;

 

       private const int ReceiveOperation = 1, SendOperation = 0;

 

       private static AutoResetEvent[]

                autoSendReceiveEvents = new AutoResetEvent[]

        {

            new AutoResetEvent(false),

            new AutoResetEvent(false)

        };

 

       /// <summary>

       /// 服务器监听端点

       /// </summary>

       private IPEndPoint _remoteEndPoint;

 

       public IOCPClient(IPEndPoint local,IPEndPoint remote)

       {

           _clientSock = new Socket(local.AddressFamily,SocketType.Stream, ProtocolType.Tcp);

           _remoteEndPoint = remote;

       }

 

       #region 连接服务器

 

       /// <summary>

       /// 连接远程服务器

       /// </summary>

       public void Connect()

       {

           SocketAsyncEventArgs connectArgs = new SocketAsyncEventArgs();

 

           connectArgs.UserToken = _clientSock;

           connectArgs.RemoteEndPoint = _remoteEndPoint;

           connectArgs.Completed += new EventHandler<SocketAsyncEventArgs>(OnConnected);

           mutex.WaitOne();

           if (!_clientSock.ConnectAsync(connectArgs))//异步连接

           {

               ProcessConnected(connectArgs);

           }

            

       }

       /// <summary>

       /// 连接上的事件

       /// </summary>

       /// <param name="sender"></param>

       /// <param name="e"></param>

       void OnConnected(object sender, SocketAsyncEventArgs e)

       {

           mutex.ReleaseMutex();

           //设置Socket已连接标志。

           _connected = (e.SocketError == SocketError.Success);

       }

       /// <summary>

       /// 处理连接服务器

       /// </summary>

       /// <param name="e"></param>

       private void ProcessConnected(SocketAsyncEventArgs e)

       {

           //TODO

       }

 

       #endregion

 

       #region 发送消息

       /// <summary>

       /// 向服务器发送消息

       /// </summary>

       /// <param name="data"></param>

       public void Send(byte[] data)

       {

           SocketAsyncEventArgs asyniar = new SocketAsyncEventArgs();

           asyniar.Completed += new EventHandler<SocketAsyncEventArgs>(OnSendComplete);

           asyniar.SetBuffer(data, 0, data.Length);

           asyniar.UserToken = _clientSock;

           asyniar.RemoteEndPoint = _remoteEndPoint;

           autoSendReceiveEvents[SendOperation].WaitOne();

           if (!_clientSock.SendAsync(asyniar))//投递发送请求,这个函数有可能同步发送出去,这时返回false,并且不会引发SocketAsyncEventArgs.Completed事件

           {

               // 同步发送时处理发送完成事件

               ProcessSend(asyniar);

           }

       }

 

       /// <summary>

       /// 发送操作的回调方法

       /// </summary>

       /// <param name="sender"></param>

       /// <param name="e"></param>

       private void OnSendComplete(object sender, SocketAsyncEventArgs e)

       {

           //发出发送完成信号。

           autoSendReceiveEvents[SendOperation].Set();

           ProcessSend(e);

       }

 

       /// <summary>

       /// 发送完成时处理函数

       /// </summary>

       /// <param name="e">与发送完成操作相关联的SocketAsyncEventArg对象</param>

       private void ProcessSend(SocketAsyncEventArgs e)

       {

           //TODO

       }

       #endregion

 

       #region 接收消息

       /// <summary>

       /// 开始监听服务端数据

       /// </summary>

       /// <param name="e"></param>

       public void StartRecive(SocketAsyncEventArgs e)

       {

           //准备接收。

           Socket s = e.UserToken as Socket;

           byte[] receiveBuffer = new byte[255];

           e.SetBuffer(receiveBuffer, 0, receiveBuffer.Length);

           e.Completed += new EventHandler<SocketAsyncEventArgs>(OnReceiveComplete);

           autoSendReceiveEvents[ReceiveOperation].WaitOne();

           if (!s.ReceiveAsync(e))

           {

               ProcessReceive(e);

           }

       }

 

       /// <summary>

       /// 接收操作的回调方法

       /// </summary>

       /// <param name="sender"></param>

       /// <param name="e"></param>

       private void OnReceiveComplete(object sender, SocketAsyncEventArgs e)

       {

           //发出接收完成信号。

           autoSendReceiveEvents[ReceiveOperation].Set();

           ProcessReceive(e);

       }

 

       /// <summary>

       ///接收完成时处理函数

       /// </summary>

       /// <param name="e">与接收完成操作相关联的SocketAsyncEventArg对象</param>

       private void ProcessReceive(SocketAsyncEventArgs e)

       {

           if (e.SocketError == SocketError.Success)

           {

               // 检查远程主机是否关闭连接

               if (e.BytesTransferred > 0)

               {

                   Socket s = (Socket)e.UserToken;

                   //判断所有需接收的数据是否已经完成

                   if (s.Available == 0)

                   {

                       byte[] data = new byte[e.BytesTransferred];

                       Array.Copy(e.Buffer, e.Offset, data, 0, data.Length);//从e.Buffer块中复制数据出来,保证它可重用

 

                       //TODO 处理数据

                   }

 

                   if (!s.ReceiveAsync(e))//为接收下一段数据,投递接收请求,这个函数有可能同步完成,这时返回false,并且不会引发SocketAsyncEventArgs.Completed事件

                   {

                       //同步接收时处理接收完成事件

                       ProcessReceive(e);

                   }

               }

           }

       }

 

       #endregion

 

 

       public void Close()

       {

           _clientSock.Disconnect(false);

       }

 

       /// <summary>

       /// 失败时关闭Socket,根据SocketError抛出异常。

       /// </summary>

       /// <param name="e"></param>

 

       private void ProcessError(SocketAsyncEventArgs e)

       {

           Socket s = e.UserToken as Socket;

           if (s.Connected)

           {

               //关闭与客户端关联的Socket

               try

               {

                   s.Shutdown(SocketShutdown.Both);

               }

               catch (Exception)

               {

                   //如果客户端处理已经关闭,抛出异常

               }

               finally

               {

                   if (s.Connected)

                   {

                       s.Close();

                   }

               }

           }

           //抛出SocketException

           throw new SocketException((Int32)e.SocketError);

       }

 

 

       /// <summary>

       /// 释放SocketClient实例

       /// </summary>

       public void Dispose()

       {

           mutex.Close();

           autoSendReceiveEvents[SendOperation].Close();

           autoSendReceiveEvents[ReceiveOperation].Close();

           if (_clientSock.Connected)

           {

               _clientSock.Close();

           }

       }

 

   }

登录后复制

这个类我没有测试,但是理论上是没问题的。

 以上就是C#,SocketAsyncEventArgs,服务器的内容,更多相关内容请关注PHP中文网(www.php.cn)!


本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
2 周前 By 尊渡假赌尊渡假赌尊渡假赌
仓库:如何复兴队友
1 个月前 By 尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island冒险:如何获得巨型种子
4 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

使用 C# 的活动目录 使用 C# 的活动目录 Sep 03, 2024 pm 03:33 PM

使用 C# 的 Active Directory 指南。在这里,我们讨论 Active Directory 在 C# 中的介绍和工作原理以及语法和示例。

C# 中的随机数生成器 C# 中的随机数生成器 Sep 03, 2024 pm 03:34 PM

C# 随机数生成器指南。在这里,我们讨论随机数生成器的工作原理、伪随机数和安全数的概念。

C# 中的访问修饰符 C# 中的访问修饰符 Sep 03, 2024 pm 03:24 PM

C# 中的访问修饰符指南。我们已经讨论了 C# 中访问修饰符的简介类型以及示例和输出。

C# 数据网格视图 C# 数据网格视图 Sep 03, 2024 pm 03:32 PM

C# 数据网格视图指南。在这里,我们讨论如何从 SQL 数据库或 Excel 文件加载和导出数据网格视图的示例。

C# 中的模式 C# 中的模式 Sep 03, 2024 pm 03:33 PM

C# 模式指南。在这里,我们讨论 C# 中模式的介绍和前 3 种类型,以及其示例和代码实现。

C# 序列化 C# 序列化 Sep 03, 2024 pm 03:30 PM

C# 序列化指南。这里我们分别讨论C#序列化对象的介绍、步骤、工作原理和示例。

C# 中的 Web 服务 C# 中的 Web 服务 Sep 03, 2024 pm 03:32 PM

C# Web 服务指南。在这里,我们讨论 C# 中的 Web 服务简介,包括技术使用、限制和示例。

C# 中的质数 C# 中的质数 Sep 03, 2024 pm 03:35 PM

C# 素数指南。这里我们讨论c#中素数的介绍和示例以及代码实现。

See all articles