C/C++编程日记:用C语言实现的简单Web服务器(Linux),全代码分享!
相信大家对Apache都有所听闻,Apache是目前使用最为广泛我Web服务器。大家可以从news.netcraft.com/这个网站得到证实。
这是腾讯的uptime.netcraft.com/up/graph?site=www.qq.com.Apache强大的功能和高效的性能并且开放源代码的这种模式对我很有吸引力,但无赖自己水平有限,无法从Apache庞大的source code里面理清头绪。于是,我就冒出了个自己动手写一个小型的简单的Web服务器的想法,希望对这方面也和我一样感兴趣的朋友有所帮助。

我的实验环境为:
OS:Red Hat Enterprise Linux 5
gcc:4.1.2
libc:2.5
editor:Vim
lang:C
阅读该源代码需要以下预备知识:
C语言基础
Linux编程基础
socket编程基础(Linux)
TCP/IP基本原理
HTTP基本原理
关键字(Key Words):
Linux C, Web HTTP Server, Linux Socket.
-----------------------------------------------------------------------------------
下面是Mutu的第一个版本(0.1 Alpha),实现了WEB 服务器的最基本功能
包括以下源文件:
webserver.c----程序入口
init_socket.h init_socket.c----完成一些WEB服务器的初始化工作
get_time.h get_time.c----获得服务器的时间
http_session.h http_session.c----处理一次HTTP会话
以下是各文件源码:
webserver.c:

