c++实现简单的web服务器搭建

c++实现简单的web服务器搭建本文使用c++socket编程进行简单的web服务器搭建,出来了GET请求

大家好,又见面了,我是你们的朋友全栈君。

c++简单的web服务器搭建

web 服务器与 Http 协议

Web 浏览器(Web Browser)是一个用于文档检索和显示的客户应用程序,并通过超文本传输协议

Http(Hyper Text Transfer Protocol)与 Web 服务器相连。

通用的、低成本的浏览器节省了两层结构的 C/S 模式客户端软件的开发和维护费用。

HTTP 协议工作流程

  1. 首先客户机与服务器需要建立连接。只要单击某个超级链接,HTTP 的工作就开始了。

  2. 建立连接后,客户机发送一个请求给服务器,请求方式的格式为:统一资源标识符(URL)、 协议版本号,后边是 MIME 信息:包括请求修饰符、客户机信息和可能的内容。

  3. 服务器接到请求后,给予相应的响应信息,其格式为一个状态行,包括信息的协议版本号、 一个成功或错误的代码,后边是 MIME 信息包括服务器信息、实体信息和可能的内容。

  4. 客户端接收服务器所返回的信息通过浏览器显示在用户的显示屏上,然后客户机与服务器断开连接。

客户端请求方法

方法 描述
get 请求读一个万维网页
head 请求读一个万维网页的頭部
put 请求存储一个万维网页
post 附加一个命名的资源
delete 删除万维网页
link 连接两个已有资源
unlink 切断两个已有资源间的连接

服务器的搭建

服务器搭建需要对winsock版本以及套接字进行初始化,接着将本机的信息包括IP地址,端口进行绑定。

这样不仅本机,在局域网内的机器也是可以对服务器进行请求数据。

客户端请求的解析(请求方式以及请求资源的解析)

客户端请求的解析需要通过获取客户端的请求头来进行解析,如下图所示,Request URL是客户端请求的地址,Request Method 为 请求的方式,因此只需要拿到客户端的请求头解析出请求内容以及请求方式即可。

在这里插入图片描述

HTTP 消息结构

HTTP是基于客户端/服务端(C/S)的架构模型,通过一个可靠的链接来交换信息,是一个无状态的请求/响应协议。

一个HTTP”客户端”是一个应用程序(Web浏览器或其他任何客户端),通过连接到服务器达到向服务器发送一个或多个HTTP的请求的目的。

一个HTTP”服务器”同样也是一个应用程序(通常是一个Web服务,如Apache Web服务器或IIS服务器等),通过接收客户端的请求并向客户端发送HTTP响应数据。

HTTP使用统一资源标识符(Uniform Resource Identifiers, URI)来传输数据和建立连接。

一旦建立连接后,数据消息就通过类似Internet邮件所使用的格式[RFC5322]和多用途Internet邮件扩展(MIME)[RFC2045]来传送。

客户端请求消息

客户端发送一个HTTP请求到服务器的请求消息包括以下格式:请求行(request line)、请求头部(header)、空行和请求数据四个部分组成,下图给出了请求报文的一般格式。

在这里插入图片描述

服务器响应消息

HTTP响应也由四个部分组成,分别是:状态行、消息报头、空行和响应正文。

在这里插入图片描述

HTTP 响应头信息

