目录
一、HTTP相关概念
二、客服端请求
1、请求首部
2、 响应首部
三、线程实现HTTP并发服务器
一、HTTP相关概念
1、HTTP,全称Hyper Text Transfer Protocol,用于万维网(world wide web)进行超文本学习的传输协议
2、HTTP属于应用层,对应的传输层是TCP
3、http是基于BS模型,即浏览器服务器模型,主要完成的是客户端请求端和服务器相应端
4、Hypertext Markup Language 超文本标记语言,属于标签式语言,能够被浏览器所识别
二、客服端请求
1、请求首部
1)任何一个http请求都由三部分组成:请求首部、请求主体、请求数据
2)对于客户端请求协议包而言,一般没有请求数据,具体格式如下
2、 响应首部
1)响应首部也是由三部分组成,分别是响应头、响应主体、响应数据
2)对于响应首部而言,必须由响应数据
3)http的响应代号:1XX (信息状态错误) 2XX(成功) 、3XX(重定向状态码)、4XX(客户端错误)、5XX(服务器出错)
三、线程实现HTTP并发服务器
有BUG
#include <myhead.h>
#define SER_PORT 80
#define SER_IP "192.168.0.106"void do_notfound(int newfd);
void do_response(int newsfd, const char *url);
int send_head(int newfd, int file_size);
void send_html(int newfd, int fd);
void do_server_error(int newfd);
int get_one_line(int newfd, char msg[]);// 线程用信息初传输结构体
struct Info
{struct sockaddr_in cin;int newsfd;
};// 线程处理函数
void *deal_cli_msg(void *arg)
{int newsfd = ((struct Info *)arg)->newsfd;struct sockaddr_in cin = ((struct Info *)arg)->cin;while (1){// 5、数据收发char buf[128] = "";// 获取请求头部get_one_line(newsfd, buf);// 获取请求方法char mathod[5] = ""; // 存储请求方法strcpy(mathod, strtok(buf, " ")); // 分割字符串并将分割下来的第一个字符串存入mathodchar url[50] = "";char n_url[256] = "";strcpy(url, strtok(NULL, " ")); // 在上一次分割的基础上,将第二个字符串存入urlif (strcmp(mathod, "GET") == 0){// 完成相应工作// 重组urlsnprintf(n_url, sizeof(n_url), "./htmls%s", url);// 判断要访问的文件是否存在if (access(n_url, F_OK) == -1){// 返回错误信息do_notfound(newsfd);break;}else{// 响应do_response(newsfd, n_url);}}else{// 非法请求// do_illigal(newsfd);}}pthread_exit(EXIT_SUCCESS);
}int get_one_line(int newfd, char msg[])
{char buf = '\0';int i = 0;while (1){int res = recv(newfd, &buf, 1, 0);if (0 == res){break;}else if (1 == res){if (buf == '\n'){break;}else{msg[i] = buf;i++;}}else{perror("recv error");return -1;}}msg[i] = '\0';
}// 响应函数
void do_response(int newsfd, const char *url)
{// 1、以只读的形式打开文件int fd = open(url, O_RDONLY);if (-1 == fd){/* 服务器内部错误 */do_server_error(newsfd);return;}// 2、获取文件大的长度int file_size = lseek(fd, 0, SEEK_END);// 3、封装响应头部发送给客服端int res = send_head(newsfd, file_size);if (0 == res){return;}// 4、发送html文档给客服端lseek(fd, 0, SEEK_SET);// send_html(newfd,fd);printf("响应成功\n");close(fd); // 关闭文件
}// 头部发送函数
int send_head(int newfd, int file_size)
{char *head = "HTTP/1.1 200 OK\r\nContent-Type:text/html\r\n";char n_head[512] = "";snprintf(n_head, sizeof(n_head), "%sContent-Length:%d\r\n\r\n", head, file_size);printf("n_head = %s\n", n_head);// 将头部发送个客服端int res = send(newfd, n_head, strlen(n_head), 0);return res;
}// 发送文件函数
void send_html(int newfd, int fd)
{char buf[128] = "";while (1){/* 从文件中读取数据 */int res = read(fd, buf, sizeof(buf));if (-1 == res){/* 服务器内部出错 */do_server_error(newfd);return;}else if (0 == res){/* 读取结束 */break;}// 将数据发送给客服端send(newfd, buf, res, 0);}
}// 服务器内部出错
void do_server_error(int newfd)
{char *msg = "HTTP/1.1 502 SERVER_ERROR\r\nContrnt-Type-text/html\r\n";// 向客服端发送错误信息send(newfd, msg, strlen(msg), 0);// 定义要发送的数据char *msg_html = "<!DOCTYPE html>\n\
<html lang=\" en \">\n\
<head>\n\<meta charset=\" UTF -\8 \">\n\<meta name=\" viewport \" content=\" width = device - width,\initial - scale = 1.0 \">\n\<title>502 SERVER_ERROR</title>\n\
</head>\n\
<body>\n\<h1>服务器内部出错</h1>\n\<p>文件描述符打开出错</p>\n\
</body>\n\
</html>";send(newfd, msg_html, strlen(msg_html), 0);
}// 对象访问界面不存在
void do_notfound(int newfd)
{char *msg = "";// 向客服端发送错误信息send(newfd, msg, strlen(msg), 0);// 定义要发送的数据char *msg_html = "HTTP/1.1 404 NOT_FOUND\r\nContrnt-Type-text/html\r\n\< !DOCTYPE html >\n<html lang = \" en \">\n<head>\n < meta charset = \" UTF -8 \">\n\<meta name=\" viewport \" content=\" width = device - width,initial - scale = 1.0 \">\n\<title>502 SERVER_ERROR</title>\n\
</head>\n\
<body>\n\<h1>404 NOT_FOUND</h1>\n\<p>e界面不存在</p>\n\
</body>\n\
</html>";
}int main(int argc, char const *argv[])
{// 1、创建套接字int sfd = socket(AF_INET, SOCK_STREAM, 0);// 参数1:ipv4的网络通信// 参数2:TCP通信方式// 参数3:默认使用一个协议if (sfd == -1){perror("socket error");return -1;}printf("socket success, sfd = %d\n", sfd); // 3// 2、为套接字绑定ip地址和端口号// 2.1 填充地址信息结构体struct sockaddr_in sin;sin.sin_family = AF_INET; // 通信域sin.sin_port = htons(SER_PORT); // 端口号sin.sin_addr.s_addr = inet_addr(SER_IP); // ip地址// 2.2 绑定if (bind(sfd, (struct sockaddr *)&sin, sizeof(sin)) == -1){perror("bind error");return -1;}printf("bind success\n");// 3、将套接字设置为被动监听状态,用于接收if (listen(sfd, 128) == -1){perror("listen error");return -1;}printf("listen success\n");// 4、阻塞等待客户端的连接请求// 4.1 定义n变量用于e接收客服端的信息struct sockaddr_in cin;socklen_t addrlen = sizeof(cin);while (1) // 循环服务器{// 4.2 接收连接int newsfd = accept(sfd, (struct sockaddr *)&cin, &addrlen);if (newsfd == -1){perror("accept error");return -1;}printf("[%s:%d]:accept on\n", inet_ntoa(cin.sin_addr), ntohs(cin.sin_port));struct Info cli_info = {cin,newsfd};// 创建分支线程用于处理客服端请求pthread_t tid = -1;if (pthread_create(&tid, NULL, deal_cli_msg, &cli_info) != 0){/* 出错 */printf("tid create error");return -1;}//将线程设置为n分离态pthread_detach(tid);}close(sfd);return 0;
}