青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

那誰的技術博客

感興趣領域:高性能服務器編程,存儲,算法,Linux內核
隨筆 - 210, 文章 - 0, 評論 - 1183, 引用 - 0
數據加載中……

另類的鏈表數據結構以及算法

一般情況下,我們使用鏈表無非就是在鏈表結點中保存該鏈表中下一個元素的指針.如果為了刪除方便,可能需要保存前一個元素的指針,也就是雙向鏈表,這樣在刪除一個結點的時候就可以很快的定位到前面和后面的結點,并且改變它們相應的指向.在這些操作里面,指向鏈表元素的指針無疑是最關鍵的數據.

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

可以找到別的辦法來解決這個問題.第一可以返回一個key標示該鏈表元素,第二保證了時間的復雜度,在這里需要定義一種新的數據結構和算法來解決這個問題.

首先,我們使用一個數組,數組中的元素是指向鏈表元素的指針,而指針的索引則是每個鏈表元素的id,這樣,通過id就可以馬上定位到對應的鏈表元素.
但是這里又會出現另一個問題,該id是從零開始的,假如在一段時間之后,需要分配一個新的鏈表元素,如何知道數組中的哪個位置是可以分配的?在這里,使用了一個整型數據的數組,數組中的每個元素是該id對應的鏈表元素在鏈表中下一個結點的id(有點拗口).我們使用兩個鏈表頭,一個將已經使用的鏈表元素id連接起來,另一個則將未使用的鏈表元素id連接起來.于是,在程序初始化的時候,未使用的鏈表中保存了所有的id,而已經使用的鏈表為空.每次分配了一個新的鏈表元素,將它的id放在使用鏈表的最開始;而每次釋放一個鏈表元素,將它的id放到未使用的鏈表頭部.
同時,改變原先鏈表元素的定義,在該結構體中,保存的不再是指針,而是鏈表中前一個元素的數組索引id.而它的下一個元素id則保存在上面的那個數組中.

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

#include 
<stdio.h>

#define LIST_NODE_NULL -1
#define ARRAY_SIZE 200

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

/* 存放鏈表元素指針的數組 */
list_node
* list_array[ARRAY_SIZE];
/* 未使用鏈表的頭結點id */
int top_of_free;
/* 已使用鏈表的頭結點id */
int top_of_used;
/* 保存鏈表下一個元素結點id的鏈表 */
int next_list[ARRAY_SIZE];

void init_list()
{
    
int i;

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

    
/* 最后一個結點是空 */
    next_list[i 
- 1= LIST_NODE_NULL;

    top_of_free 
= 0;
    top_of_used 
= LIST_NODE_NULL;
}

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

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

    
/* 未使用鏈表頭結點往下走一步 */
    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鏈表頭結點 */
    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;
}



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


posted on 2009-03-22 19:14 那誰 閱讀(4535) 評論(7)  編輯 收藏 引用 所屬分類: 算法與數據結構

評論

# re: 另類的鏈表數據結構以及算法  回復  更多評論   

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

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

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

# re: 另類的鏈表數據結構以及算法  回復  更多評論   

感謝指出錯誤,已經做了修改了.
2009-03-27 19:03 | 那誰

# re: 另類的鏈表數據結構以及算法  回復  更多評論   

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

# re: 另類的鏈表數據結構以及算法  回復  更多評論   

為什么不直接使用鏈表節點指針作為key呢?
2009-03-28 15:22 | t

# re: 另類的鏈表數據結構以及算法  回復  更多評論   

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

# re: 另類的鏈表數據結構以及算法  回復  更多評論   

申請新結點空間后,應該初始化結點,否則釋放時會出現問題。
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: 另類的鏈表數據結構以及算法  回復  更多評論   

