c語(yǔ)言網(wǎng)絡(luò)編程-標(biāo)準(zhǔn)步驟(改進(jìn)版)
server.c
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <netdb.h>
#include <netinet/in.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <fcntl.h>
#include <sys/epoll.h>
#include <sys/errno.h>
#define PORT 4444
#define BACKLOG 5
#define MAX_FD 256
void setnonblocking(int);
int main(int argc, char *argv[]) {
int sock_fd, new_fd, new_fd2, epfd, nfds;
struct sockaddr_in server_addr, client_addr;
int sin_size;
int nbytes;
int on = 1;
char buffer[1024];
struct epoll_event ev, events[20];
epfd = epoll_create(MAX_FD);
if ((sock_fd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
perror("socket");
exit(1);
}
setsockopt(sock_fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
setnonblocking(sock_fd);
ev.data.fd = sock_fd;
ev.events = EPOLLIN | EPOLLET;
epoll_ctl(epfd, EPOLL_CTL_ADD, sock_fd, &ev);
memset(&server_addr, 0, sizeof(struct sockaddr_in));
server_addr.sin_family = AF_INET;
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
server_addr.sin_port = htons(PORT);
if (bind(sock_fd, (struct sockaddr *)(&server_addr), sizeof(struct sockaddr)) == -1) {
perror("bind");
exit(1);
}
if (listen(sock_fd, BACKLOG) == -1) {
perror("listen");
exit(1);
}
printf("Server start... \n");
sin_size = sizeof(struct sockaddr_in);
int i;
while (1) {
nfds = epoll_wait(epfd, events, 20, 500);
for (i = 0; i < nfds; i++) {
if (events[i].data.fd == sock_fd) {
if ((new_fd = accept(sock_fd, (struct sockaddr *)(&client_addr), &sin_size)) == -1) {
perror("accept");
exit(1);
}
printf("Server get connection from %s\n", inet_ntoa(client_addr.sin_addr));
setnonblocking(new_fd);
ev.data.fd = new_fd;
ev.events = EPOLLIN | EPOLLET;
epoll_ctl(epfd, EPOLL_CTL_ADD, new_fd, &ev);
} else if (events[i].events & EPOLLIN) {
if ((new_fd2 = events[i].data.fd) < 0) continue;
if ((nbytes = read(new_fd2, buffer, 1024)) == -1) {
perror("read");
exit(1);
} else if (nbytes == 0) {
close(new_fd2);
events[i].data.fd = -1;
}
printf("Server read from %s\n", inet_ntoa(client_addr.sin_addr));
ev.data.fd = new_fd2;
ev.events = EPOLLOUT | EPOLLET;
epoll_ctl(epfd, EPOLL_CTL_MOD, new_fd2, &ev);
} else if (events[i].events & EPOLLOUT) {
new_fd2 = events[i].data.fd;
write(new_fd2, "Server Received", 16);
ev.data.fd = new_fd2;
ev.events = EPOLLIN | EPOLLET;
epoll_ctl(epfd, EPOLL_CTL_MOD, new_fd2, &ev);
}
}
}
return 0;
}
void setnonblocking(int fd) {
int opts;
if ((opts = fcntl(fd, F_GETFL)) < 0) {
perror("fcntl(fd, F_GETFL");
exit(1);
}
opts = opts | O_NONBLOCK;
if (fcntl(fd, F_SETFL, opts) < 0) {
perror("fcntl(fd, F_SETFL, opts");
exit(1);
}
}
client.c
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#define PORT 4444
int main(int argc, char *argv[]) {
int sock_fd;
struct sockaddr_in server_addr;
struct hostent *host;
char buffer[1024];
if (argc < 2) {
perror("Need hostname");
exit(1);
}
if ((host = gethostbyname(argv[1])) == NULL) {
perror("gethostbyname");
exit(1);
}
if ((sock_fd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
perror("socket");
exit(1);
}
memset(&server_addr, 0, sizeof(struct sockaddr_in));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(PORT);
server_addr.sin_addr = *((struct in_addr *)host->h_addr);
if (connect(sock_fd, (struct sockaddr *)(&server_addr), sizeof(struct sockaddr)) == -1) {
perror("connect");
exit(1);
}
while (1) {
printf("Please input something:\n");
fgets(buffer, 1024, stdin);
write(sock_fd, buffer, strlen(buffer));
read(sock_fd, buffer, 1024);
printf("From server: %s\n", buffer);
}
close(sock_fd);
return 0;
}
相關(guān)文章
C語(yǔ)言字符串函數(shù)介紹與模擬實(shí)現(xiàn)詳解
這篇文章主要介紹了C語(yǔ)言實(shí)現(xiàn)字符串操作函數(shù)的實(shí)例的相關(guān)資料,開(kāi)發(fā)程序的時(shí)候經(jīng)常使用到一些字符串函數(shù),例如求字符串長(zhǎng)度,拷貝字符串……,需要的朋友可以參考下2021-09-09詳談C++中虛基類(lèi)在派生類(lèi)中的內(nèi)存布局
下面小編就為大家?guī)?lái)一篇詳談C++中虛基類(lèi)在派生類(lèi)中的內(nèi)存布局。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-12-12C語(yǔ)言實(shí)現(xiàn)在數(shù)組A上有序合并數(shù)組B的方法
這篇文章主要介紹了C語(yǔ)言實(shí)現(xiàn)在數(shù)組A上有序合并數(shù)組B的方法,包含了數(shù)組操作的完整實(shí)現(xiàn)過(guò)程以及相應(yīng)的代碼分析與改進(jìn),具有不錯(cuò)的借鑒價(jià)值,需要的朋友可以參考下2014-09-09C/C++使用fmt庫(kù)實(shí)現(xiàn)格式化字符串
fmt庫(kù)是一個(gè)高效、易用的C++格式化庫(kù),可以幫助我們方便地進(jìn)行字符串格式化、輸出、日志記錄等操作,下面我們就來(lái)學(xué)習(xí)一下fmt格式化字符串的具體操作吧2023-12-12C++實(shí)現(xiàn)對(duì)回收站里的文件進(jìn)行操作的示例代碼
這篇文章主要為大家詳細(xì)介紹了C++如何使用代碼對(duì)回收站里的文件進(jìn)行操作,譬如文件的刪除與恢復(fù)等,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價(jià)值,感興趣的小伙伴快跟隨小編一起學(xué)習(xí)學(xué)習(xí)吧2023-06-06淺析string類(lèi)字符串和C風(fēng)格字符串之間的區(qū)別
string類(lèi)是標(biāo)準(zhǔn)庫(kù)的類(lèi),并不是內(nèi)置類(lèi)型,標(biāo)準(zhǔn)庫(kù)就像是我們自己定義的類(lèi)差不多的,string類(lèi)型對(duì)象沒(méi)有標(biāo)配'\0'結(jié)尾的2013-09-09使用C++和Direct3D (d3d)獲取屏幕截圖并根據(jù)傳入分辨率進(jìn)行縮放圖片大小(最新推薦)
這篇文章主要介紹了使用C++和Direct3D (d3d)獲取屏幕截圖并根據(jù)傳入分辨率進(jìn)行縮放圖片大小,本文給大家講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-04-04OpenCV實(shí)現(xiàn)圖像的直線檢測(cè)
這篇文章主要為大家詳細(xì)介紹了OpenCV實(shí)現(xiàn)圖像直線檢測(cè)的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01