zoukankan      html  css  js  c++  java
  • nginx学习六 高级数据结构之双向链表ngx_queue_t

    1 ngx_queue_t简单介绍

    ngx_queue_t是nginx提供的一个轻量级的双向链表容器,它不负责存储数据,既不提供数据的内存分配。它仅仅有两个指针负责把数据链入链表。它跟stl提供的queue不同,stl提供的queue帮助用户存储数据。用户仅仅须要相容器里加入数据就可以,而ngx_queue_t,用户必须自己提供存储数据的内存,而且必须定义一种数据结构把ngx_queue_t包括在当中,然后利用ngx_queue_t提供的函数来进行对应的操作。

    2 ngx_queue_t结构及其操作

    2.1 ngx_queue_t

    struct ngx_queue_s {
        ngx_queue_t  *prev;
        ngx_queue_t  *next;
    };
    
    ngx_queue_t仅仅提供一个指向前驱和一个指向后继的指针。结构很easy,这也是其可以通用性的原因。

    2.2 操作函数

    ngx_queue_init(q)            //初始化链表
    ngx_queue_empty(h)           //推断链表是否为空                                                   
    ngx_queue_insert_head(h, x)  //在头部插入一个元素                                       
    ngx_queue_insert_after       //在h元素前面插入一个元素
    ngx_queue_insert_tail(h, x)  //在h尾部插入一个元素 
    ngx_queue_head(h)            //返回第一个元素
    #define ngx_queue_last(h)    //返回最后一个元素 
    ngx_queue_sentinel(h)        //返回链表容器结构体的指针
    ngx_queue_next(q)            //返回下一个q的下一个元素  
    ngx_queue_prev(q)            //返回q的前一个元素
    ngx_queue_remove(x)          //删除x结点                                           
    ngx_queue_split(h, q, n)     //把h分为两个链表h和n,而且n的第一元素为q
    ngx_queue_add(h, n)          //把链表n添加到h链表的尾部
    ngx_queue_data(q, type, link)//取出包括q的type类型的地址。这样我们就能够訪问type内的成员

    ngx_queue_t提供了14个经常使用的操作给用户使用,基本涵盖了插入、删除、移动。訪问数据等等操作,这14个函数都是宏定义。有兴趣的能够看下源代码。很easy。

    这里说一个最后一个操作函数ngx_queue_data:

    #define ngx_queue_data(q, type, link)                                         
        (type *) ((u_char *) q - offsetof(type, link))
    

    q为ngx_queue_t的指针, type是用户自己定义的包括ngx_queue_t的数据类型type。link是type的成员。类型是ngx_queue_t。

    这里举个列子来说明这个操作的使用方法。

    先看一个自己定义的结构体:

    typedef struct
    {
    	ngx_int_t num;
    	ngx_str_t str;
    	ngx_queue_t queue;
    }TestNode;

    假设我们有一个ngx_queue_t的指针q指向testNode.queue,如今我们不知到testNode的地址,仅仅知道queue,假设我们想訪问testNode里面的成员num,我们必须知道testNode的地址。这样才干訪问其num成员。如何知道testNode的地址呢?这时候ngx_queue_data就闪亮登场了。

    我们能够用一下语句来取得testNode的地址:

    TestNode* testnode  = ngx_queue_data(q, TestNode, queue);
    这样我们就能够訪问num了。
    2.3 ngx_queue_middle

    这个函数取出链表中间位置的节点。用一个慢指针midlle和一个快指针next,middle没走一步,next走两步,当next指针到达链表未的时候。midlle就指向链表的中间位置。


    ngx_queue_t *
    ngx_queue_middle(ngx_queue_t *queue)
    {
        ngx_queue_t  *middle, *next;
    
        middle = ngx_queue_head(queue);
    
        if (middle == ngx_queue_last(queue)) {
            return middle;
        }
    
        next = ngx_queue_head(queue);
    
        for ( ;; ) {//middle每前进 一步,next都要前进两步。直到链表的尾部
            middle = ngx_queue_next(middle);
    
            next = ngx_queue_next(next);
    
            if (next == ngx_queue_last(queue)) {
                return middle;
            }
    
            next = ngx_queue_next(next);
    
            if (next == ngx_queue_last(queue)) {
                return middle;
            }
        }
    }

    2.4 ngx_ngx_queue_sort

    void
    ngx_queue_sort(ngx_queue_t *queue,
        ngx_int_t (*cmp)(const ngx_queue_t *, const ngx_queue_t *))
    {
        ngx_queue_t  *q, *prev, *next;
    
        q = ngx_queue_head(queue);
    
        if (q == ngx_queue_last(queue)) {
            return;
        }
    
        //遍历链表中的每个元素,然后遍历它前面的元素是否比它大。直到找到不比它大第一个元素。然后插入。
        for (q = ngx_queue_next(q); q != ngx_queue_sentinel(queue); q = next) {
    
            prev = ngx_queue_prev(q);
            next = ngx_queue_next(q);
    
            ngx_queue_remove(q);
    
            do {//遍历它前面的元素
                if (cmp(prev, q) <= 0) {
                    break;
                }
    
                prev = ngx_queue_prev(prev);
    
            } while (prev != ngx_queue_sentinel(queue));
    
            ngx_queue_insert_after(prev, q);//q前面的元素必须是小于q
        }
    }

    3 一个样例

    #include <stdio.h>
    #include <stdlib.h>
    #include <ngx_config.h>
    #include <ngx_core.h>
    #include <ngx_conf_file.h>
    #include <nginx.h>
    #include <ngx_queue.h>
    
    //////////////////////////////////////////////////////////////
    //////////////////////////////////////////////////////////////
    //这两个东东必须写,不为有编译错误
    volatile ngx_cycle_t  *ngx_cycle;
    
    void ngx_log_error_core(ngx_uint_t level, ngx_log_t *log, ngx_err_t err,
                const char *fmt, ...)
    {
    }
    //////////////////////////////////////////////////////////////
    //////////////////////////////////////////////////////////////
    
    typedef struct
    {
    	ngx_int_t num;
    	ngx_str_t str;
    	ngx_queue_t queue;
    }TestNode;
    
    ngx_int_t compare_node(const ngx_queue_t *left, const ngx_queue_t *right)
    {
    	TestNode* left_node  = ngx_queue_data(left, TestNode, queue);
    	TestNode* right_node = ngx_queue_data(right, TestNode, queue);
    	
    	return left_node->num > right_node->num;
    }
    
    
    int main()
    {
        ngx_queue_t QueHead;
    	ngx_queue_init(&QueHead);
    
    	TestNode Node[10];
    	ngx_int_t i;
    	for (i=0; i<10; ++i)
    	{
    		Node[i].num = rand()%100;
    	}
    
        ngx_queue_insert_head(&QueHead, &Node[0].queue);
    	ngx_queue_insert_tail(&QueHead, &Node[1].queue);
    	ngx_queue_insert_after(&QueHead, &Node[2].queue);
        ngx_queue_insert_head(&QueHead, &Node[4].queue);
    	ngx_queue_insert_tail(&QueHead, &Node[3].queue);
        ngx_queue_insert_head(&QueHead, &Node[5].queue);
    	ngx_queue_insert_tail(&QueHead, &Node[6].queue);
    	ngx_queue_insert_after(&QueHead, &Node[7].queue);
        ngx_queue_insert_head(&QueHead, &Node[8].queue);
    	ngx_queue_insert_tail(&QueHead, &Node[9].queue);
    
    	ngx_queue_t *q;
    	for (q = ngx_queue_head(&QueHead); q != ngx_queue_sentinel(&QueHead); q = ngx_queue_next(q))
    	{
    		TestNode* Node = ngx_queue_data(q, TestNode, queue);
    		printf("Num=%d
    ", Node->num);
    	}
    
        ngx_queue_sort(&QueHead, compare_node);
    	
        printf("
    ");
    	for (q = ngx_queue_head(&QueHead); q != ngx_queue_sentinel(&QueHead); q = ngx_queue_next(q))
    	{
    		TestNode* Node = ngx_queue_data(q, TestNode, queue);
    		printf("Num=%d
    ", Node->num);
    	}
        
    	return 0;
    }

    http://blog.csdn.net/xiaoliangsky/article/details/39646141
  • 相关阅读:
    ASP.NET Forums 2.0 修改集锦(一)
    在自己的应用程序中显示Windows关于对话框
    ASP.NET Forums 2.0 本地化修改(四)
    Flash对双字节的支持问题
    DotNetNuke 2.1.2安装指南
    ASP.NET Forums 2.0 本地化修改(五) 增加页面Meta标记的keywords和description
    ASP.NET Forums 2.0 本地化修改(三)
    JavaScript利用正则表达式自己写数字判断函数
    hash表基础知识(转载)
    求子数组的最大和
  • 原文地址:https://www.cnblogs.com/zfyouxi/p/5177875.html
Copyright © 2011-2022 走看看