如何用C写一个web服务器之GCC项目编译

前言

本想着接下来大概实现一下 CGI 协议,但是实现过程中被一个问题卡住了:

C进程与php进程的交互数据类型问题:

在 C 进程中我准备将服务器处理后的请求数据存储在一个结构体内,然后将此结构体中的信息传给 PHP,而 PHP 进程内也会有一个全局数组与之对应,可是众所周之,结构体是 C 进程内的内存数据,是无法直接传给 PHP 使用的。

这时候我们也需要一种“协议”来解决进程数据类型的异构性。当然这个解决方案确定起来还是很简单的,无非是对C结构体进行序列化,使用xml,json,protobuf(没用过)之一,花费时间多的地方在实现过程。 原来想自己造个轮子,实现一下json类型的编解码,觉得有些偏离了主题了,于是考虑使用一个开源库cJSON;

可是自己没有过 C 大型项目的开发经验,写的都是小 demo,gcc -o name source.c 足以解决问题了,没有过编译多个文件、组织项目的经验,下载到源码后一脸懵逼,搜索到的编译资料都是一些较为零散的内容,不成体系,不过在自己的多次尝试下终于成功地将 cJSON 引入到项目中了,这里稍做一下总结。

绕了好久,终于来到了本篇文章的主题:项目编译,主要介绍一些用 GCC 在 linux 下项目编译链接的步骤。

编译步骤

先说一下一个C源文件的编译一般步骤:

1.预处理(preprocess):主要是在代码层面的处理,包括文件的引入,展开宏定义,删除注释,添加行号等,生成的文件以.i结尾。

gcc -E test.c -o test.i

2.编译(compilation):编译是在代码语法层面的处理,生成对应的汇编语言代码,生成以.s为后缀的汇编语言文件;

gcc -S test.i -o test.s

3.汇编(Assembly):将汇编语言代码生成可执行的机器码,生成以.o为后缀的目标文件。

gcc -c test.s -o test.o

4.链接(Linking):将各个.o目标文件连接起来,并解决库依赖,生成无后缀的可直接执行文件。

gcc -o test test.o

如果我们直接使用后面的命令,那么前面的步骤也会自动执行。如我们常使用的 gcc -o 实际上是一次性完成了所有的步骤的。

以上的中间文件,大家可以使用文本查看工具来查看其中内容来验证其功能。

静态库和动态库

库文件有动态和静态之分,他们的命名规范为 lib库名.后缀,在链接目标文件和库时,使用 -l 库名(空格可省略)选项,也可以添加-L /path来规定优先搜索库文件的目录。

例如:C中的数学函数库math.h的动态库文件名为libm.so,那么我们编译连接文件时就需要添加-lm的选项。如果要指定库文件路径为/usr/lib64/libm.so,那么可添加-L /usr/lib64来指定库文件优先查找目录。

另外静态和动态库文件搜索目录顺序不一样,下面分别详细介绍:

静态库

静态库文件一般是以.a为后缀的库文件,它在编译连接时会将库文件的内容全部添加到可执行文件中,在编译连接完成后,静态库文件便不再影响可执行文件。

它的优点是简单粗暴,但如果库文件内部有改动的话需要重新对所有引用此库文件的可执行文件重新编译。

一般编译步骤如下:

gcc -c static.c -o static.o // 编译静态库文件的源文件
ar -r static.a static.o // 生成静态库文件
gcc -o main -lstatic // 连接静态库文件生成可执行文件

编译连接时,静态库文件搜索目录顺序为:

1.编译连接时 -L 参数指定的目录;

2.环境变量目录 LIBRARY_PATH;

3.固定目录 /lib、/usr/lib、/usr/local/lib等;

动态库

动态库文件一般以.so结尾,它在编译连接时只把动态库的文件添加到可执行文件,只在程序运行时才加载库文件。这种方式的优点是非常灵活,如果动态库文件内部有变动,那么只需重要重新编译库文件即可。

它的一般编译步骤如下:

gcc -c dynamic.c -fpic -o dynamic.o // 编译动态库文件的源文件 -fpic 表示编译为位置独立的代码,使之可以被放在可执行文件内存中的任何地方
gcc -shared dynamic.o -o dynamic.so // 生成动态库文件
gcc -o main -L . -ldynamic // 连接当前文件夹下的动态库文件

