Linux IO 多重化 epoll ネットワーク プログラミング

Linux IO 多重化 epoll ネットワーク プログラミング

序文

この章では、基本的な Linux 関数と epoll 呼び出しを使用して、Linux 上で実行できる完全なサーバーおよびクライアントの例を記述します。クライアントとサーバーの機能は次のとおりです。

  • クライアントは標準入力から行を読み取り、それをサーバーに送信します。
  • サーバーはネットワークから行を読み取り、クライアントに出力します。
  • クライアントはサーバーからの応答を受信し、この行を標準出力に出力します。

サーバ

コードは次のとおりです。

#include <unistd.h>
#include <sys/types.h> /* 基本的なシステムデータ型 */
#include <sys/socket.h> /* 基本的なソケット定義 */
#include <netinet/in.h> /* sockaddr_in{} およびその他のインターネット定義 */
#include <arpa/inet.h> /* inet(3) 関数 */
#include <sys/epoll.h> /* epoll関数 */
#include <fcntl.h> /* 非ブロッキング */
#include <sys/resource.h> /*setrlimit */
#include <stdlib.h>
#include <errno.h>
#include <stdio.h>
#include <文字列.h>
#定義 MAXEPOLLSIZE 10000
#定義 MAXLINE 10240
int ハンドル(int 接続);
int setnonblocking(int sockfd)
{
  (fcntl(sockfd, F_SETFL, fcntl(sockfd, F_GETFD, 0)|O_NONBLOCK) == -1) の場合 {
    -1 を返します。
  }
  0を返します。
}
int main(int argc, char **argv)
{
  int servPort = 6888;
  整数 listenq = 1024;
  int listenfd、connfd、kdpfd、nfds、n、nread、curfds、acceptCount = 0;
  構造体 sockaddr_in servaddr、cliaddr;
  socklen_t socklen = sizeof(struct sockaddr_in);
  構造体 epoll_event ev;
  構造体epoll_eventイベント[MAXEPOLLSIZE];
  構造体rlimitrt;
  char buf[MAXLINE];
  /* 各プロセスが開くことができるファイルの最大数を設定します */
  rt.rlim_max = rt.rlim_cur = MAXEPOLLSIZE;
  (setrlimit(RLIMIT_NOFILE, &rt) == -1)の場合 
  {
    perror("setrlimitエラー");
    -1 を返します。
  }
  bzero(&servaddr, sizeof(servaddr));
  servaddr.sin_family = AF_INET; 
  servaddr.sin_addr.s_addr = htonl (INADDR_ANY);
  servaddr.sin_port = htons(servPort);
  listenfd = socket(AF_INET、SOCK_STREAM、0); 
  (listenfd == -1)の場合{
    perror("ソケットファイルを作成できません");
    -1 を返します。
  }
  整数オプト = 1;
  setsockopt(listenfd、SOL_SOCKET、SO_REUSEADDR、&opt、sizeof(opt));
  (setnonblocking(listenfd) < 0) の場合 {
    perror("setnonblockエラー");
  }
  (bind(listenfd, (struct sockaddr *) &servaddr, sizeof(struct sockaddr)) == -1)の場合 
  {
    perror("バインドエラー");
    -1 を返します。
  } 
  (listen(listenfd, listenq) == -1)の場合 
  {
    perror("エラーを聞く");
    -1 を返します。
  }
  /* epoll ハンドルを作成し、リスニング ソケットを epoll セットに追加します */
  kdpfd = epoll_create(MAXEPOLLSIZE);
  ev.events = EPOLLIN | EPOLLET;
  ev.data.fd = listenfd;
  (epoll_ctl(kdpfd, EPOLL_CTL_ADD, listenfd, &ev) < 0 の場合) 
  {
    fprintf(stderr, "epoll セット挿入エラー: fd=%d\n", listenfd);
    -1 を返します。
  }
  曲線 = 1;
  printf("epollserver の起動、ポート %d、最大接続数は %d、バックログは %d\n", servPort、MAXEPOLLSIZE、listenq);
  のために (;;) {
    /* イベントが発生するのを待つ */
    nfds = epoll_wait(kdpfd、イベント、curfds、-1);
    (nfds == -1)の場合
    {
      perror("epoll_wait");
      続く;
    }
    /* すべてのイベントを処理する */
    (n = 0; n < nfds; ++n)の場合
    {
      if (events[n].data.fd == listenfd) 
      {
        connfd = accept(listenfd、(struct sockaddr *)&cliaddr、&socklen);
        (接続数 < 0)の場合 
        {
          perror("エラーを受け入れる");
          続く;
        }
        sprintf(buf, "フォーム%sを受け入れます:%d\n", inet_ntoa(cliaddr.sin_addr), cliaddr.sin_port);
        printf("%d:%s", ++acceptCount, buf);

        (curfds >= MAXEPOLLSIZE)の場合{
          fprintf(stderr, "接続が多すぎます。%d\n 以上です", MAXEPOLLSIZE);
          close(接続);
          続く;
        } 
        (setnonblocking(connfd) < 0) の場合 {
          perror("setnonblockingエラー");
        }
        ev.events = EPOLLIN | EPOLLET;
        ev.data.fd = connfd;
        (epoll_ctl(kdpfd, EPOLL_CTL_ADD, connfd, &ev) < 0 の場合)
        {
          fprintf(stderr, "ソケット '%d' を epoll に追加できませんでした: %s\n", connfd, strerror(errno));
          -1 を返します。
        }
        カーフds++;
        続く;
      } 
      // クライアント要求を処理する if (handle(events[n].data.fd) < 0) {
        epoll_ctl(kdpfd、EPOLL_CTL_DEL、イベント[n].data.fd、&ev);
        curfds--;
      }
    }
  }
  listenfdを閉じます。
  0を返します。
}
int ハンドル(int 接続) {
  int 読み取り;
  char buf[MAXLINE];
  nread = read(connfd, buf, MAXLINE); //クライアントソケットストリームを読み取る if (nread == 0) {
    printf("クライアントは接続を閉じます\n");
    close(接続);
    -1 を返します。
  } 
  (nread < 0) の場合 {
    perror("読み取りエラー");
    close(接続);
    -1 を返します。
  }  
  write(connfd, buf, nread); //クライアントに応答する return 0;
}

