apache源码分析(一)

    大家好,我是artleaf,也就是培培。从今天开始,我将会进行apache源码的分析。其实之前就很想分析apache源码了,但是一直是自己偷懒,给自己找借口,因为apache源码比较多,大概有28万行吧,很艰巨的工作呀。而且本身我也是一个水平很低的小程序员,深怕自己吃不下来,毕竟apache是开源里面已经相当成熟的软件了,里面的代码的质量是相当的高,是多少牛人经过无数次推敲而出来的,我一个对it技术似懂非懂的人,怎么能弄懂这个软件,而且懂it的人都知道,apache占领着世界60%的服务器市场,这么一个强悍的软件,要想弄懂它不是一件容易的事情,而且也不是一朝一夕的事情。但是总归是要做的,现在不做,以后可能就没有时间做了,所以硬着头皮,开始一点一点的分析吧。所以此贴会不定期更新,因为我没有办法保证自己每一周可以分析出来一些东西,但是我会尽量做好。

apache 需要你了解C语言的知识,这个是必须的,因为apache是由C语言写出来的,其次是学过数据结构,Linux的知识暂时不必要求太深,本身我的Linux知识很肤浅,但是随着分析的深入,需要再知道什么知识,我们在说。我们首先是要分析apache的内存结构,就是apacheapr库,这个对于上述知识已经够了。我分析的是apachehttpd-2.2.15这个版本,apr库中memory中的apr_pools.c这个程序有2602行。我们首先分析的就是这个程序。为什么要分析这个程序呢?其实很久之前就有一个人在做分析apache的源码了,就是张中庆师兄,张师兄在csdn上有博客叫做tingya的专栏,大家有兴趣的可以去看看,张师兄的技术就比较强悍了,我刚开始看apache的知识的时候,就是通过张师兄的apache分析来看的,但是张师兄水平比较高,所以很多地方都是一笔带过,讲的很精练,像我水平这么低的人,要琢磨半天才能明白张师兄的深意。这里我开始的分析,就没有张师兄那么厉害,我只是为了提升自己对软件开放的理解,也对大型软件的掌握,同时提升下自己的计算机软件编程技术水平才想分析apache这个软件的。在tingya的专栏里面我们可以看到为什么会先分析apr库,下面我选自张师兄的话:对于APR中的所有的对象中,内存池对象应该是其余对象内存分配的基础,不仅是APR中的对象,而且对于整个Apache中的大部分对象的内存都是从内存池中进行分配的,因此我们将把内存池作为整个APR的基础。apr库是apache的基础,所以我们从apr库开始分析。但是之前我们先分析下main函数,来给apache有个有个整体的认识。

好,首先我们先来看看apachemain()函数。这个是apache的开始,

#include "apr.h"

#include "apr_strings.h"

#include "apr_getopt.h"

#include "apr_general.h"

#include "apr_lib.h"

#include "apr_md5.h"

#include "apr_time.h"

#include "apr_version.h"

#include "apu_version.h"

#include "apr_want.h"

#include "ap_config.h"

#include "httpd.h"

#include "http_main.h"

#include "http_log.h"

#include "http_config.h"

#include "http_core.h"

#include "http_vhost.h"

#include "apr_uri.h"

#include "util_ebcdic.h"

#include "ap_mpm.h"

#include "mpm_common.h"

这些是apachemain函数中声明的21个头文件,我们可以看到这里面第一个是apr库的声明。之后有字符串处理的头文件也有加密的头文件等等。暂时我们先不用管这么多。这里只是先给出一个概念。之后我们看下main()函数的变量的定义,

char c;

    int configtestonly = 0;

    const char *confname = SERVER_CONFIG_FILE;

    const char *def_server_root = HTTPD_ROOT;

    const char *temp_error_log = NULL;

    const char *error;

    process_rec *process;

    server_rec *server_conf;

    apr_pool_t *pglobal;

    apr_pool_t *pconf;

    apr_pool_t *plog;

    apr_pool_t *ptemp;

    apr_pool_t *pcommands;

    apr_getopt_t *opt;

    apr_status_t rv;

    module **mod;

    const char *optarg;

    APR_OPTIONAL_FN_TYPE(ap_signal_server) *signal_server;

我们可以看到,红色的部分都是从内存池中分配出来的,关于内存池的概念,我们下一次对apr库分析的时候会给出解释。从这些变量我们也可以看出,内存分配也是多么的基础。之后我们来看下main函数里面调用的各个函数。

AP_MONCONTROL(0);

process = init_process(&argc, &argv);

    ap_server_argv0 = process->short_name;

apr_pool_create(&pcommands, pglobal);

    apr_pool_tag(pcommands, "pcommands");

    ap_server_pre_read_config  = apr_array_make(pcommands, 1, sizeof(char *));

    ap_server_post_read_config = apr_array_make(pcommands, 1, sizeof(char *));

    ap_server_config_defines   = apr_array_make(pcommands, 1, sizeof(char *));

