日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
最短路怎么可能盡可能地長呢?

[[416100]]

創(chuàng)新互聯(lián)公司是一家專業(yè)提供德興企業(yè)網(wǎng)站建設(shè),專注與網(wǎng)站制作、成都網(wǎng)站設(shè)計、HTML5建站、小程序制作等業(yè)務(wù)。10年已為德興眾多企業(yè)、政府機構(gòu)等服務(wù)。創(chuàng)新互聯(lián)專業(yè)網(wǎng)站制作公司優(yōu)惠進行中。

本文轉(zhuǎn)載自微信公眾號「Piper蛋窩」,作者Piper蛋。轉(zhuǎn)載本文請聯(lián)系Piper蛋窩公眾號。

記錄一道題解, 題目來自 Acwing.com 第 11 場周賽.

https://www.acwing.com/activity/content/59/

沒參加. 如果讓我做的話我做不出來, 難度是困難, 不是一道模板題, 用的知識點 bfs 啥的都簡單, 但是考的是分析能力.

我的筆記:

https://github.com/PiperLiu/ACMOI_Journey/tree/master/notes

最大化最短路[1]

給定一個 個點 條邊的無向連通圖。

圖中所有點的編號為 。

圖中不含重邊和自環(huán)。

指定圖中的 個點為特殊點。

現(xiàn)在,你必須選擇兩個特殊點,并在這兩個點之間增加一條邊。

所選兩點之間允許原本就存在邊。

我們希望,在增邊操作完成以后,點 到點 的最短距離盡可能大。

輸出這個最短距離的最大可能值。

注意,圖中所有邊(包括新增邊)的邊長均為 。

輸入格式

第一行包含三個整數(shù) 。

第二行包含 個整數(shù) ,表示 個特殊點的編號, 之間兩兩不同。

接下來 行,每行包含兩個整數(shù) ,表示點 和點 之間存在一條邊。

輸出格式

一個整數(shù),表示最短距離的最大可能值。

數(shù)據(jù)范圍

前六個測試點滿足 。

所有測試點滿足 ,,,,。

輸入樣例1:

 
 
 
 
  1. 5 5 3 
  2. 1 3 5 
  3. 1 2 
  4. 2 3 
  5. 3 4 
  6. 3 5 
  7. 2 4 

輸出樣例1:

 
 
 
 

輸入樣例2:

 
 
 
 
  1. 5 4 2 
  2. 2 4 
  3. 1 2 
  4. 2 3 
  5. 3 4 
  6. 4 5 

輸出樣例2:

 
 
 
 

競賽中等難度題目,重點在分析。

分析第一步,分情況討論。

題目中要求,必須在特殊點中選擇兩個點,這兩個點之間會新增一條邊。優(yōu)化目標(biāo)是,新增邊后, 1 到 n 的最短路徑最大。

從 1 到 n 的最短路徑只可能有以下三種情況(如上圖):

  • 不經(jīng)過 a to b 這條線
  • 經(jīng)過 a -> b ,則距離是 x[a] + 1 + y[b]
  • 經(jīng)過 b -> a ,則距離是 x[b] + 1 + y[a]
  • 說明:x[a] 為 1 到 a 的距離,y[b] 為 n 到 b 的距離

如果我們在 a 與 b 中增加一條邊,則最終最短路的距離為以下三者中取最小值:

  • 原有最短路長度
  • x[a] + 1 + y[b]
  • x[b] + 1 + y[a]

我們沒辦法改變「原有最短路長度」,因此只能希望 min(x[a] + 1 + y[b], x[b] + 1 + y[a]) 這個值越大越好。

因此,我們要考慮所有特殊點的兩兩組合,然后,找出最大的 min(x[a] + 1 + y[b], x[b] + 1 + y[a]) 的 a b 組合。

找兩兩組合

我們沒法直接找 min(x[a] + 1 + y[b], x[b] + 1 + y[a]) 的最大值,得進行一步推導(dǎo):

  • x[a] + 1 + y[b] <= x[b] + 1 + y[a]
  • 即 x[a] - y[a] <= x[b] - y[b]
  • 即,當(dāng) x[a] - y[a] <= x[b] - y[b] 時, min(x[a] + 1 + y[b], x[b] + 1 + y[a]) 為 x[a] + 1 + y[b]
  • 即,我們找 a, b 滿足 x[a] - y[a] <= x[b] - y[b] (這個約束條件也可使我們遍歷所有的 a, b 組合),使得 x[a] + 1 + y[b] 最大