算法導論10.3介紹了這種用法,包括后面提到的用兩個鏈表來維護用過的和為用過的節點
2010-08-22 02:00 | Dbger
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲欧洲一区| 亚洲欧美精品在线观看| 亚洲精品一二区| 亚洲理伦电影| 免费视频最近日韩| 亚洲一区二区影院| 欧美国产一区二区在线观看 | 怡红院av一区二区三区| 亚洲男人第一网站| 欧美制服第一页| 一本久道久久久| 欧美激情一区二区三区蜜桃视频| 毛片一区二区三区| 亚洲一区免费网站| 亚洲一区二区三区在线观看视频 | 国产欧美91| 欧美在线视频不卡| 蜜臀久久99精品久久久久久9 | 免费观看不卡av| 一区二区亚洲| 久久久久久高潮国产精品视| 欧美国产日韩亚洲一区| 这里只有精品电影| 日韩视频免费观看高清完整版| 欧美日韩你懂的| 欧美一级播放| 日韩写真视频在线观看| 性欧美xxxx视频在线观看| 中文精品视频| 国内精品免费午夜毛片| 蜜桃av噜噜一区二区三区| 一区二区三区成人| 免费在线欧美黄色| 久久精品国产亚洲a| 亚洲激情成人在线| 欧美另类99xxxxx| 久久蜜桃精品| 亚洲欧美日韩综合| 欧美激情无毛| 久久九九免费视频| 久久精品一区二区三区中文字幕 | 国色天香一区二区| 国产女优一区| 欧美日韩成人激情| 欧美大片免费看| 亚洲成在人线av| 久久香蕉国产线看观看av| 亚洲欧美日韩在线高清直播| 亚洲视频在线看| 亚洲午夜久久久久久尤物| 亚洲精品小视频在线观看| 亚洲精品在线二区| 亚洲亚洲精品三区日韩精品在线视频| 日韩午夜电影av| 亚洲自拍偷拍色片视频| 蜜臀av国产精品久久久久| 久久久久国产一区二区| 欧美成在线观看| 一本久道久久综合狠狠爱| 欧美高清你懂得| 亚洲精品美女久久7777777| 一本久久精品一区二区| 欧美私人网站| 亚洲激情在线播放| 亚洲激情午夜| 亚洲高清二区| 久久国产精品色婷婷| 99re6热只有精品免费观看| 亚洲一区二区三区免费在线观看| 国产精品久久久久免费a∨大胸 | 亚洲精品久久久一区二区三区| 欧美ed2k| 久久在线免费视频| 欧美国产专区| 亚洲天堂成人在线观看| 麻豆精品视频在线| 国产精品视频精品| 亚洲激情小视频| 蜜臀久久99精品久久久久久9| 亚洲视频久久| 欧美不卡高清| 国产欧美三级| 亚洲综合国产激情另类一区| 欧美一区二区三区成人| 亚洲国产婷婷香蕉久久久久久| 免费毛片一区二区三区久久久| 国产女人水真多18毛片18精品视频 | 久久深夜福利| 国产精品视频一二三| 亚洲一区二区欧美| 国产精品99久久久久久久女警| 欧美日韩第一区| 一本色道久久综合一区| 欧美aa在线视频| 亚洲一级二级在线| 国产精品久久国产三级国电话系列 | 99这里只有精品| 欧美精品粉嫩高潮一区二区| 亚洲天堂男人| 午夜在线a亚洲v天堂网2018| 国产欧美一区二区三区另类精品 | 久久夜精品va视频免费观看| 欧美精品免费在线| 国内精品视频久久| 久久综合网hezyo| 噜噜噜噜噜久久久久久91| 国产日韩欧美一区二区三区在线观看| 一区二区福利| 亚洲精品中文字幕女同| 欧美精品一区二| 亚洲伊人伊色伊影伊综合网 | 美女视频一区免费观看| 国产精品乱人伦中文| 亚洲欧美日韩一区二区在线| 午夜精品久久久久久久白皮肤 | 亚洲图片在线观看| 亚洲乱码国产乱码精品精| 国产一区二区欧美| 一本久久知道综合久久| 亚洲精品乱码久久久久| 亚洲午夜伦理| 狠狠入ady亚洲精品| 日韩一本二本av| 国产精品美女午夜av| 欧美国产日产韩国视频| 国产日韩在线播放| 一本一本a久久| 亚洲私人影吧| 国产精品日韩一区二区三区| 久久精品99无色码中文字幕 | 玖玖国产精品视频| 国产精品99久久久久久久久| 午夜欧美大片免费观看| 日韩一级片网址| 欧美片网站免费| 亚洲精品免费在线播放| 亚洲免费福利视频| 欧美日韩一区不卡| 欧美激情中文字幕一区二区 | 欧美制服丝袜| 在线播放不卡| 国产精品美女一区二区在线观看| 亚洲精品免费在线| 狠狠久久亚洲欧美| 久久av老司机精品网站导航| 亚洲欧美韩国| 国产一区二区三区奇米久涩| 久久久爽爽爽美女图片| 91久久夜色精品国产九色| 亚洲精品乱码久久久久久蜜桃91| 欧美日韩在线播| 日韩网站免费观看| 亚洲日本va午夜在线电影| 久久久精品久久久久| 亚洲国产一区二区a毛片| 午夜国产欧美理论在线播放| 在线观看亚洲专区| 欧美性大战久久久久| 欧美岛国在线观看| 亚洲男人第一av网站| 国产欧美日韩精品丝袜高跟鞋| 国产精品a久久久久| 日韩视频在线永久播放| 美女视频网站黄色亚洲| 在线中文字幕一区| 亚洲国产精品成人久久综合一区| 欧美午夜视频在线| 亚洲午夜激情网站| 麻豆精品视频在线观看| 亚洲午夜久久久久久尤物 | 亚洲美女av网站| 欧美一区二区大片| 樱桃视频在线观看一区| 欧美成ee人免费视频| 亚洲精品在线看| 亚洲国产日韩欧美在线动漫| 亚洲一区二区在线免费观看视频| 国产精品久久久久久超碰| 欧美3dxxxxhd| 久热国产精品| 欧美一级网站| 久久久久久久综合日本| 日韩视频免费观看高清完整版| 午夜精品www| 91久久精品国产91久久性色tv | 国产情侣久久| 欧美日韩久久精品| 欧美精品aa| 欧美精品日韩www.p站| 欧美国产综合视频| 国产精品久久久久99| 国产免费观看久久| 国产精品久久久久av免费| 蜜臀久久99精品久久久画质超高清| 欧美在线播放视频| 久久久久国产免费免费| 久久精品国产精品亚洲精品| 久久久99免费视频| 国产精品不卡在线| 欧美日韩一区免费|