Linux 项目开发 WEB服务器开发_基于linux的web服务器的设计与开发

2.基于多线程的并发服务器模型

创建流式(基于TCP协议)套接字(socket)
设置套接字选项(setsockopt)
准备地址并绑定(bind)
启动侦听(listen)
等待并接受连接请求(accept)<---+ <-连接请求- 浏览器
                            |
    创建子线程持有accept返回  |
    的连接套接字与浏览器通信----+
        接收并解析来自浏览器的HTTP请求<--+ <-浏览器
        找到浏览器请求的资源            |
        组织HTTP响应并发送给浏览器------+  ->浏览器

3.用多个功能模块组成完整的项目

mime模块:文件扩展名与内容类型映射表
mime.h
http模块:实现HTTP协议
http.h、http.c
套接字模块:实现网络通信
socket.h、socket.c
资源模块:查找文件资源
resource.h、resource.c
客户机模块:实现处理客户机业务的子线程
client.h、client.c
信号模块:初始化信号处理
signals.h、signals.c
服务器模块:实现服务器的主线程
server.h、server.c
主模块:实现main函数
main.c
构建脚本:makefile

二、源代码

1、http模块:实现HTTP协议

代码:http.h

// 声明与HTTP协议有关的数据类型和函数
#ifndef \_HTTP\_H
#define \_HTTP\_H
#include <limits.h>
#include <sys/types.h>
// HTTP请求
typedef struct tag_HttpRequest {
 
    char method[32];         // 方法
    char path[PATH_MAX + 1]; // 路径
    char protocol[32];       // 协议
    char connection[32];     // 连接
}   HTTP_REQUEST;
// 解析请求
int parseRequest(char const\* req,
    HTTP_REQUEST\* hreq);
// HTTP响应
typedef struct tag_HttpRespond {
 
    char  protocol[32];   // 协议
    int   status;         // 状态
    char  desc[256];      // 描述
    char  type[32];       // 类型
    off_t length;         // 长度
    char  connection[32]; // 连接
}   HTTP_RESPOND;
// 构造响应头
void constructHead(HTTP_RESPOND const\* hres,
    char\* head);
#endif // \_HTTP\_H

gcc -c http.h

代码:http.c

// 定义与HTTP协议有关的函数
#include <unistd.h>
#include <sys/syscall.h>
#include <stdio.h>
#define \_\_USE\_GNU
#include <string.h>
#include <time.h>
#include "http.h"
// 解析请求
int parseRequest(char const\* req,
    HTTP_REQUEST\* hreq) {
 
    /\*
 \* GET / HTTP/1.1\r\n
 \* Host: localhost:8000\r\n
 \* User-Agent: Mozilla/5.0\r\n
 \* Accept: text/html\r\n
 \* Connection: keep-alive\r\n\r\n
 \*/
    sscanf(req, "%s%s%s", hreq->method,
        hreq->path, hreq->protocol);
    char\* connection = strcasestr(
        req, "connection");
    if (connection)
        sscanf(connection, "%\*s%s",
            hreq->connection);
    printf("%d.%ld> [%s][%s][%s][%s]\n",
        getpid(), syscall(SYS_gettid),
        hreq->method, hreq->path,
        hreq->protocol, hreq->connection);
    if (strcasecmp(hreq->method, "get")) {
 
        printf("%d.%ld> 无效方法\n",
            getpid(), syscall(SYS_gettid));
        return -1;
    }
    if (strcasecmp(hreq->protocol, "http/1.0") &&
        strcasecmp(hreq->protocol, "http/1.1")) {
 
        printf("%d.%ld> 无效协议\n",
            getpid(), syscall(SYS_gettid));
        return -1;
    }
    return 0;
}
// 构造响应头
void constructHead(HTTP_RESPOND const\* hres,
    char\* head) {
 
    /\*
 \* HTTP/1.1 200 OK\r\n
 \* Server: Tarena WebServer 1.0\r\n
 \* Date: Tue, 14 Jan 2020 01:39:00 GMT\r\n
 \* Content-Type: text/html\r\n
 \* Content-Length: 2048\r\n
 \* Connection: keep-alive\r\n\r\n
 \*/
    char dateTime[32];
    time_t now = time(NULL);
    strftime(dateTime, sizeof(dateTime),
        "%a, %d %b %Y %T GMT", gmtime(&now));
    sprintf(head,
        "%s %d %s\r\n"
        "Server: Tarena WebServer 1.0\r\n"
        "Date: %s\r\n"
        "Content-Type: %s\r\n"
        "Content-Length: %ld\r\n"
        "Connecion: %s\r\n\r\n",
        hres->protocol, hres->status, hres->desc,
        dateTime, hres->type, hres->length,
        hres->connection);
}

2、套接字模块:实现网络通信

代码:socket.h

// 声明网络通信函数
#ifndef \_SOCKET\_H
#define \_SOCKET\_H
// 初始化套接字
int initSocket(short port);
// 接受客户机连接
int acceptClient(void);
// 接收请求
char\* recvRequest(int conn);
// 发送响应头
int sendHead(int conn, char const\* head);
// 发送响应体
int sendBody(int conn, char const\* path);
// 终结化套接字
void deinitSocket(void);
#endif // \_SOCKET\_H

代码:socket.c

// 定义网络通信函数
#include <unistd.h>
#include <fcntl.h>
#include <sys/syscall.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "socket.h"
static int s_sock = -1; // 侦听套接字
// 初始化套接字
int initSocket(short port) {
 
    printf("%d> 创建套接字\n", getpid());
    if ((s_sock = socket(PF_INET, SOCK_STREAM,
        0)) == -1) {
 
        perror("socket");
        return -1;
    }
    printf("%d> 设置套接字\n", getpid());
    int on = 1;
    if (setsockopt(s_sock, SOL_SOCKET,
        SO_REUSEADDR, &on, sizeof(on)) == -1) {
 
        perror("setsockopt");
        return -1;
    }
    printf("%d> 绑定端口号\n", getpid());
    struct sockaddr_in addr;
    addr.sin_family = AF_INET;
    addr.sin_port = htons(port);
    addr.sin_addr.s_addr = INADDR_ANY;
    if (bind(s_sock, (struct sockaddr\*)&addr,
        sizeof(addr)) == -1) {
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值