http服务器_本地简易http服务器

2023-11-19

本地简易http服务器

下面实现一个简单的http服务器,听起来高大上的样子,其实就是网络通信加上http协议。运用上篇的网络编程的基础模型,上篇的网络通信是开两个vs程序,一个做服务器、一个做客户端,互相发送数据。http服务器其实也是同样的道理,只是浏览器用来做客户端,再开一个vs程序作为服务器,监听相应的端口,二者互传数据,数据被封装成http协议格式。
关于http协议:我看了《图解HTTP》这本书,通俗易懂。
大致就是一个请求报文和一个相应报文。

8444ea73822ea3c8e2fa83b0adb4cd63.png

实例:
(通过浏览器访问服务器,查看报文)
请求报文:

6aa5d90bc51714f0696194de978d11eb.png

服务器启动,浏览器访问,这是浏览器发送的请求报文,可见分为很多行,每一行包括内容和属性(若内容、属性不了解含义可查阅相关资料)。该请求报文只有报文首部,报文首部结束后紧接着是空行,报文主体并不是必须的。
(拿到请求报文,应该逐行解析该报文,简单起见,这里不解析,认为只要有连接,就发送响应报文)
响应报文:

97ed58a26167aa8fe048951a2c2843e0.png

HTTP/1.0 200 OK
Content-Type: text/html

//包体file_buf

读取一个项目目录下的index.html文件放入file_buf,发送到浏览器

adf84e7f1f9b10cad33a100f200c7240.png

浏览器接收到数据进行页面显示:

73b1f421a21958955a80b4f432a20373.png

//服务器代码

#include "stdafx.h"
#include 
#include 

#pragma comment(lib, "Ws2_32.lib")
#pragma warning(disable:4996)

#define PORT 12345
#define QUEUE_MAX_COUNT 5
#define BUFF_SIZE 2048 
#define FBUFF_SIZE 1024


int main(int argc, char** argv){
    //socket初始化
    WSADATA wsaData = { 0 };
    int iResult = WSAStartup(MAKEWORD(2, 2), &wsaData);
    if (iResult != 0) {
        wprintf(L"WSAStartup failed: %d\n", iResult);
        return 1;
    }

    int server_fd = socket(AF_INET, SOCK_STREAM, 0);
    if (server_fd == -1) {
        printf("socket创建失败!");
        return 0;
    }

    int port = PORT;
    struct sockaddr_in server_addr;
    memset(&server_addr, 0, sizeof(server_addr));
    // 设置端口,IP,和TCP/IP协议族 
    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(PORT);
    server_addr.sin_addr.s_addr = htonl(INADDR_ANY);

    // 绑定套接字到端口 
    if (bind(server_fd, (struct sockaddr *)&server_addr,
        sizeof(server_addr)) 0) {
        printf("绑定失败!");
        closesocket(server_fd);
        return 0;
    }

    // 启动socket监听请求,开始等待客户端发来的请求 
    if (listen(server_fd, QUEUE_MAX_COUNT) 0) {
        printf("监听失败!");
        closesocket(server_fd);
        return 0;
    }

    printf("http server running on port %d\n", port);

    int client_fd;
    struct sockaddr_in client_addr;
    int client_addr_len = sizeof(client_addr);

    char send_buf[BUFF_SIZE];
    char recv_buf[BUFF_SIZE];

    while (1) {
        // 调用了accept函数,阻塞了程序,直到接收到客户端的请求
        client_fd = accept(server_fd, (struct sockaddr *)&client_addr,
            &client_addr_len);
        if (client_fd 0) {
            printf("接收失败!");
            closesocket(server_fd);
            break;
        }
        printf("accept a client\n");

        // 调用recv函数接收客户端发来的请求信息
        int hello_len = recv(client_fd, recv_buf, BUFF_SIZE, 0);
        //处理发来的请求,这里先不处理

        char file_buf[FBUFF_SIZE];
        memset(file_buf, '`', FBUFF_SIZE);
        FILE* file = fopen("index.html", "r");
        if (file == NULL) {
            printf("文件打开失败!!!\n");
            return -1;
        }

        //计算文件的大小
        fseek(file, 0, SEEK_END);   //将文件指针移动文件结尾
        int size = ftell(file); //求出当前文件指针距离文件开始的字节数

        fseek(file, 0, SEEK_SET);   //将文件指针移动文件头
        while (!feof(file)) {
            //读文件
            fread(file_buf, sizeof(char), size, file);
            if (ferror(file)) {
                perror("Read error");
                break;
            }
        }

        //send
        // 发送响应给客户端
        int len = 0;
        int res = 0;
        //拼装包头
        res = sprintf(send_buf, "HTTP/1.0 200 OK\r\n");
        len += res;
        res = sprintf(send_buf + len, "Content-Type: text/html\r\n");
        len += res;
        res = sprintf(send_buf + len, "\r\n");
        len += res;
        //拼装包体
        strcat(send_buf + len, file_buf);
        //res = sprintf(send_buf + len, "Hello World!\r\n");
        int ll = 0;
        for (int i = 0; i             if (file_buf[i] != '`')
                ll++;
        }
        len += ll;

        send(client_fd, send_buf, len, 0);

        // 关闭客户端套接字
        closesocket(client_fd);
    }

    //关闭客户端socket
    closesocket(server_fd);

    return 0;
}

