• Libmicrohttpd简介


    GNU Libmicrohttpd是一个用来在项目中内嵌http服务器的C语言库,它具有以下几个非常鲜明的特点:

    1. C语言库,小而快。
    2. API非常简单,且都是可重入的。
    3. 兼容HTTP1.1。
    4. 支持4种多线程模型(select、poll、pthread、thread poll)。
    5. 跨平台。
    6. 生成的二制文件只有32K(不包含TLS/SSL等额外功能)。

    1. 下载Libmicrohttpd,编译后添加项目

    为了使用Libmicrohttpd,我们需要将其添加进项目中。这里我们选择编译源代码生成静态库,因为官方给的下载静态库版本链接在使用时会有问题,可能是运行库版本不一致; 并且在使用静态库的情况下,我们只需要引用两个文件就可以了(一个头文件、一个库文件),项目结构不会混乱不清。

      生成一个头文件和一个静态库文件

    2. main函数

    main函数非常简单,核心调用只有2个函数:MHD_start_daemonMHD_stop_daemon,分别开始和停止http服务器。

    int main()
    {
        const int port = 8888;
    
        struct MHD_Daemon* daemon = 
            MHD_start_daemon(MHD_USE_SELECT_INTERNALLY, port
            , NULL, NULL, connectionHandler, NULL, MHD_OPTION_END);
        if (daemon == NULL) {
            std::cout << "cannot start server!
    ";
            return -1;
        }
    
        std::cin.get();
        MHD_stop_daemon(daemon);
        return 0;
    }

    MHD_start_daemon函数包含非常多的参数,这也意味着它集成了很多的功能,这里我们只关注四个参数,其它都为NULL:

    1. MHD_USE_INTERNAL_POLLING_THREAD。这个参数与其他两个参数(MHD_USE_POLL_INTERNAL_THREAD、MHD_USE_EPOLL_INTERNAL_THREAD)一起构成了microhttpd支持的三种模式:select、poll、epoll。用户必须选择这三种模式之一。具体信息见源码。
    2. port。端口号。
    3. connectHandler。处理请求的函数。
    4. MHD_OPTION_END。由于MHD_start_daemon最后一个参数是一个变参,因此MHD_OPTION_END用来表示变参终止。

    MHD_stop_daemon函数比较简单,这里不介绍了。

    3 请求处理函数

    所有的请求处理都发生在connectionHandler中:

    int  connectionHandler(
        void *cls,
        struct MHD_Connection *connection,
        const char *url,
        const char *method,
        const char *version,
        const char *upload_data,
        size_t *upload_data_size,
        void **con_cls)
    {
        const char* pageBuffer = "<html><body>Hello, I'm lgxZJ!</body></html>";
    
        struct MHD_Response *response;
        response = MHD_create_response_from_buffer(strlen(pageBuffer),
            (void*)pageBuffer, MHD_RESPMEM_PERSISTENT);
    
        if (MHD_add_response_header(response, "Content-Type", "text/html") == MHD_NO) {
            std::cout << "MHD_add_response_header error
    ";
            return MHD_NO;
        }
        if (MHD_queue_response(connection, MHD_HTTP_OK, response) == MHD_NO) {
            std::cout << "MHD_queue_response error
    ";
            return MHD_NO;
        }
        MHD_destroy_response(response);
    
        return MHD_YES;
    }

    这个函数签名包含了所有用来处理请求的有用信息,这里不逐一介绍了。microhttpd库提供了函数来方便我们响应请求,这里我们重点看创建响应。microhttpd库提供了两种方法来创建请求:从buffer创建从文件创建。但是后者需要传入一个文件描述符,这在windows上不是很方便。

    我们这里用缓冲创建。需要注意的是最后一个参数,这是一个MHD_ResponseMemoryMode枚举值,表示我们使用的buffer内容是固定不变的。这种枚举类型还包含其他2种代表瞬时缓冲类型的值,分别表示缓冲区是在heap上的,和非heap(例如stack)上的。 用不同的缓冲区时要记得用不同的枚举值。 接下来设置MIME类型,把缓冲入队,并释放MHD_Response结构体。对于正确响应,我们返回MHD_YES;不能响应的,我们返回MHD_NO。

  • 相关阅读:
    react-native 安卓支持 gif动态图
    react-navigation,StackNavigator,TabNavigator 导航使用
    poj3694 网络(板子题)
    POJ1275 Cashier Employment 差分约束(优化 二分)
    Codeforces Round #667 (Div. 3)
    POJ1201 Intervals 差分约束(贪心也可)
    YY的GCD
    Codeforces Round #666 (Div. 2) A~E
    Rendezvous
    创世纪
  • 原文地址:https://www.cnblogs.com/shan2017/p/7772767.html
Copyright © 2020-2023  润新知