找兩兩組合

如上,我們將特殊的點按照 x[i] - y[i] 升序排序;我們令 b 為第一層循環(huán):即首先確定 b 的位置(圖中為 i ) , a 的話,選擇選擇從起點到 i 的最大值即可,因為我們的目標(biāo)是圖中紅色的線值最大,即 y[b] + 1 + x[a] 。

 
 
 
 
  1. #include  
  2. #include  
  3. #include  
  4. using namespace std; 
  5.  
  6. const int N = 2e5 + 10, M = 4e5 + 10; 
  7. int n, m, k; 
  8. int a[N], dist1[N], dist2[N];  // 特殊點,題解中的x[]和y[] 
  9. int h[N], e[M], ne[M], idx; 
  10. int q[N];  // bfs 用的隊列 
  11.  
  12. void add(int a, int b) 
  13.     e[idx] = b, ne[idx] = h[a], h[a] = idx ++ ; 
  14.  
  15. void bfs(int st, int dist[]) 
  16.     int tt = 0, hh = 0; 
  17.     q[0] = st; 
  18.      
  19.     // 傳入?yún)?shù)的 dist 是一個指針 
  20.     // 不可以用 sizeof dist 
  21.     memset(dist, 0x3f, 4 * N); 
  22.     dist[st] = 0; 
  23.  
  24.     while (hh <= tt) 
  25.     { 
  26.         int t = q[hh ++]; 
  27.         // printf("t = %d h[t] = %d \n", t, h[t]); 
  28.          
  29.         for (int i = h[t]; ~i; i = ne[i]) 
  30.         { 
  31.             int j = e[i]; 
  32.             if (dist[j] > dist[t] + 1) 
  33.             { 
  34.                 dist[j] = dist[t] + 1; 
  35.                 // printf("dist[%d] = %d t = %d\n", j, dist[j], tt); 
  36.                 q[++ tt] = j; 
  37.             } 
  38.         } 
  39.     } 
  40.  
  41. int main() 
  42.     scanf("%d%d%d", &n, &m, &k); 
  43.     memset(h, -1, sizeof h);  // 莫忘! 
  44.     for (int i = 0; i < k; ++ i) 
  45.     { 
  46.         scanf("%d", &a[i]); 
  47.     } 
  48.     for (int i = 0; i < m; ++ i) 
  49.     { 
  50.         int x, y; 
  51.         scanf("%d%d", &x, &y); 
  52.         add(x, y); 
  53.         add(y, x); 
  54.     } 
  55.  
  56.     bfs(1, dist1); 
  57.     // printf("==bfs2\n"); 
  58.     bfs(n, dist2); 
  59.      
  60.     // 開始按照題解來,先按照 dist1[i] - dist2[i] 排序 
  61.     sort(a, a + k, [&](int a, int b "&") { 
  62.         return dist1[a] - dist2[a] < dist1[b] - dist2[b]; 
  63.     }); 
  64.      
  65.     // b 作為最外層循環(huán),找最大的 dist1[a] + 1 + dist2[b] 
  66.     int x = dist1[a[0]], res = 0;  // 對于第 b = 第一個點,a 也只能為第 0 個點(這里 x 是題解中紅線的左上端點) 
  67.     for (int i = 1; i < k; i ++ ) 
  68.     { 
  69.         int t = a[i];  // 這里 dist2[t] 是題解中紅線的右下端點 
  70.         res = max(res, dist2[t] + 1 + x); 
  71.         x = max(dist1[t], x); 
  72.     } 
  73.      
  74.     // 最后與本來的最短路比較 
  75.     res = min(res, dist1[n]); 
  76.      
  77.     printf("%d", res); 

經(jīng)驗:

這里,我們將數(shù)組傳入函數(shù) int dist[] ,不能使用 memset(dist, 0x3f, sizeof dist); 因為 dist 僅僅是一個指針,而非數(shù)組;我們的 dist 長度為 N ,且為 int 類型,因此 memset(dist, 0x3f, N * 4);

參考資料

[1]最大化最短路: https://www.acwing.com/problem/content/3800/

 


當(dāng)前文章:最短路怎么可能盡可能地長呢?
網(wǎng)址分享:http://www.dlmjj.cn/article/djoiijc.html