能力不足、水平有限,若有任何不当之处,恳请指教。

Linux命令
gdb调试过程中,用backtrace查看当前线程调用堆栈

6a2e9654529b7643a931117eb7c545ed.png

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

http服务器_本地简易http服务器 的相关文章

  • webpack 的externals配置

    官网解释 webpack 中的 externals 配置提供了不从 bundle 中引用依赖的方式 解决的是 所创建的 bundle 依赖于那些存在于用户环境 consumer environment 中的依赖 怎么理解呢 意思是如果需要引
  • Linux FTP服务

    目录 一 FTP概述 1 Standard模式 主动模式 2 Passice模式 被动模式 二 搭建和配置FTP服务 1 实验环境 2 软件安装 3 设置匿名用户访问FTP服务 4 开启服务 关闭防火墙和增强型安全功能 5 匿名访问测试 1

随机推荐

  • Spring Data JPA中常用的注解详解

    我们先看看类前面的两个注解 Entity 对实体注释 任何Hibernate映射对象都要有这个注释 Table name 自定义的表名 完成表和实体之间的映射 跟上面的注解默认需要配置的 Id 把这个类里面所在的变量设置为主键Id Gene
  • 单片机生成PDF文件(STM32+LIBHARU)

    之前发过一个帖子 是STM32移植了一个pdf库 不过那个库功能比较简单 有很多东西都没有 比如加载图片和文件加密 这次主要是为了文件加密 翻阅了一下资料 单片机做PDF文件 网上资料不是很多 可以说基本上找不着 github上翻源码 翻了
  • windows查看功耗

    可以在Windows系统中查看功耗的方法有以下几种 使用任务管理器 在Windows桌面上按下Ctrl Alt Delete键 选择任务管理器 然后在 性能 选项卡中单击 电源使用情况 查看功耗信息 使用系统监视器 打开系统监视器 在Win
  • java 图片Base64转成文件流,直接在浏览器访问图片

    因为业务 需要把图片base64转成文件流 然后直接在浏览器访问就可以看见图片 回显图片 author liuhengliang return ModelAndView RequestMapping value image private
  • 基于51单片机的温度烟雾报警系统设计

    功能 本实例是基于51单片机的温度烟雾报警系统 主要硬件由51单片机最小系统 LCD1602液晶屏电路 烟雾检测电路 温度感应电路 蜂鸣器报警电路 ADC0832转换电路 LED指示灯电路和按键电路构成 1 LCD1602液晶屏 第一行显示
  • Colab .ipynb 从本地/云端/GitHub 导入并使用.py文件

    1 下载 上传目标文件 1 从GitHub下载所需repo 下载Code 下载 Zip 文件保存到本地 2 上传至云端Google Drive 上传成功后就能在My Drive看到文件夹了 尝试过直接Clone repo到Colab 但是后
  • python爬取考研网的信息

    今天我们使用python来爬取考研网站的信息 目标网站 https yz chsi com cn zsml queryAction do 使用的库 requests bs4 pandas 这些库统一可以使用pip进行统一安装 pip ins
  • VSCode 插件安装:中文(简体)语言包(附带:不生效解决方案)

    文章目录 VSCode 安装插件 中文 简体 语言包 中文语言包不生效解决方案 打开 命令面板 配置显示语言 选择中文 重启VSCode 效果 中文界面 VSCode 安装插件 中文 简体 语言包 插件市场搜索 中文 选择如下插件安装 Ch
  • Vue3自定义指令之前端水印功能实现

    一 前置知识 Vue 中的自定义指令 先来说说 vue2和vue3中自定义全局指令的区别 相同点 指令的应用场景 原理是一致的 不同点 生命周期钩子函数名 指令定义的格式不一样 vue2中自定义全局指令 定义 注册一个全局自定义指令 v f
  • 一篇关于如何用深度学习完成自动上色(Automatic Image Colorization)的论文浅析

    论文标题是 Let there be Color Joint End to end Learning of Global and Local Image Priors for Automatic Image Colorization wit
  • 【转载】英语动词过去式ed的发音规则

    规则动词词尾加 ed有三种读音 1 在清辅音后读作 t 如 asked helped watched stopped 2 在浊辅音和元音后读作 d 如 enjoyed studied moved called 3 在t d后读作 id 如
  • 基于vue-cli3模板的axios封装项目

    为了更便捷的使用项目框架 本模板为空白项目 但是已经为大家封装了axios方法和post get请求 内有基础案例 请大家按着自己项目需要进行修改使用 axios interceptors response use response gt
  • 采样位数、采样率、波特率

    实例 16bit 16K 115200 1 采样位数 即采样值或取样值 就是将采样样本幅度量化 它是用来衡量声音波动变化的一个参数 也可以说是声卡的分辨率 它的数值越大 分辨率也就越高 所发出声音的能力越强 在计算机中采样位数一般有8位和1
  • Spring Cloud 微服务

    Spring Cloud 微服务架构介绍 单体架构 单体架构也称之为单体系统或者是单体应用 就是把系统中所有的功能 模块耦合在一个应用中的架构方式 特点 打包成一个独立的单元 导成一个唯一的 jar 包或者是 war 包 会一个进程的方式来
  • 金算盘高手论坛资料中心_3D006期 菜鸟论坛精英PK专栏 速来围观!!

    点上方 菜鸟选号论坛 点右上角 选 星标 每日上午更新 星标置顶与大神不走散 苹果是置顶 安卓是星标 点击 菜鸟选号论坛 关注我们 论坛明星版块 集全网各路高手之大乘 打造草根明星 展示舞台 同时主要是为广大关注彩友 在每一期 推出个人擅长
  • swift编程入门(非常详细)从零基础入门到精通,看完这一篇就够了

    文章目录 1 读后概述 2 语法笔记 2 1 说明 2 2 基础类型 2 3 运算符 2 4 字符与字符串 2 4 控制流 2 5 函数 2 6 闭包 2 7 枚举 结构体 类 2 8 类的构造与析构 2 9 属性监听器 2 10 类的继承
  • python 图像处理(7):对比度与亮度调整

    图像亮度与对比度的调整 是放在skimage包的exposure模块里面 1 gamma调整 原理 I Ig 对原图像的像素 进行幂运算 得到新的像素值 公式中的g就是gamma值 如果gamma gt 1 新图像比原图像暗 如果gamma
  • 毕业设计-基于微信小程序的电影推荐系统

    目录 前言 课题背景与简介 实现设计思路 一 电影推荐算法的设计与实现 二 电影推荐系统分析与设计 实现效果样例 更多帮助 前言 大四是整个大学期间最忙碌的时光 一边要忙着备考或实习为毕业后面临的就业升学做准备 一边要为毕业设计耗费大量精力
  • 基于情感词典的情感分析

    思路以及代码都来源于下面两篇文章 一个不知死活的胖子 Python做文本情感分析之情感极性分析 Ran Fengzheng 的博客 基于情感词典的文本情感极性分析相关代码 基于情感词典的情感分析应该是最简单的情感分析方法了 大致说一下使用情
  • http服务器_本地简易http服务器

    本地简易http服务器 下面实现一个简单的http服务器 听起来高大上的样子 其实就是网络通信加上http协议 运用上篇的网络编程的基础模型 上篇的网络通信是开两个vs程序 一个做服务器 一个做客户端 互相发送数据 http服务器其实也是同