编译连接时,动态库文件搜索目录顺序为:

1.编译连接时 -L 参数指定目录;

2.环境变量目录 LD_LIBRARY_PATH;

3.配置文件/etc/ld.so.conf中配置的目录

4.固定目录 /lib、/usr/lib等。

CMakeLists

写到这里还不是结尾,我们要考虑如果文件非常多怎么办,难道每一次都要输入n多个源文件名吗?如果软件完成后,用户使用时可不想记住这些复杂的命令和文件。

自动化才是目标,我们考虑使用自动化编译工具 cmake,那么接下来我们就要编写适合项目文件的编译配置文件 CMakeLists。

CMakeLists 是一个 txt 文件,它就像是项目的编译指南,是给用 cmake 工具用的。其语法类似于 shell,但内置了许多函数,这里我们介绍几个简单的语法,编写一个简单的 CMakeLists.txt。

当前文件结构:

|__ CMakeLists.txt

    |__ test.c

    |__ cJSON.c

    |__ include

    |   |__ cJSON.h

    |__ lib

下面是一个动态库的编译CmakeList,将解释放在注释中。

PROJECT(test)  # 项目名称
cmake_minimum_required(VERSION 2.8) # 选择一个cmake版本

SET(LIBRARY_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/lib) # 设定产生库的目录
SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/bin) # 设定产生的可执行文件的目录

ADD_EXECUTABLE(test test.c) # 这里要先声明产生的可执行文件,以便后面连接

SET(cJSON cJSON.c)  # 设置文件变量
ADD_LIBRARY(cJSON SHARED ${cJSON}) # 此语句用文件变量生成一个动态链接库
TARGET_LINK_LIBRARIES(test cJSON) # 连接可执行文件与动态链接库

FIND_LIBRARY(MATH_LIB libm.so /usr/lib64)  # 在/usr/lib64文件夹下找libm.so(cJSON需要)
IF(MATH_LIB)
    TARGET_LINK_LIBRARIES(test ${MATH_LIB}) # 找到之后连接上
ENDIF()

MESSAGE("cmake complete, use make to compile!") # 在命令行输出提示语句

运行 cmake . && make完成项目的构建。

此时的目录结构为(略过了 cmake 产生的临时文件):

|__ CMakeLists.txt

    |__ test.c

    |__ cJSON.c

    |__ include

    |   |__ cJSON.h

    |__ lib

    |   |__ libcJSON.so

    |__ bin

        |__ test

以上就是如何用C写一个web服务器之GCC项目编译的详细内容,更多关于用C写一个web服务器之GCC项目编译的资料请关注我们其它相关文章!

(0)

