nginx的哈希表结构--ngx_hash_t

哈希表

   哈希表是一个基础的数据结构,中所周知,数组的随即访问效率是最高的,原因的是数组的随即访问可以通过索引直接定位到数据的实际地址,而无需遍历数组实现,这正是哈希表的思想,而数组也可以看成是索引为数字的特殊哈希表

    哈希表实现了给定索引,直接计算出被索引数据存储地址的功能,这个过程是通过两次映射实现的

1.给定索引数据K,通过一个映射函数f(k)计算出实际的key值
2.通过计算出的key值在info数组中直接得到偏移

即:
k-------> f(k) ------->info[f(k)]
    
    上诉过程,f(k)就是哈希函数,Info数组就是哈希表

     在理想状态下,k与key是一 一对应的,且info数组中不存在空元素,而实际情况往往是无法做到的
      两个不同的K经过f(k)得到了相同的key的情况就称为哈希碰撞/冲突,解决碰撞有三种思路:

        1.开放地址法
         2.再哈希法
         3.链地址法

          nginx采用的是链地址法,通过分桶的方式解决冲突的情况
          在逻辑上,Nginx的哈希表是一个二维数组,这个数组的大小受两个因素影响;桶的大小和桶的个数
          
       nginx的哈希算法
             
          需要注意的是nginx的哈希表一经初始化就不可更改,既不能增加元素,也不能删除元素
          正是因为这个特性,所以可以预先知道所有元素的偏移,因此采用链地址法可以有效解决冲突问题,同时使用数组来代替链表解决冲突则是更优的选择

nginx哈希表结构

       上面提到了,nginx的哈希表实际上是一个二维数组,他采用开放地址法通过分桶解决冲突问题,因此,nginx采用了两层结构实现哈希表
  
// struct ngx_hash_t
//哈希表结构
typedef  struct {
   ngx_hash_elt_t       **buckets;       //哈希桶数组
    ngx_uint_t                size;              //哈希桶数组长度
  }ngx_hash_t; 

buckets 是一个指向ngx_hash_elt_t数组的指针,由他构成了一个ngx_hash_elt_t为元素的二维数组,每一维的长度则由他指向的ngx_hash_elt_t数组来决定

// struct ngx_hash_elt_t
//哈希桶结构
typedef struct {
    void                   *value;               //为null代表该元素为当前bucket最尾元素
     u_short               len;                 //key长度
     u_char                name[1];         //key的值
  }ngx_hash_elt_t;

哈希表所有的key ,value则存储在哈希表索引结构ngx_hash_key_t中,用于在哈希表创建时提供数据

//struct ngx_hash_key_t
//哈希索引结构
typedef struct {
  ngx_str_t              key;                    //索引 
  ngx_uint_t            key_hash ;        //索引指向的哈希值
  void                      *value ;              //内容 
}ngx_hash_key_t;

哈希表创建结构   -- ngx_hash_init_t

//哈希表创建时指定的哈希函数
typedef ngx_uint_t  (*ngx_hash_key_pt)  (u_char *data,size_t len);

// struct ngx_hash_init_t
//哈希表创建限制结构
typedef struct {
ngx_hash_t                    *hash;              //用于管理哈希表的结构体
ngx_hash_key_pt            key;                 // 哈希函数
ngx_uint_t                       max_size;         //哈希桶个数最大值
ngx_uint_t                        bucket_size;    //哈希桶大小
char                                  *name;             //哈希表名字
ngx_pool_t                        *pool;              //哈希表所使用的内存池
ngx_pool_t                         *temp_pool;   //估算临时使用的内存池                      
}ngx_hash_init_t;

这个结构中不仅定义了即将要创建的哈希表桶的最大个数和桶大小的限制,还有指定的哈希函数、用于分配哈希表的内存池等创建哈希表所必须的属性资源


哈希表的创建   ---ngx_hash_init
    
   这个函数通过预先已经存储了所有的数据ngx_hash_key_t 结构和存储限制条件ngx_hash_init_t结构创建了hinit->hash的ngx_hash_t结构
    
    在这个函数中首先进行了限制条件的校验,保证空间足够,然后通过循环,计算了每个数据的大小和偏移,最终将哈希表创建在内存池中一块连续的内存中