apr_getopt_init(&opt, pcommands, process->argc, process->argv);

apr_pool_create(&plog, pglobal);

apr_pool_tag(plog, "plog");

apr_pool_create(&ptemp, pconf);

apr_pool_tag(ptemp, "ptemp");

apr_pool_destroy(ptemp);

ap_fixup_virtual_hosts(pconf, server_conf);

ap_fini_vhost_config(pconf, server_conf);

apr_hook_sort_all();

apr_pool_clear(plog);

apr_pool_destroy(ptemp);

apr_pool_lock(pconf, 1);

ap_run_optional_fn_retrieve();

apr_pool_lock(pconf, 0);

destroy_and_exit_process(process, 0);

这里几乎包含了main函数里面所有的函数的调用,但是还是有一部分,在这里我们先忽略,大家不要被我罗列在这里的这些只知其名,不知其意的函数所吓倒,我们一点一点的分析。我们首先只需要关注apr库中内存池的各个函数。也就是我上面用红色标注的函数。

虽然我们在主函数中屏蔽掉了很多代码,或许是在主程序中非常有用的代码,但是我们先要分析的是apr库里面的内存池部分,所有暂时可以看作是不重要的,等我们把内存池分析完成之后,我们在回过头来看这些代码。等我们把整个apache的数据结构掌握之后,这些代码的分析就会变的非常的简单和快速,而最开始的我们先从最简单的部分开始分析,所有下一次分析中,我们开始着手进行内存池的分析,这个是apr库的基础,也是整个apache的基础。下一次,我将带来对内存池的分析,敬请大家期待。

来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/21120473/viewspace-660022/,如需转载,请注明出处,否则将追究法律责任。

转载于:http://blog.itpub.net/21120473/viewspace-660022/

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Apache HttpClient 是一个流行的 Java HTTP 客户端库,用于在 Java 应用程序中进行 HTTP 通信。在这里,我将讨论 Apache HttpClient 的一些核心概念和源代码实现。 1. HttpClient 的核心概念 HttpClient 由以下几个核心类组成: - HttpClient:核心类,用于执行 HTTP 请求。 - HttpRequestBase:所有 HTTP 请求类的基类。 - HttpEntity:HTTP 实体,用于表示 HTTP 消息的内容和元数据。 - HttpResponse:HTTP 响应类。 - HttpConnection:HTTP 连接类,用于管理连接池。 - HttpParams:HTTP 参数类,用于存储 HTTP 请求和响应的参数。 2. HttpClient 的源代码实现 HttpClient 的源代码实现分为以下几个部分: 2.1 创建 HttpClient 实例 创建 HttpClient 实例的方法有两种: - 使用 HttpClientBuilder:通过 HttpClientBuilder 类的 build() 方法创建 HttpClient 实例。 - 直接创建 DefaultHttpClient:使用 DefaultHttpClient 类的构造函数创建 HttpClient 实例。 2.2 发送 HTTP 请求 HttpClient 中发送 HTTP 请求的主要类是 CloseableHttpClient,它是 HttpClient 接口的实现类。CloseableHttpClient 的 execute() 方法用于发送 HTTP 请求,它接收一个 HttpRequestBase 类型的参数,HttpRequestBase 是所有 HTTP 请求类的基类。 具体的发送 HTTP 请求的步骤如下: - 创建 HttpClient 实例。 - 创建 HttpGet、HttpPost 或其他 HttpRequestBase 类型的请求对象。 - 执行请求并获取响应。 2.3 接收 HTTP 响应 HttpClient 中接收 HTTP 响应的主要类是 CloseableHttpResponse,它是 HttpResponse 接口的实现类。CloseableHttpResponse 的 getEntity() 方法用于获取 HTTP 实体,它是 HttpEntity 接口的实现类。 具体的接收 HTTP 响应的步骤如下: - 执行请求并获取响应。 - 从响应中获取 HTTP 实体。 - 从 HTTP 实体中获取内容和元数据。 2.4 连接管理 HttpClient 中连接管理的主要类是 PoolingHttpClientConnectionManager,它是 HttpConnection 接口的实现类。PoolingHttpClientConnectionManager 的 getMaxTotal() 方法用于获取最大连接数,setMaxTotal() 方法用于设置最大连接数。 具体的连接管理步骤如下: - 创建 HttpClient 实例。 - 创建 PoolingHttpClientConnectionManager 实例。 - 将 PoolingHttpClientConnectionManager 实例注入到 HttpClient 实例中。 2.5 参数设置 HttpClient 中参数设置的主要类是 HttpClientParams,它是 HttpParams 接口的实现类。HttpClientParams 的 setConnectionManagerTimeout() 方法用于设置连接管理器超时时间,setSoTimeout() 方法用于设置 Socket 超时时间。 具体的参数设置步骤如下: - 创建 HttpClient 实例。 - 创建 HttpClientParams 实例。 - 将 HttpClientParams 实例注入到 HttpClient 实例中。 以上就是 Apache HttpClient 的核心概念和源代码实现。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值