相关推荐

  • c++ 如何在libuv中实现tcp服务器

    1.说明 libuv 中实现 tcp server 的步骤和原生 socket 步骤类似,回忆一下 linux 下原生 socket 实现 tcp server 的步骤: 初始化 socket 环境,获取 socket 套接字: bind() 方法绑定套接字到本地IP: listen() 方法监听 socket,获取新连接: accept() 方法接受客户端连接,返回客户端套接字: recv() 方法接受客户端的数据: send() 方法向客户端发送数据: closesocket() 方法关闭套

  • 如何用C写一个web服务器之I/O多路复用

    前言 I/O模型 接触过 socket 编程的同学应该都知道一些 I/O 模型的概念,linux 中有阻塞 I/O.非阻塞 I/O.I/O 多路复用.信号驱动 I/O 和 异步 I/O 五种模型. 其他模型的具体概念这里不多介绍,只简单地提一下自己理解的 I/O 多路复用:简单的说就是由一个进程来管理多个 socket,即将多个 socket 放入一个表中,在其中有 socket 可操作时,通知进程来处理, I/O 多路复用的实现方式有 select.poll 和 epoll. select/p

  • C语言多线程服务器的实现实例

    本文基于 C 标准库提供的网络通信 API,使用 TCP ,实现一个简单的多线程服务器 Demo . 首先要看 API API 字节序转换 函数原型: #include <arpa/inet.h> uint64_t htonll(uint64_t hostlonglong); uint32_t htonl(uint32_t hostlong); uint16_t htons(uint16_t hostshort); uint64_t ntohll(uint64_t netlonglong);

  • 如何用C写一个web服务器之基础功能

    服务器架构 目标架构 以 nginx 的思想来考虑本服务器架构,初步考虑如下图: 当然 php 进程也可以替换为其他的脚本语言,可以更改源码中的 command 变量实现. 服务器有一个 master 进程,其有多个子进程为 worker 进程,master 进程受理客户端的请求,然后分发给 worker 进程,worker 进程处理 http 头信息后将参数传递给 php 进程处理后,将结果返回到上层,再响应给客户端. 也考虑过使用 php-fpm 的 worker 进程池方式,那样的话 ph

  • Linux gcc命令的具体使用

    01. 命令概述 gcc命令使用GNU推出的基于C/C++的编译器,是开放源代码领域应用最广泛的编译器,具有功能强大,编译代码支持性能优化等特点. gcc是GNU编译器套件(GNU Compiler Collection),它包括了C.C++.Objective-C.Fortran.Java.Ada.Go语言和D语言的前端,也包括了这些语言的库(如libstdc++.libgcj等等).GCC的初衷是为GNU操作系统专门编写的一款编译器.GNU系统是彻底的自由软件.此处,"自由"的含义

  • GCC 编译使用动态链接库和静态链接库的方法

    1 库的分类 根据链接时期的不同,库又有静态库和动态库之分. 静态库是在链接阶段被链接的(好像是废话,但事实就是这样),所以生成的可执行文件就不受库的影响了,即使库被删除了,程序依然可以成功运行. 有别于静态库,动态库的链接是在程序执行的时候被链接的.所以,即使程序编译完,库仍须保留在系统上,以供程序运行时调用.(TODO:链接动态库时链接阶段到底做了什么) 2 静态库和动态库的比较 链接静态库其实从某种意义上来说也是一种粘贴复制,只不过它操作的对象是目标代码而不是源码而已.因为静态库被链接后库

  • 浅谈Linux环境下gcc优化级别

    代码优化可以说是一个非常复杂而又非常重要的问题,以笔者多年的linux c开发经验来说优化通常分为两个方面,一是人为优化,也就是基于编程经验采用更简易的数据结构函数等来降低编译器负担,二是采用系统自带的优化模式,也就是gcc - o系列,下面我将简述一下各级优化的过程以及实现. gcc - o1 首先o1上面还有一个o0,那个是不提供任何优化,项目中几乎不会使用,而o1使用就非常广泛了,o1是最基本的优化,主要对代码的分支,表达式,常量来进行优化,编译器会在较短的时间下将代码变得更加短小,这样体

  • GCC 编译c程序的方法及过程解析

    目前 Linux 下最常用的 C 语言编译器是 GCC ( GNU Compiler Collection ),它是 GNU 项目中符合 ANSI C 标准的编译系统,能够编译用 C . C++ 和 Object C 等语言编写的程序. GCC 不仅功能非常强大,结构也异常灵活.最值得称道的一点就是它可以通过不同的前端模块来支持各种语言,如Java . Fortran . Pascal . Modula-3 和 Ada 等.开放.自由和灵活是 Linux 的魅力所在,而这一点在 GCC 上的体现

  • 关于g++和gcc的相同点和区别详解

    gcc和g++的区别和联系 gcc和g++都是GNU(一个组织)的编译器. 1.对于.c后缀的文件,gcc把它当做是C程序:g++当做是C++程序: 2.对于.cpp后缀的文件,gcc和g++都会当做c++程序. 3.编译阶段,g++会调用gcc; 4.连接阶段,通常会用g++来完成,这是因为gcc命令不能自动和c++程序使用的库连接.   gcc/g++在执行编译工作的时候,总共需要4步 1.预处理,生成.i的文件[预处理器cpp] 2.将预处理后的文件转换成汇编语言,生成文件.s[编译器eg

  • 如何用C写一个web服务器之GCC项目编译

    前言 本想着接下来大概实现一下 CGI 协议,但是实现过程中被一个问题卡住了: C进程与php进程的交互数据类型问题: 在 C 进程中我准备将服务器处理后的请求数据存储在一个结构体内,然后将此结构体中的信息传给 PHP,而 PHP 进程内也会有一个全局数组与之对应,可是众所周之,结构体是 C 进程内的内存数据,是无法直接传给 PHP 使用的. 这时候我们也需要一种"协议"来解决进程数据类型的异构性.当然这个解决方案确定起来还是很简单的,无非是对C结构体进行序列化,使用xml,json,

  • 如何用C写一个web服务器之CGI协议

    目录 前言 CGI CGI请求 CGI响应 Nginx和PHP的CGI实现 SAPI PHP-FPM 纠偏 代码实现 http_parser cJSON 前言 这次更新主要实现一下 CGI 协议. 先放上GitHub链接https://github.com/zhenbianshu/tinyServer 作为一个服务器,基本要求是能受理请求,提取信息并将消息分发给 CGI 解释器,再将解释器响应的消息包装后返回客户端.在这个过程中,除了和客户端 socket 之间的交互,还要牵扯到第三个实体 -

  • 如何给ss bash 写一个 WEB 端查看流量的页面

    由于刚毕业的穷大学生,和朋友合租了一台服务器开了多个端口提供 ss 服务,懒得配置 ss-panel,就使用了 ss-bash 来监控不同端口的流量,但每次都要等上服务器才能看到流量使用情况,很麻烦,于是就写了个简单的页面来提供 WEB 访问. JavaScript 版本 用 crontab 定时把流量记录文件复制到 WEB 目录下,写个 JS 脚本作数据处理. function successFunction(data) { var allRows = data.split(/\r?\n|\r

  • 使用python 写一个静态服务(实战)

    师父布置的任务,让我写一个服务练练手,搞清楚socket的原理和过程后跑了一个小demo,很有成就感,代码内容也比较清晰易懂,很有教育启发意义. 代码 # coding:utf-8 import socket from multiprocessing import Process HTML_ROOT_DIR = "" def handle_client(client_socket): """处理客户端请求""" # 获取客户端

  • Python Tornado框架轻松写一个Web应用的全过程

    目录 Tornado是什么 安装 试试看使用tornado框架来写一个web application 总结 Tornado是什么 学委之前在看Jupyter组件的源码的时候,发现了tornado这个web框架. 不仅仅做一个web框架, 通过使用非阻塞网络I/O,Tornado可以扩展到数万个开放连接. 这样非常适合long polling, WebSockets以及其他需要与每个用户建立长期连接的应用程序. 好,下面安装试用一下. 安装 pip install tornado pip 不会用的

  • 如何用Python写一个简单的通讯录

    目录 用Python写一个简单的通讯录 一.构思 1.定义空列表和一个空字典来存储 2.定义功能选项 3.添加通讯录功能 3.2 删除学员功能 二.整体项目演示 用Python写一个简单的通讯录 一.构思 1.定义空列表和一个空字典来存储 list1=[] #用于储存字典中的信息 dict1={} #用于储存联系人信息 2.定义功能选项 def Menu(): print('请选择功能--------\n' '1.添加学员\n' '2.删除学员\n' '3.修改学员\n' '4.查询学员\n'

  • shell linux中如何用shell写一个占用CPU的脚本

    使用场景: 向公司申请的虚机资源自己工作用的比较方便,因占用较小basis要求回收掉,现写一个脚本,让CPU跑满一些. 首先看下共有几颗逻辑CPU cat /proc/cpuinfo |grep "processor"|wc -l 上图可以看到是4颗,我现在跑满2颗 脚本如下 #! /bin/bash # filename killcpu.sh endless_loop() { echo -ne "i=0; while true do i=i+100; i=100 done&

  • javascript如何用递归写一个简单的树形结构示例

    现在有一个数据,需要你渲染出对应的列表出来: var data = [ {"id":1}, {"id":2}, {"id":3}, {"id":4}, ]; var str="<ul>"; data.forEach(function(v,i){ str+="<li><span>"+v.id+"</span></li>&

随机推荐