HTTP请求头提供了关于请求,响应或者其他的发送实体的信息。

  1. Content-Encoding
    文档的编码(Encode)方法。只有在解码之后才可以得到Content-Type头指定的内容类型。利用gzip压缩文档能够显著地减少HTML文档的下载时间。Java的GZIPOutputStream可以很方便地进行gzip压缩,但只有Unix上的Netscape和Windows上的IE 4、IE 5才支持它。因此,Servlet应该通过查看Accept-Encoding头(即request.getHeader(“Accept-Encoding”))检查浏览器是否支持gzip,为支持gzip的浏览器返回经gzip压缩的HTML页面,为其他浏览器返回普通页面。

  2. Content-Length
    表示内容长度。只有当浏览器使用持久HTTP连接时才需要这个数据。如果你想要利用持久连接的优势,可以把输出文档写入 ByteArrayOutputStream,完成后查看其大小,然后把该值放入Content-Length头,最后通过byteArrayStream.writeTo(response.getOutputStream()发送内容。

  3. Content-Type
    表示后面的文档属于什么MIME类型。Servlet默认为text/plain,但通常需要显式地指定为text/html。由于经常要设置Content-Type,因此HttpServletResponse提供了一个专用的方法setContentType。

  4. Date
    当前的GMT时间。你可以用setDateHeader来设置这个头以避免转换时间格式的麻烦。

  5. Expires
    应该在什么时候认为文档已经过期,从而不再缓存它?

  6. Last-Modified
    文档的最后改动时间。客户可以通过If-Modified-Since请求头提供一个日期,该请求将被视为一个条件GET,只有改动时间迟于指定时间的文档才会返回,否则返回一个304(Not Modified)状态。Last-Modified也可用setDateHeader方法来设置。

  7. Location
    表示客户应当到哪里去提取文档。Location通常不是直接设置的,而是通过HttpServletResponse的sendRedirect方法,该方法同时设置状态代码为302。

  8. Refresh
    表示浏览器应该在多少时间之后刷新文档,以秒计。除了刷新当前文档之外,你还可以通过setHeader(“Refresh”, “5; URL=http://host/path”)让浏览器读取指定的页面。
    注意这种功能通常是通过设置HTML页面HEAD区的<META HTTP-EQUIV=“Refresh” CONTENT=”5;URL=http://host/path”>实现,这是因为,自动刷新或重定向对于那些不能使用CGI或Servlet的HTML编写者十分重要。但是,对于Servlet来说,直接设置Refresh头更加方便。

    注意Refresh的意义是”N秒之后刷新本页面或访问指定页面”,而不是”每隔N秒刷新本页面或访问指定页面”。因此,连续刷新要求每次都发送一个Refresh头,而发送204状态代码则可以阻止浏览器继续刷新,不管是使用Refresh头还是<META HTTP-EQUIV=“Refresh” …>。

    注意Refresh头不属于HTTP 1.1正式规范的一部分,而是一个扩展,但Netscape和IE都支持它。

  9. Server
    服务器名字。Servlet一般不设置这个值,而是由Web服务器自己设置。

  10. Set-Cookie
    设置和页面关联的Cookie。Servlet不应使用response.setHeader(“Set-Cookie”, …),而是应使用HttpServletResponse提供的专用方法addCookie。参见下文有关Cookie设置的讨论。

HTTP 状态码

HTTP 响应头信息HTTP content-type
HTTP 状态码
当浏览者访问一个网页时,浏览者的浏览器会向网页所在服务器发出请求。当浏览器接收并显示网页前,此网页所在的服务器会返回一个包含 HTTP 状态码的信息头(server header)用以响应浏览器的请求。

HTTP 状态码的英文为 HTTP Status Code。
常见的 HTTP 状态码:
200 – 请求成功
301 – 资源(网页等)被永久转移到其它URL
404 – 请求的资源(网页等)不存在
500 – 内部服务器错误

详细设计

对客户端的请求头进行解析

recv(client_fd,buff,99,0)  

服务器的建立(包括初始化winsock版本,初始化套接字)

    char recbuf[2048];
    WSADATA wsaData;
    int iResult = WSAStartup(MAKEWORD(2,2), &wsaData);
    SOCKADDR_IN svr_addr, cli_addr;
    int sin_len = sizeof(cli_addr);

    int sock = socket(AF_INET, SOCK_STREAM, 0);
    if (sock < 0)
    {
        cout<< "无法开启socket";
        exit(0);
    }

    int port = 81;
    svr_addr.sin_family = AF_INET;
    svr_addr.sin_addr.s_addr = INADDR_ANY;
    svr_addr.sin_port = htons(port);

    if (bind(sock, ( SOCKADDR *) &svr_addr, sizeof(svr_addr)) == -1)
    {
        closesocket(sock);
        cout<<"无法绑定\n";
    }

    listen(sock, 5);

获得客户端的请求头

        if((numbytes = recv(client_fd,buff,99,0)) == -1)
        {
            perror("recv");
            exit(1);
        }

对请求头进行解析(分析出其中的请求方式)

    char data[1000];
    char cd[500];
    char args[500];
    strcpy(args,"./");
    if(sscanf(buff, "%s%s", cd, args+2)!=2)
    {
        return;
    }
 //  cout<<" 111 "<<cd<<endl;
  //  cout<<" 111 "<<args<<endl;
    if(strcmp(cd,"GET")!=0)
    {
        cout<<"请求类型错误"<<endl;
        return;
    }

对请求的资源进行分析

char response[] = "HTTP/1.1 200 OK\r\n"
                  "Content-Type: text/html; charset=UTF-8\r\n\r\n"
                  "<!DOCTYPE html>"
                  "<html lang=\"zh-CN\">"
                  "<head>"
                  "<meta charset=\"utf-8\">"
                  "<title>Hello World</title>"
                  "<style>body {width: 35em;margin: 200px auto;font-family: Tahoma, Verdana, Arial, sans-serif;}"
                  "</style>"
                  "</head>"
                  "<body>"
                  "<p>This is a simple webserver<p>"
                  "<p><em>And this html does not support ZH-CN</em></p>"
                  "</body></html>\r\n";


string s;
    char str[100];
    cout<<args<<endl;
    if(!strcmp(args,".//index.html"))
    {
        FILE * fp;
        if((fp=fopen(args,"rt"))==NULL)
        {
          //  cout<<"不行"<<endl;
        }
        else
        {
        //    cout<<"行"<<endl;
            while((fgets(str,1024,fp))!=NULL)
            {
                s += str;
                cout<<s<<endl;
            }
          //  cout<<"我giao"<<endl;
        }
        fclose(fp);
    cout<<s<<endl;
   // cout<<response<<endl;
    cout<< send(client_fd, r, sizeof(r), 0)<<"Bytes已发送" <<endl; // 发送请求的资源
    }else{
        cout<<response<<endl;
        cout<< send(client_fd, response, sizeof(response), 0)<<"Bytes已发送" <<endl;
    }

完整代码(这里只是对GET请求进行处理)

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include<bits/stdc++.h>
#include <WinSock2.h>
#include<stdio.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<assert.h>
#include<unistd.h>

using namespace std;
int client_fd;
int numbytes;
char buff[100];
char response[] = "HTTP/1.1 200 OK\r\n"
                  "Content-Type: text/html; charset=UTF-8\r\n\r\n"
                  "<!DOCTYPE html>"
                  "<html lang=\"zh-CN\">"
                  "<head>"
                  "<meta charset=\"utf-8\">"
                  "<title>Hello World</title>"
                  "<style>body {width: 35em;margin: 200px auto;font-family: Tahoma, Verdana, Arial, sans-serif;}"
                  "</style>"
                  "</head>"
                  "<body>"
                  "<p>This is a simple webserver<p>"
                  "<p><em>And this html does not support ZH-CN</em></p>"
                  "</body></html>\r\n";


char r[] = "HTTP/1.1 200 OK\r\n""Content-Type: text/html; charset=UTF-8\r\n\r\n""<!DOCTYPE html><html><head><meta charset='utf-8'><</head><body>    <h1>yyy</h1>    <p>111</p></body></html>";

void solve()
{
    char data[1000];
    char cd[500];
    char args[500];
    strcpy(args,"./");
    if(sscanf(buff, "%s%s", cd, args+2)!=2)
    {
        return;
    }
 //  cout<<" 111 "<<cd<<endl;
  //  cout<<" 111 "<<args<<endl;
    if(strcmp(cd,"GET")!=0)
    {
        cout<<"请求类型错误"<<endl;
        return;
    }
    // int fd = open(args,O_RDONLY);
    // int x = send(client_fd,response, sizeof(response)-1,0);
//    send(client_fd,fd,NULL,2500);
    string s;
    char str[100];
    cout<<args<<endl;
    if(!strcmp(args,".//index.html"))
    {
        FILE * fp;
        if((fp=fopen(args,"rt"))==NULL)
        {
          //  cout<<"不行"<<endl;
        }
        else
        {
        //    cout<<"行"<<endl;
            while((fgets(str,1024,fp))!=NULL)
            {
                s += str;
                cout<<s<<endl;
            }
          //  cout<<"我giao"<<endl;
        }
        fclose(fp);
    cout<<s<<endl;
   // cout<<response<<endl;
    cout<< send(client_fd, r, sizeof(r), 0)<<"Bytes已发送" <<endl;
    }else{
        cout<<response<<endl;
        cout<< send(client_fd, response, sizeof(response), 0)<<"Bytes已发送" <<endl;
    }
    //  FILE * fp = open(args,O_RDONLY);

    // cout<< pBuf <<endl;
    return;
}
int main()
{
    char recbuf[2048];
    WSADATA wsaData;
    int iResult = WSAStartup(MAKEWORD(2,2), &wsaData);
    SOCKADDR_IN svr_addr, cli_addr;
    int sin_len = sizeof(cli_addr);

    int sock = socket(AF_INET, SOCK_STREAM, 0);
    if (sock < 0)
    {
        cout<< "无法开启socket";
        exit(0);
    }

    int port = 81;
    svr_addr.sin_family = AF_INET;
    svr_addr.sin_addr.s_addr = INADDR_ANY;
    svr_addr.sin_port = htons(port);

    if (bind(sock, ( SOCKADDR *) &svr_addr, sizeof(svr_addr)) == -1)
    {
        closesocket(sock);
        cout<<"无法绑定\n";
    }

    listen(sock, 5);
    while (1)
    {
        int len = sizeof(SOCKADDR);
        client_fd = accept(sock, (SOCKADDR *) &cli_addr, &sin_len);
        printf("客户端已连接\n");
        cout<<inet_ntoa(cli_addr.sin_addr)<<endl;
        if (client_fd == -1)
        {
            perror("accept()客户端连接出错");
            continue;
        }
        if((numbytes = recv(client_fd,buff,99,0)) == -1)
        {
            perror("recv");
            exit(1);
        }
        cout<<buff<<endl;

        solve();

        // cout<<response<<endl;
    }
    closesocket(client_fd);
    WSACleanup();
}

运行截图

在这里插入图片描述
在这里插入图片描述

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/140290.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

  • pycharm运行快捷键设置_pycharm保存代码快捷键

    pycharm运行快捷键设置_pycharm保存代码快捷键调试快捷键shift+Alt+F10终止快捷键Ctrl+F2

  • 阿里云服务器开放端口如何设置_阿里云服务器8888端口

    阿里云服务器开放端口如何设置_阿里云服务器8888端口阿里云服务器开放端口阿里云服务器默认是只开放了部分端口,我们部署自己的服务需要监听一下80,8080等端口时,就需要自己设置安全策略,本文介绍如何设置阿里云的安全组,开放需要的端口步骤点击阿里云的的控制台点击进入云服务器点击进入安全组菜单,点击创建安全组按钮,添加一个新的安全组2.进入创建新安全组页面填写一下必要的信息,然后配置访问规则,包括入站和出站,点击手动添加一条,设置开放所有的端口,包括端口和授权对象,点击创建安全组按钮,将创建一条新的安全组出站我们也可以配置,默

  • SpringBoot 介绍「建议收藏」

    SpringBoot 介绍「建议收藏」SpringBoot介绍目录1.简介2.WhySpringBoot1.简介    SpringBoot最开始基于Spring4.0设计,是由Pivotal公司提供的框架。    SpringBoot发展史:2003年RodJohnson成立Interface公司,产品是SpringFramework2004年,Spring框架开源,公司改名为SpringSource2008年,收购ApacheSer

  • LOTO课5:三极管音频放大电路实践[通俗易懂]

    LOTO课5:三极管音频放大电路实践[通俗易懂]我们在项目中经常会遇到音频信号的采集处理,我们今天做一个最简单的音频采集模块。它的电路其实就是在我们上节课的三极管的放大电路上的一个改进,在上一节课三极管放大电路的基础之上,将输出信号换成驻极体话筒,输出端加上截止频率在20KHZ左右的RC低通滤波电路,通过滤波电路来滤除频率在20KHZ以上的噪声信号。上一节课关于三极管放大的文章链接如下:添加链接描述设计的电路原理图如图所示,通过传感器获取一个交流的小信号,经过三级管放大电路放大信号之后再进行输出:等不及打板,手工焊接了一个样品进行试验:通过

  • 计算机二级考试数据结构与算法知识点_计算机二级算法与数据结构视频

    计算机二级考试数据结构与算法知识点_计算机二级算法与数据结构视频按照自己的理解写的解题思路,如有错误希望指正。1.算法的复杂度: ①时间复杂度:执行算法所需的计算工作量(又叫:基本运算次数) ②空间复杂度:执行算法所需的内存 它们是没有任何关系的!!!2.求二叉树序列类题目 要点:前序—根左右 中序—左根右 后序—左右根 例1:已知前序ABCDE,中序BCADE,求后序;同类型,已知任意两个求第三个 解题思路: 由前序知道A是根,结合中序,CB是左子树,DE…

  • VR社交软件测试-AltspaceVR

    VR社交软件测试-AltspaceVR

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号