/*
* file:webserver.c
*/
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<strings.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include"get_time.h"
#include"init_socket.h"
#include"http_session.h"
intmain(intargc,char*argv[])
{
intlisten_fd;
intconnect_fd;
structsockaddr_inserver_addr;
structsockaddr_inclient_addr;
bzero(&server_addr,sizeof(structsockaddr_in));
bzero(&client_addr,sizeof(structsockaddr_in));
if(init_socket(&listen_fd,&server_addr)==-1)
{
perror("init_socket() error. in webserver.c");
exit(EXIT_FAILURE);
}
socklen_taddrlen=sizeof(structsockaddr_in);
pid_tpid;
while(1)
{
if((connect_fd=accept(listen_fd,(structsockaddr*)&client_addr,&addrlen))==-1)
{
perror("accept() error. in webserver.c");
continue;
}
if((pid=fork())>0)
{
close(connect_fd);
continue;
}
elseif(pid==0)
{
close(listen_fd);
printf("pid %d process http session from %s : %d\n",getpid(),inet_ntoa(client_addr.sin_addr),htons(client_addr.sin_port));
if(http_session(&connect_fd,&client_addr)==-1)
{
perror("http_session() error. in webserver.c");
shutdown(connect_fd,SHUT_RDWR);
printf("pid %d loss connection to %s\n",getpid(),inet_ntoa(client_addr.sin_addr));
exit(EXIT_FAILURE);/* exit from child process, stop this http session */
}
printf("pid %d close connection to %s\n",getpid(),inet_ntoa(client_addr.sin_addr));
shutdown(connect_fd,SHUT_RDWR);
exit(EXIT_SUCCESS);
}
else
{
perror("fork() error. in webserver.c");
exit(EXIT_FAILURE);
}
}
shutdown(listen_fd,SHUT_RDWR);
return0;
}
init_socket.h
/*
* file:init_socket.h
*/
#ifndefINIT_SOCKET_H
#defineINIT_SOCKET_H
#include<netinet/in.h>
#defineBACKLOG 20/* length of listening queue on socket */
#definePORT 8080/* web server listening port */
/* initialize the socket on server, include below
socket();
bind();
listen();
*/
/* listen_fd : the web server listen file decriptor
server_addr: the web server ipv4 address
RETURNS: success on 0, error on -1
*/
intinit_socket(int*listen_fd,structsockaddr_in*server_addr);
#endif
init_socket.c
/*
* file:init_socket.c
*/
#include<stdio.h>
#include<strings.h>
#include<unistd.h>
#include<netinet/in.h>
#include<sys/types.h>
#include<sys/socket.h>
#include"init_socket.h"
intinit_socket(int*listen_fd,structsockaddr_in*server_addr)
{
if((*listen_fd=socket(AF_INET,SOCK_STREAM,0))==-1)
{
perror("socket() error. in init_socket.c");
return-1;
}
/* set reuse the port on server machine */
intopt=SO_REUSEADDR;
if(setsockopt(*listen_fd,SOL_SOCKET,SO_REUSEADDR,&opt,sizeof(opt))==-1)
{
perror("setsockopt() error. in init_socket.c");
return-1;
}
server_addr->sin_family=AF_INET;
server_addr->sin_port=htons(PORT);
server_addr->sin_addr.s_addr=htonl(INADDR_ANY);
if(bind(*listen_fd,(structsockaddr*)server_addr,sizeof(structsockaddr_in))==-1)
{
perror("bind() error. in init_socket.c");
return-1;
}
if(listen(*listen_fd,BACKLOG)==-1)
{
perror("listen() error. in init_socket.c");
return-1;
}
return0;
}
get_time.h
/*
* file: get_time.h
*/
#ifndefGET_TIME_H
#defineGET_TIME_H
#defineTIME_BUFFER_SIZE 40/* buffer size of time_buffer */
char*get_time_str(char*time_buf);
#endif
get_time.c
/*
* file:get_time.c
*/
#include<time.h>
#include<stdio.h>
#include<string.h>
#include"get_time.h"
/* get the time on server,
return: the ascii string of time , NULL on error
argument: time_buf the buffer to store time_string
*/
char*get_time_str(char*time_buf)
{
time_tnow_sec;
structtm*time_now;
if(time(&now_sec)==-1)
{
perror("time() in get_time.c");
returnNULL;
}
if((time_now=gmtime(&now_sec))==NULL)
{
perror("localtime in get_time.c");
returnNULL;
}
char*str_ptr=NULL;
if((str_ptr=asctime(time_now))==NULL)
{
perror("asctime in get_time.c");
returnNULL;
}
strcat(time_buf,str_ptr);
returntime_buf;
}
http_session.c
/*
* file: http_session.h
*/
#ifndefHTTP_SESSION_H
#defineHTTP_SESSION_H
#include<netinet/in.h>
#defineRECV_BUFFER_SIZE 1024/* 1KB of receive buffer */
#defineSEND_BUFFER_SIZE 1050000/* 1.xMB of send buffer */
#defineURI_SIZE 128/* length of uri request from client browse */
#defineTIME_OUT_SEC 10/* select timeout of secend */
#defineTIME_OUT_USEC 0/* select timeout of usecend */
#defineFILE_OK 200
#defineFILE_FORBIDEN 403/* there are no access permission*/
#defineFILE_NOT_FOUND 404/* file not found on server */
#defineUNALLOW_METHOD 405/* un allow http request method*/
#defineFILE_TOO_LARGE 413/* file is too large */
#defineURI_TOO_LONG 414/* */
#defineUNSUPPORT_MIME_TYPE 415
#defineUNSUPPORT_HTTP_VERSION 505
#defineFILE_MAX_SIZE 1048576/* 1MB the max siee of file read from hard disk */
#defineALLOW"Allow:GET"/* the server allow GET request method*/
#defineSERVER"Server:Mutu(0.1 Alpha)/Linux"
/* if the connect protocol is http then this function deal with it */
inthttp_session(int*connect_fd,structsockaddr_in*client_addr);
/* if http protocol return 1, else return 0 */
intis_http_protocol(char*msg_from_client);
/* get the request header's uri */
char*get_uri(char*req_header,char*uri_buf);
/* get the uri status,access return 0, not exist return 1, permission deny return 2, error return -1 */
intget_uri_status(char*uri);
/* get the mime type of the file request in uri from client's browse */
char*get_mime_type(char*uri);
/* read the file which requested by client in uri ,and store in entity_buf.
success return bytes readed,error return -1
*/
intget_file_disk(char*uri,unsignedchar*entity_buf);
/* set http replay header's status:
200:ok
404:file not found
*/
intset_rep_status();
intset_error_information(unsignedchar*send_buf,interrorno);
intreply_normal_information(unsignedchar*send_buf,unsignedchar*file_buf,intfile_size,char*mime_type);
#endif
如何访问该服务器呢?
首先你要知道运行服务器主机的IP,在服务器主机上输入如下命令(需要超级用户权限):
ifconfig

