## [$AcWing$ $452$. 文化之旅](https://www.acwing.com/problem/content/description/454/) ### 一、题目描述 有一位使者要游历各国,他每到一个国家,都能学到一种文化,但他不愿意学习任何一种文化超过一次(即如果他学习了某种文化,则他就不能到达其他有这种文化的国家)。 不同的国家可能有相同的文化。 不同文化的国家对其他文化的看法不同,**有些文化会排斥外来文化(即如果他学习了某种文化,则他不能到达排斥这种文化的其他国家)**。  > 注:仔细阅读,整明白是谁排斥谁,别整反了。 现给定各个国家间的地理关系,各个国家的文化,每种文化对其他文化的看法,以及这位使者游历的起点和终点(在起点和终点也会学习当地的文化),国家间的道路距离,试求从起点到终点最少需走多少路。 **输入格式** 第一行为五个整数 $N,K,M,S,T$,每两个整数之间用一个空格隔开,依次代表国家个数(国家编号为 $1$ 到 $N$),文化种数(文化编号为 $1$ 到 $K$),道路的条数,以及起点和终点的编号(保证 $S$ 不等于 $T$)。 第二行为 $N$ 个整数,每两个整数之间用一个空格隔开,其中第 $i$ 个数 $C_i$,表示国家 $i$ 的文化为 $C_i$。  接下来的 $K$ 行,每行 $K$ 个整数,每两个整数之间用一个空格隔开,记第 $i$ 行的第 $j$ 个数为 $a_{i,j}$,$a_{i,j}= 1$ 表示文化 $i$ 排斥外来文化 $j$($i$ 等于 $j$ 时表示排斥相同文化的外来人),$a_{i,j}= 0$ 表示不排斥(**注意 $i$ 排斥 $j$ 并不保证 $j$ 一定也排斥 $i$**)。 接下来的 $M$ 行,每行三个整数 $u,v,d$,每两个整数之间用一个空格隔开,表示国家 $u$ 与国家 $v$ 有一条距离为 $d$ 的可双向通行的道路(保证 $u$ 不等于 $v$,**两个国家之间可能有多条道路**)。 **输出格式** 输出只有一行,一个整数,表示使者从起点国家到达终点国家最少需要走的距离数(如果无解则输出 $−1$)。 **数据范围** $2≤N≤100,1≤K≤100,1≤M≤N2,1≤Ci≤K,1≤u,v≤N,1≤d≤1000,1≤S,T≤N,S≠T$ **样例** 输入样例: ```cpp {.line-numbers} 2 2 1 1 2 1 2 0 1 0 0 1 2 10 ``` **输出样例**: ```cpp {.line-numbers} 10 ``` ### 二、算法 **暴力+剪枝+ $floyd$** 有文化限制 不能直接最短路去做 数据范围较小 可以用$floyd$水过 **启发式剪枝** 不考虑限制的最短路距离一定是小于等于考虑限制的距离 $d[i][j]$表示$i->j$的最短距离 $ans$表示程序求出的解 $dist$表示从起点到$u$的距离 那么 $$\large dist + d[u][T] <= ans$$ 那么$dist + d[u][T] >= ans$ 的$ans$就是无用的,可以$continue$ 再加一点玄学优化,倒着搜,正着搜会被卡,其余的直接暴力 ### 三、实现代码 ```cpp {.line-numbers} #include using namespace std; const int N = 110; const int INF = 0x3f3f3f3f; int n, K, m, S, T; int c[N]; // 文化 int cg[N][N]; // 互斥关系 int g[N][N]; // 地图,不一定连通 int d[N][N]; // floyd用的多源最短路数组 int path[N], top; // 记录已经学过的文化 int ans = INF; // 答案 void dfs(int u, int dist) { // 从u出发,dist:已经走的步数 if (dist + d[S][u] >= ans) return; // 启发式剪枝,如果已经走的步数+从u到S的最短距离,已经大于目前的最小答案,那么,此路不通 if (u == S) { // 如果走到了S,收集答案 ans = dist; return; } // 还没有走到起点,从哪个点走过来的 for (int i = 1; i <= n; i++) { if (g[i][u] < INF) { // 可能从i点过来的 bool flag = true; // 文件上没有排斥 for (int j = 0; j < top; j++) { // 遍历一下所有已经走过的文化,是不是存在文化上的冲突 if (cg[path[j]][c[i]]) { // 如果i国家的文化,与后面已知国家的文化冲突,那么一定不是从i过去的 flag = false; // 不可能从i走到u break; } } if (flag) { // 文化上没有排斥 path[top++] = c[i]; // i号城市的文化c[i]进入数组 dfs(i, dist + g[i][u]); // 继续反向搜索 top--; // 回溯 } } } } int main() { // 加快读入 ios::sync_with_stdio(false), cin.tie(0); // 一般在数据量大于10W时考虑用scanf,如果scanf过不去,再考虑使用快读 cin >> n >> K >> m >> S >> T; // 先读入文化 for (int i = 1; i <= n; i++) cin >> c[i]; for (int i = 1; i <= K; i++) // 读入排斥关系 for (int j = 1; j <= K; j++) cin >> cg[i][j]; // 每个文化排斥自己 for (int i = 1; i <= K; i++) cg[i][i] = 1; // 两个国家之间可能有多条道路,可能有重边 memset(g, 0x3f, sizeof g); // 任意两点间距离设置为正无穷,表示不连通 for (int i = 1; i <= n; i++) g[i][i] = 0; // 一开始每个国家都与自己连通 // 读入所有的边 while (m--) { int a, b, c; cin >> a >> b >> c; g[a][b] = g[b][a] = min(g[a][b], c); } // Floyd 用于启发式剪枝,提前预处理出任意两点间的最短距离 memcpy(d, g, sizeof d); for (int k = 1; k <= n; k++) for (int i = 1; i <= n; i++) for (int j = 1; j <= n; j++) d[i][j] = min(d[i][j], d[i][k] + d[k][j]); path[top++] = c[T]; // 直播时忘记加入终点了,初始化终点 dfs(T, 0); // 从后往前搜索,玄学倒序,专克出题人 if (ans == INF) ans = -1; // 没有任何一个能走到S的,说明无解 cout << ans << endl; return 0; } ```