• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            那誰(shuí)的技術(shù)博客

            感興趣領(lǐng)域:高性能服務(wù)器編程,存儲(chǔ),算法,Linux內(nèi)核
            隨筆 - 210, 文章 - 0, 評(píng)論 - 1183, 引用 - 0
            數(shù)據(jù)加載中……

            另類的鏈表數(shù)據(jù)結(jié)構(gòu)以及算法

            一般情況下,我們使用鏈表無(wú)非就是在鏈表結(jié)點(diǎn)中保存該鏈表中下一個(gè)元素的指針.如果為了刪除方便,可能需要保存前一個(gè)元素的指針,也就是雙向鏈表,這樣在刪除一個(gè)結(jié)點(diǎn)的時(shí)候就可以很快的定位到前面和后面的結(jié)點(diǎn),并且改變它們相應(yīng)的指向.在這些操作里面,指向鏈表元素的指針無(wú)疑是最關(guān)鍵的數(shù)據(jù).

            考慮這樣一個(gè)問(wèn)題,如果兩個(gè)進(jìn)程進(jìn)行通信,A進(jìn)程負(fù)責(zé)管理鏈表,B進(jìn)程向A進(jìn)程發(fā)出分配或者刪除鏈表元素的請(qǐng)求.這種情況下,像上面所描述的那樣A進(jìn)程直接向B進(jìn)程返回鏈表元素的指針是不能做到的了,很自然的,可以想到返回另一個(gè)key標(biāo)示該鏈表元素.但是,當(dāng)需要查找或者刪除該鏈表元素的時(shí)候,就不能像上面那樣馬上定位到鏈表元素的位置了,需要遍歷整個(gè)鏈表.原來(lái)常量級(jí)時(shí)間復(fù)雜度的算法,在使用情形變換了之后變成了O(n)級(jí)別的復(fù)雜度.

            可以找到別的辦法來(lái)解決這個(gè)問(wèn)題.第一可以返回一個(gè)key標(biāo)示該鏈表元素,第二保證了時(shí)間的復(fù)雜度,在這里需要定義一種新的數(shù)據(jù)結(jié)構(gòu)和算法來(lái)解決這個(gè)問(wèn)題.

            首先,我們使用一個(gè)數(shù)組,數(shù)組中的元素是指向鏈表元素的指針,而指針的索引則是每個(gè)鏈表元素的id,這樣,通過(guò)id就可以馬上定位到對(duì)應(yīng)的鏈表元素.
            但是這里又會(huì)出現(xiàn)另一個(gè)問(wèn)題,該id是從零開始的,假如在一段時(shí)間之后,需要分配一個(gè)新的鏈表元素,如何知道數(shù)組中的哪個(gè)位置是可以分配的?在這里,使用了一個(gè)整型數(shù)據(jù)的數(shù)組,數(shù)組中的每個(gè)元素是該id對(duì)應(yīng)的鏈表元素在鏈表中下一個(gè)結(jié)點(diǎn)的id(有點(diǎn)拗口).我們使用兩個(gè)鏈表頭,一個(gè)將已經(jīng)使用的鏈表元素id連接起來(lái),另一個(gè)則將未使用的鏈表元素id連接起來(lái).于是,在程序初始化的時(shí)候,未使用的鏈表中保存了所有的id,而已經(jīng)使用的鏈表為空.每次分配了一個(gè)新的鏈表元素,將它的id放在使用鏈表的最開始;而每次釋放一個(gè)鏈表元素,將它的id放到未使用的鏈表頭部.
            同時(shí),改變?cè)孺湵碓氐亩x,在該結(jié)構(gòu)體中,保存的不再是指針,而是鏈表中前一個(gè)元素的數(shù)組索引id.而它的下一個(gè)元素id則保存在上面的那個(gè)數(shù)組中.

            如果上面我的解釋還不夠明白,可以看看下面的代碼:

            #include 
            <stdio.h>

            #define LIST_NODE_NULL -1
            #define ARRAY_SIZE 200

            /* 鏈表元素定義 */
            typedef 
            struct list_node
            {
                
            int prev;   /* 下一個(gè)鏈表元素在list_array中的id */
            }list_node;

            /* 存放鏈表元素指針的數(shù)組 */
            list_node
            * list_array[ARRAY_SIZE];
            /* 未使用鏈表的頭結(jié)點(diǎn)id */
            int top_of_free;
            /* 已使用鏈表的頭結(jié)點(diǎn)id */
            int top_of_used;
            /* 保存鏈表下一個(gè)元素結(jié)點(diǎn)id的鏈表 */
            int next_list[ARRAY_SIZE];

            void init_list()
            {
                
            int i;

                
            for (i = 0; i < ARRAY_SIZE; ++i)
                {
                    list_array[i] 
            = NULL;
                    
            /* 初始時(shí),next_list中每個(gè)結(jié)點(diǎn)的值都是下一個(gè)id */
                    next_list[i] 
            = i + 1;
                }

                
            /* 最后一個(gè)結(jié)點(diǎn)是空 */
                next_list[i 
            - 1= LIST_NODE_NULL;

                top_of_free 
            = 0;
                top_of_used 
            = LIST_NODE_NULL;
            }

            int alloc_list_node()
            {
                
            int id;
                
                
            /* 從未使用鏈表頭部取出一個(gè)id */
                id 
            = top_of_free;

                
            if (LIST_NODE_NULL == id)
                {
                    
            return LIST_NODE_NULL;
                }

                
            /* 未使用鏈表頭結(jié)點(diǎn)往下走一步 */
                top_of_free 
            = next_list[top_of_free];

                
            if (NULL == list_array[id])
                {
                    list_array[id] 
            = (list_node*)malloc(sizeof(list_node));
                    
            if (NULL == list_array[id])
                    {
                        
            return LIST_NODE_NULL;
                    }
                }

                
            if (LIST_NODE_NULL == top_of_used)
                {
                   
            next_list[id] = top_of_used; 
                    top_of_used = id;

                }
                
            else
                {
                    
            /* 修改prev和next */ 
                    list_array[top_of_used]
            ->prev = id;
                    list_array[id]
            ->prev = LIST_NODE_NULL;

                    next_list[id] 
            = top_of_used;
                    top_of_used 
            = id;
                }

                
            return id;
            }

            void free_list_node(int id)
            {
                
            int prev, next;

                prev 
            = list_array[id]->prev;
                next 
            = next_list[id];

                
            /* 修改next和prev */
                if (
            LIST_NODE_NULL != prev)
                {
                    next_list[prev] 
            = next_list[id];
                }
                
            if (LIST_NODE_NULL != next && NULL != list_array[next])
                {
                    list_array[next]
            ->prev = prev;
                }

                
            if (id == top_of_used)
                {
                    top_of_used 
            = next_list[id];
                }

                
            /* 將鏈表id返回到free鏈表頭部并且修改free鏈表頭結(jié)點(diǎn) */
                next_list[id] 
            = top_of_free;
                top_of_free 
            = id;
            }

            int main()
            {
                
            int id;

                init_list();

                id 
            = alloc_list_node();
                free_list_node(id);

                
            return 0;
            }



            這個(gè)想法很巧妙,有效的避免了查找和刪除數(shù)組元素帶來(lái)的開銷.我不知道具體的出處在哪里,如果您知道,勞煩告訴我一聲:)


            posted on 2009-03-22 19:14 那誰(shuí) 閱讀(4501) 評(píng)論(7)  編輯 收藏 引用 所屬分類: 算法與數(shù)據(jù)結(jié)構(gòu)

            評(píng)論

            # re: 另類的鏈表數(shù)據(jù)結(jié)構(gòu)以及算法  回復(fù)  更多評(píng)論   

            如果我沒(méi)理解錯(cuò)誤的話:
            1 這個(gè)應(yīng)該是靜態(tài)鏈表的變形。
            2 示例程序似乎有些地方有問(wèn)題.
            (1)分配算法里邊的:
            if (LIST_NODE_NULL == top_of_used)
            {
            top_of_used = id;
            }
            是否應(yīng)該改為:
            if (LIST_NODE_NULL == top_of_used)
            {
            next_list[id] = top_of_used;
            top_of_used = id;

            }
            (2)釋放函數(shù):
            next_list[prev] = next_list[next_list[id]];
            if (NULL != list_array[next])
            {
            list_array[next]->prev = prev;
            }
            next_list[prev]和list_array[next]都有可能造成數(shù)組越界。

            另外,這個(gè)next_list[prev] = next_list[next_list[id]];是否應(yīng)該改為
            next_list[prev] = next_list[id]?
            2009-03-27 13:31 | capable

            # re: 另類的鏈表數(shù)據(jù)結(jié)構(gòu)以及算法  回復(fù)  更多評(píng)論   

            感謝指出錯(cuò)誤,已經(jīng)做了修改了.
            2009-03-27 19:03 | 那誰(shuí)

            # re: 另類的鏈表數(shù)據(jù)結(jié)構(gòu)以及算法  回復(fù)  更多評(píng)論   

            別客氣,應(yīng)該的。
            從你的博客,我學(xué)到了不少東西,希望能夠在這里看到更多你的大作。
            2009-03-27 22:14 | capable

            # re: 另類的鏈表數(shù)據(jù)結(jié)構(gòu)以及算法  回復(fù)  更多評(píng)論   

            為什么不直接使用鏈表節(jié)點(diǎn)指針作為key呢?
            2009-03-28 15:22 | t

            # re: 另類的鏈表數(shù)據(jù)結(jié)構(gòu)以及算法  回復(fù)  更多評(píng)論   

            應(yīng)該是 Justin Heyes-Jones 在 A* Algorithm code 中的 fast simple allocator 使用的結(jié)構(gòu)
            http://code.google.com/p/a-star-algorithm-implementation/downloads/list
            2009-04-11 01:48 | alan5281

            # re: 另類的鏈表數(shù)據(jù)結(jié)構(gòu)以及算法  回復(fù)  更多評(píng)論   

            申請(qǐng)新結(jié)點(diǎn)空間后,應(yīng)該初始化結(jié)點(diǎn),否則釋放時(shí)會(huì)出現(xiàn)問(wèn)題。
            list_array[identity] = new list_node;
            if(NULL == list_array[identity])
            {
            return(list_node_null);
            }
            list_array[identity]->prev = list_node_null;
            2009-06-03 08:59 | zuima

            # re: 另類的鏈表數(shù)據(jù)結(jié)構(gòu)以及算法  回復(fù)  更多評(píng)論   

            算法導(dǎo)論10.3介紹了這種用法,包括后面提到的用兩個(gè)鏈表來(lái)維護(hù)用過(guò)的和為用過(guò)的節(jié)點(diǎn)
            2010-08-22 02:00 | Dbger
            精品久久久久久久无码| 一本大道久久东京热无码AV| 久久青草国产精品一区| 久久久久香蕉视频| av午夜福利一片免费看久久| 久久精品国产精品亚洲| 久久久国产乱子伦精品作者| 久久人妻少妇嫩草AV无码蜜桃| 国产免费久久精品99re丫y| 91精品国产综合久久精品| 亚洲人成网站999久久久综合| 五月丁香综合激情六月久久| 久久久久亚洲AV成人网| 国产精品久久亚洲不卡动漫| 99久久免费国产精品特黄| 久久99精品国产麻豆婷婷| 久久国产欧美日韩精品| 久久精品无码一区二区WWW| 久久精品国产99久久香蕉| 99久久国语露脸精品国产| 亚洲精品无码久久一线| 区亚洲欧美一级久久精品亚洲精品成人网久久久久 | 亚洲∧v久久久无码精品| 国产综合成人久久大片91| 国内精品久久国产大陆| 亚洲AV日韩AV永久无码久久 | 看久久久久久a级毛片| 亚洲精品美女久久久久99小说| 情人伊人久久综合亚洲| 国产午夜免费高清久久影院| 亚洲国产精品无码久久久秋霞2 | 国产精品免费福利久久| 久久综合久久自在自线精品自| 久久亚洲sm情趣捆绑调教| 久久精品国产欧美日韩99热| 欧洲性大片xxxxx久久久| 欧美与黑人午夜性猛交久久久| 热综合一本伊人久久精品| 午夜肉伦伦影院久久精品免费看国产一区二区三区 | 9191精品国产免费久久| 国产一区二区精品久久岳|