如果你的是以太网(ethernet),那么会看到这样一行
inet addr:xxx.xxx.xxx broadcast:xxx.xxx.xxx.xxx mask:255.xxx.xxx.xxx
xxx代表数字(000-255),第一个inet addr后面的数字便是你的网卡地址。
如果你是在本机进行测试,那IP地址可以直接用127.0.0.1(回环地址,localhost)
取得服务器的IP后,用你喜欢的一款浏览器便可以访问WEB SERVER的内容了。
方法为:在浏览器的地址栏内输入:
http://xxx.xxx.xxx.xxx:8080
回车,即可(xxx.xxx.xxx.xxx无刚取得的服务器IP地址,8080为预设的端口)。
C/C++编程日记:用C语言实现的简单Web服务器(Linux),全代码分享!的更多相关文章
- Socket网络编程--简单Web服务器(1)
这一次的Socket系列准备讲Web服务器.就是编写一个简单的Web服务器,具体怎么做呢?我也不是很清楚流程,所以我找来了一个开源的小的Web服务器--tinyhttpd.这个服务器才500多行的代码 ...
- Socket网络编程--简单Web服务器(6)
本来是想实现ssl连接的,但是弄了好久都不成功,就索性不做了,等以后有能力再做了.所以这一小节就是本次的最后一节了.就简单的说几个注意点. 1.加个配置文件 使用单例模式,使用一个类,该类保存一些信息 ...
- 伯克利SocketAPI(一) socket的C语言接口/最简单的服务器和对应的客户端C语言实现
1. 头文件 2. API函数 3. 最简单的服务器和对应的客户端C语言实现 3.1 server #include <sys/types.h> #include <sys/sock ...
- Socket网络编程--简单Web服务器(5)
这一小节我们将实现服务器对get和post的请求进行对cgi程序的调用.对于web服务器以前的章节已经实现了对get和post请求的调用接口,接下来给出对应接口的实现. int WebServer:: ...
- C/C++编程笔记:C语言开发球球大作战(源码分享),你想试试吗?
游戏背景 <球球大作战>是Superpop一款自主研du发的免费手机网络游戏. 以玩家间的实时互动PK产生游戏乐趣为设计宗旨,通过简单的规则将玩家操作直接转化为游戏策略,体验智谋碰撞的战斗 ...
- Python网络编程(3)——SocketServer模块与简单并发服务器
主要类型 该模块有四个比较主要的类,其中常用的是 TCPServer 和 UDPServer. 1. TCPServer 2. UDPServer 3. UnixStreamServer,类似于TCP ...
- Socket网络编程--简单Web服务器(2)
上一小节通过阅读开源的Web服务器--tinyhttpd.大概知道了一次交互的请求信息和应答信息的具体过程.接下来我就自己简单的实现一个Web服务器. 下面这个程序只是实现一个简单的框架出来.这次先实 ...
- Socket网络编程--简单Web服务器(3)
上一小节已经实现了浏览器发送请求,然后服务器给出应答信息,然后浏览器显示出服务器发送过来的网页.一切看起来都是那么的美好.这一小节就准备实现可以根据地址栏url的不同来返回指定的网页.目前还不考虑带参 ...
- Socket网络编程--简单Web服务器(4)
上一小节已经实现了对图片的传输,接下来就是判断文件是否为js,css,png等格式.我们增加一个函数用于判断格式 int WebServer::get_filetype(char *type,char ...
随机推荐
- java实现内网通信
package newTest; import java.io.OutputStream; import java.net.ServerSocket; import java.net.Socket; ...
- [LeetCode]15. 三数之和(数组)(双指针)
题目 给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?找出所有满足条件且不重复的三元组. 注意:答案中不可以包含重复的三 ...
- rocketmq-console修改logo,修改ip,修改port及完整编译安装图文版
一.下载源码到本地 这里使用IDEA,作为编译工具 https://gitee.com/mrliuNumberOne/rocketmq-externals.git 导入成功后如图: 二.Maven编译 ...
- apisix网关-构建docker镜像构建及插件化开发
高能劝退:lua开发,适合小白看!!! 前段时间有个项目,用的java程序做网关,压测tps只有1k多点,惨不忍睹. 后来公司有个大佬改用apisix做网关,tps飙升到1w多. 于是对神奇的apis ...
- springboot 配置和使用过滤器
首先在Application文件中添加注解@ServletComponentScan自动扫描当前类的同包以及子包,这样才能将filter装入bean package com.example.acade ...
- elasticsearch备份和还原(基于hdfs)
备份和还原,为什么elasticsearch还需要备份呢,明明可以设置副本做到高可用,那怕啥呢? 其实在实际的生产环境中,一般最终的结果数据都是要备份的,这样的做的目的,就是能够以最快的速度还原数据, ...
- 容器云平台No.8~kubernetes负载均衡之ingress-nginx
Ingress 是什么? Ingress 公开了从集群外部到集群内服务的 HTTP 和 HTTPS 路由. 流量路由由 Ingress 资源上定义的规则控制. 可以将 Ingress 配置为服务提供外 ...
- CVE-2020-0796(Windows SMBv3) RCE漏洞复现
CVE-2020-0796 攻击机:win10:192.168.205.1 靶机win10:192.168.205.132 关闭defender防火墙 0x01 影响版本 Windows 10 190 ...
- 初探JVM
JVM探究 请你谈谈你对JVM的理解?java8虚拟机和之前的变化更新? 什么是OOM,什么是栈溢出StackOverFlowError?怎么分析? JVM的常用调优参数? 内存快照如何抓取,怎么分析 ...
- Spring Boot 第三弹,一文带你了解日志如何配置?
前言 日志通常不会在需求阶段作为一个功能单独提出来,也不会在产品方案中看到它的细节.但是,这丝毫不影响它在任何一个系统中的重要的地位. 今天就来介绍一下Spring Boot中的日志如何配置. Spr ...