コンパイル

サーバーをコンパイルして起動する

gcc epollserver.c -o epollserver
./epollserver

要約する

以上がこの記事の全内容です。この記事の内容が皆様の勉強や仕事に何らかの参考学習価値をもたらすことを願います。123WORDPRESS.COM をご愛顧いただき、誠にありがとうございます。これについてもっと知りたい場合は、次のリンクをご覧ください。

以下もご興味があるかもしれません:
  • Linux IOの詳細な紹介
  • Linux で iostat コマンドを使用するチュートリアル
  • Linux のソケット IO モデルの興味深い説明
  • Linux の 5 つの IO モデルの詳細な紹介
  • Linux の高性能ネットワーク IO と Reactor モデルの分析

<<:  JavaScript で最も高速なループはどれですか?

>>:  MySQL 圧縮の使用シナリオとソリューション

推薦する

MYSQLでリモートアクセス権限を有効にする方法

1. MySQLデータベースにログインするmysql -u ルート -pユーザーテーブルを表示する ...

Javascript における分割代入構文の詳細な説明

序文ES6 で初めて導入された「構造化代入構文」を使用すると、配列やオブジェクトの値を異なる変数に挿...

ウェブページの幅を携帯電話の画面(ビューポート)の幅に自動的に適応させる実装コード

一般的な書き方は次のとおりです。 XML/HTML コードコンテンツをクリップボードにコピー<...

スライディングカルーセル効果を実現する js

この記事では、スライディングカルーセル効果を実現するためのjsの具体的なコードを参考までに共有します...

内部 IP アクセスのみを許可する Nginx プロキシ設定を追加する方法

位置 / { インデックス index.jsp; proxy_next_upstream http...

FTP、FTPS、SFTPの違いについて簡単に説明します

目次FTP、FTPS、SFTP の概要FTP FTPS FTPサーバーFTPソフトウェアのアクティブ...

CocosCreator でカメラトラッキングに cc.follow を使用する方法

Cocos Creator バージョン: 2.3.4デモのダウンロード: https://files...

element-plus でオンデマンドインポートとグローバルインポートを実装する方法

目次オンデマンドインポート:グローバルインポートオンデマンドインポート:プラグインをインストールする...

Linux リモート管理と sshd サービス検証の知識ポイントの詳細な説明

1. SSHリモート管理SSH の定義SSH (Secure Shell) は、主にキャラクタ イン...

Mac OS10.12 に mysql5.7.18 をインストールするチュートリアル

ウェブ全体を検索して、さまざまな落とし穴を見つけましたが、問題は解決しませんでした。ついに自分でも分...

Alibaba Cloud centos7にmysql8.0.22をインストールする詳細なチュートリアル

1. MySQLインストールパッケージをダウンロードするまず、https://dev.mysql.c...

JS for ループで setTimeout を使用する 4 つのソリューション

目次概要解決策 1: クロージャ解決策2: 構造を分割する解決策3:解決策4: setTimeout...

JVM 上の高性能データ形式ライブラリ パッケージである Apache Arrow の紹介とアーキテクチャ (Gkatziouras)

Apache Arrow は、BigQuery を含むさまざまなビッグデータ ツールで使用される一...

jsとcssのブロッキング問題の詳細な分析

目次DOMContentLoadedとロードjs ブロッキングとは何ですか? CSS ブロッキングと...