哈希表的查询  -- ngx_hash_find
  
     哈希表的查询,首先通过上面说的key % size 得到桶数组序号,然后通过buckets指针获取桶数组首地址,然后遍历对应的桶数组,最终返回查询结果


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
1. --prefix=${PATH_INSTALL}/nginx: --prefix 指定安装目录,${PATH_INSTALL}/nginx 为安装目录的路径。可选值为任意路径,根据实际需要进行设置。此参数会影响到 nginx 的安装位置,例如配置文件的位置、日志文件的位置等。 2. --user=nginx: --user 指定 nginx 运行的用户,默认为 nobody。可选值为任意用户,根据实际需要进行设置。此参数会影响到 nginx 运行的权限。 3. --group=nginx: --group 指定 nginx 运行的用户组,默认为 nobody。可选值为任意用户组,根据实际需要进行设置。此参数会影响到 nginx 运行的权限。 4. --with-http_ssl_module: --with-http_ssl_module 开启 SSL/TLS 功能,支持 HTTPS 协议。可选值为 --with-http_ssl_module 或 --without-http_ssl_module。如果需要支持 HTTPS 协议,则必须开启此选项。 5. --with-http_realip_module: --with-http_realip_module 开启真实 IP 模块,用于获取客户端真实 IP 地址。可选值为 --with-http_realip_module 或 --without-http_realip_module。如果需要获取客户端真实 IP 地址,则必须开启此选项。 6. --with-http_addition_module: --with-http_addition_module 开启添加响应头模块,用于添加自定义的响应头信息。可选值为 --with-http_addition_module 或 --without-http_addition_module。 7. --with-http_sub_module: --with-http_sub_module 开启替换响应内容模块,用于替换响应内容中的关键字。可选值为 --with-http_sub_module 或 --without-http_sub_module。 8. --with-http_dav_module: --with-http_dav_module 开启 WebDAV 模块,用于支持 WebDAV 协议。可选值为 --with-http_dav_module 或 --without-http_dav_module。 9. --with-http_flv_module: --with-http_flv_module 开启 FLV 视频流模块,用于支持 FLV 格式的视频流。可选值为 --with-http_flv_module 或 --without-http_flv_module。 10. --with-http_mp4_module: --with-http_mp4_module 开启 MP4 视频流模块,用于支持 MP4 格式的视频流。可选值为 --with-http_mp4_module 或 --without-http_mp4_module。 11. --with-http_gunzip_module: --with-http_gunzip_module 开启 Gzip 解压缩模块,用于支持 Gzip 压缩格式。可选值为 --with-http_gunzip_module 或 --without-http_gunzip_module。 12. --with-http_gzip_static_module: --with-http_gzip_static_module 开启 Gzip 静态文件压缩模块,用于对静态文件进行 Gzip 压缩。可选值为 --with-http_gzip_static_module 或 --without-http_gzip_static_module。 13. --with-http_stub_status_module: --with-http_stub_status_module 开启状态页面模块,用于查看 nginx 的状态信息。可选值为 --with-http_stub_status_module 或 --without-http_stub_status_module。 14. --with-stream: --with-stream 开启 TCP/UDP 代理模块,用于支持 TCP/UDP 协议。可选值为 --with-stream 或 --without-stream。 15. --with-stream_ssl_module: --with-stream_ssl_module 开启 SSL/TLS 功能,支持 TCP/UDP 的 SSL/TLS 加密。可选值为 --with-stream_ssl_module 或 --without-stream_ssl_module。 16. --with-http_v2_module: --with-http_v2_module 开启 HTTP/2 模块,用于支持 HTTP/2 协议。可选值为 --with-http_v2_module 或 --without-http_v2_module。 17. --with-pcre: --with-pcre 指定使用 PCRE 库进行正则表达式匹配。可选值为 --with-pcre 或 --without-pcre。 18. --with-openssl=/www/server/nginx/src/openssl: --with-openssl 指定使用 OpenSSL 库进行 SSL/TLS 加密。可选值为 OpenSSL 库的路径。如果开启了 SSL/TLS 功能,则必须指定此选项。 19. --with-stream_ssl_preread_module: --with-stream_ssl_preread_module 开启 TCP/UDP SSL/TLS 握手前置模块,用于在握手前解析 SSL/TLS 协议。可选值为 --with-stream_ssl_preread_module 或 --without-stream_ssl_preread_module。 20. --with-http_image_filter_module: --with-http_image_filter_module 开启图片处理模块,用于对图片进行缩放、裁剪等操作。可选值为 --with-http_image_filter_module 或 --without-http_image_filter_module。 21. --with-ipv6: --with-ipv6 开启 IPv6 支持。可选值为 --with-ipv6 或 --without-ipv6。 22. --with-ld-opt=-Wl,-E: --with-ld-opt 指定链接器选项,-Wl,-E 表示启用链接器的 export-dynamic 选项。可选值为任意链接器选项,根据实际需要进行设置。此参数会影响到 nginx 的链接器选项。 23. --with-cc-opt=-Wno-error: --with-cc-opt 指定编译器选项,-Wno-error 表示忽略编译器的错误提示。可选值为任意编译器选项,根据实际需要进行设置。此参数会影响到 nginx 的编译器选项。 24. --with-ld-opt=-ljemalloc: --with-ld-opt 指定链接器选项,-ljemalloc 表示链接 jemalloc 库。可选值为任意链接器选项,根据实际需要进行设置。此参数会影响到 nginx 的链接器选项。 25. --add-module=/www/server/nginx/src/ngx_devel_kit: --add-module 指定添加第三方模块,/www/server/nginx/src/ngx_devel_kit 为第三方模块的路径。可选值为任意第三方模块的路径,根据实际需要进行设置。此参数会影响到 nginx 的模块加载顺序。 26. --add-module=/www/server/nginx/src/lua_nginx_module: --add-module 指定添加第三方模块,/www/server/nginx/src/lua_nginx_module 为第三方模块的路径。可选值为任意第三方模块的路径,根据实际需要进行设置。此参数会影响到 nginx 的模块加载顺序。 27. --add-module=/www/server/nginx/src/ngx_cache_purge: --add-module 指定添加第三方模块,/www/server/nginx/src/ngx_cache_purge 为第三方模块的路径。可选值为任意第三方模块的路径,根据实际需要进行设置。此参数会影响到 nginx 的模块加载顺序。 28. --add-module=/www/server/nginx/src/ngx_http_substitutions_filter_module-master: --add-module 指定添加第三方模块,/www/server/nginx/src/ngx_http_substitutions_filter_module-master 为第三方模块的路径。可选值为任意第三方模块的路径,根据实际需要进行设置。此参数会影响到 nginx 的模块加载顺序。 29. --add-module=/www/server/nginx/src/nginx-dav-ext-module: --add-module 指定添加第三方模块,/www/server/nginx/src/nginx-dav-ext-module 为第三方模块的路径。可选值为任意第三方模块的路径,根据实际需要进行设置。此参数会影响到 nginx 的模块加载顺序。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值