前向星和鏈式前向星(詳解 模板)

2021-10-07 02:00:10 字數 2761 閱讀 5232

前向星是一種特殊的邊集陣列,我們把邊集陣列中的每一條邊按照起點從小到大排序,如果起點相同就按照終點從小到大排序,並記錄下以某個點為起點的所有邊在陣列中的起始位置和儲存長度,那麼前向星就構造好了。

用len[i]來記錄所有以i為起點的邊在陣列中的儲存長度。

用head[i]記錄以i為邊集在陣列中的第乙個儲存位置。

樣例:

其中邊的輸入順序:

1 2

2 33 4

1 34 1

1 54 5

我們對其中的邊按照定義排序完後:

編號: 1 2 3 4 5 6 7

起點u: 1 1 1 2 3 4 4

終點v: 2 3 5 3 4 1 5

那麼,前向星存圖有什麼優勢嗎?利用前向星,我們可以在o(1)的時間內找到以i為起點的第一條邊以o(len[i])的時間找到以i為起點的所有邊。前向星特別適合用來優化spfa、dfs以及bfs。

但是,在這裡有乙個問題,就是前向星還是需要加上排序的時間,如果是快排大概是o(n*log(n));而鏈式前向星則不需要排序也能得到。

我覺得,直接不好理解鏈式前向星,我們先來看其**,然後通過其**來體味其中的道理:

struct nodeedge[maxn]

;int cnt =0;

int head[maxn]

;//head初始化為-1

void

add(

int u,

int v,

int w)

其中:

edge[i].to表示第i條邊的終點

edge[i].next表示與第i條邊同起點的下一條邊的儲存位置

edge[i].w為邊的權值。

**陣列head**它是用來表示以i為起點的第一條邊儲存的位置

注意head陣列一般初始化為-1

給出**我們可能還是很難理解,我們用前面那個例子模擬一遍就會有所理解了:

樣例:

其中邊的輸入順序:

1 2

2 33 4

1 34 1

1 54 5

edge[0].to = 2; edge[0].next = -1; head[1] = 0;

edge[1].to = 3; edge[1].next = -1; head[2] = 1;

edge[2].to = 4; edge[2],next = -1; head[3] = 2;

edge[3].to = 3; edge[3].next = 0; head[1] = 3;

edge[4].to = 1; edge[4].next = -1; head[4] = 4;

edge[5].to = 5; edge[5].next = 3; head[1] = 5;

edge[6].to = 5; edge[6].next = 4; head[4] = 6;

這裡我們可以看出來,head[i]儲存的是以i為起點的所有邊中編號最大的那個,而把這個當作頂點i的第一條起始邊的位置。而head[i]中的值,就是代表以i為起點的所有邊中編號最大的那條邊在edge陣列中的下標。

我們再來看看怎麼遍歷:

for

(int i=head[u]

;~i;i=edge[i]

.next)

比如這裡u=1,指的是我們遍歷所有以1為起點的邊,首先是編號為5的邊(edge[5]),其中點是5(edge[5].to = 5),下乙個是編號為3的邊(edge[5].next = 3)…直到edge.next = -1。至此就遍歷完了所有以1為起點的邊。

這會,我們再來回過頭看鏈式前向星的**就很好理解了:

首先,對於add函式中前兩行**,其目的就是為了儲存該條邊的權值及其終點。關鍵在於下面兩行**:在更新head[u]之前我們需要先儲存當前的head[u],然後再更新head的編號(cnt)。

/******鏈式前向星建圖********/

struct nodeedge[maxn]

;int cnt =0;

int head[maxn]

;//head初始化為-1

void

add(

int u,

int v,

int w)

/***************/

int dis[maxn]

;//dis[i]=源點s->i最短路徑

bool vis[maxn]

;//vis[i]表示i是否在佇列

void

spfa

(int s)

dis[s]=0

;//源點到自身距離為0

queue<

int>q;

//使用c++自帶佇列

q.push

(s);

//源點入隊

vis[s]

=false

;while

(!q.

empty()

)//若佇列不為空}}

}}

前向星和鏈式前向星

我們首先來看一下什麼是前向星.前向星是一種特殊的邊集陣列,我們把邊集陣列中的每一條邊按照起點從小到大排序,如果起點相同就按照終點從小到大排序,並記錄下以某個點為起點的所有邊在陣列中的起始位置和儲存長度,那麼前向星就構造好了.用len i 來記錄所有以i為起點的邊在陣列中的儲存長度.用head i 記...

前向星和鏈式前向星

前向星 前向星是一種特殊的邊集陣列,我們把邊集陣列中的每一條邊按照起點從小到大排序,如果起點相同就按照終點從小到大排序,並記錄下以某個點為起點的所有邊在陣列中的起始位置。鏈式前向星 鏈式前向星其實就是靜態建立的鄰接表,時間效率為o m 空間效率也為o m 遍歷效率也為o m next表示當前結點的下...

前向星和鏈式前向星

1 前向星 前向星是以儲存邊的方式來儲存圖,先將邊讀入並儲存在連續的陣列中,然後按照邊的起點進行排序,這樣陣列中起點相等的邊就能夠在陣列中進行連續訪問了。它的優點是實現簡單,容易理解,缺點是需要在所有邊都讀入完畢的情況下對所有邊進行一次排序,帶來了時間開銷,實用性也較差,只適合離線演算法。圖一 2 ...