【图论】有向图的强连通分量

算法提高课笔记

文章目录

  • 理论基础
    • SCC板子
  • 例题
    • 受欢迎的牛
      • 题意
      • 思路
      • 代码
    • 学校网络
      • 题意
      • 思路
      • 代码
    • 最大半连通子图
      • 题意
      • 思路
      • 代码
    • 银河
      • 题意
      • 思路
      • 代码

理论基础

什么是连通分量?

对于一个有向图,分量中任意两点u,v,必然可以从u走到v,且从v走到u,这样的分量叫做连通分量

如果一个连通分量加上任意一个点都不是连通分量了,就把它叫做 强连通分量

强连通分量的主要作用:将任意一个有向图转化成一个有向无环图即拓扑图(通过缩点的方式),缩点就是将所有连通分量缩成一个点

如何求强连通分量呢?

按照DFS的顺序搜,我们可以将边分为以下四类:

  1. 树枝边:(x, y),x是y的父结点
  2. 前向边:(x, y),x是y的祖先结点
  3. 后向边:(x, y),y是x的祖先结点
  4. 横叉边:往之前搜过的其他点搜
    在这里插入图片描述
    怎么判断一个点是否在强连通分量中?
  • 情况一:存在后向边指向祖先结点
  • 情况二:先走到横叉边,横叉边再走到祖先结点

(反正一定可以走到某个祖先)

基于这个想法—— Tarjan 算法求强连通分量(SCC)

先给每个结点按照 DFS 访问顺序确定一个时间戳,时间戳越小说明越先访问到

dfn[u]:遍历到 u 的时间戳
low[u]:从 u 开始走,能遍历到的最小时间戳
id[i]:i 所在连通分量的编号

u是其所在强连通分量的最高点 <-> dfb[u] == low[u]

SCC板子

void tarjan(int u)
{dfn[u] = low[u] = ++ timestamp; // 先将dfn和low都初始化为时间戳stk.push(u), in_stk[u] = true; // u加入栈中for (int i = h[u]; ~i; i = ne[i]){int j = e[i]; // 取出u的所有邻点jif (!dfn[j]) // 如果j还没被遍历{tarjan(j);low[u] = min(low[u], low[j]); // 用low[j]更新low[u]}else if (in_stk[j]) low[u] = min(low[u], dfn[j]); // 如果j已入栈 则用dfn[j]更新low[u]}if (dfn[u] == low[u]) // 如果该点是所在强连通分量的最高点{++ scc_cnt; // 强连通分量数量加一int y;do {y = stk.top(); // 取出栈顶元素stk.pop();in_stk[y] = false;id[y] = scc_cnt; // 标记每个点所在的连通分量编号} while (y != u); // 直到取到此连通分量的最高点为止}
}

缩点的步骤:

  1. 遍历所有点 i
  2. 遍历 i 的所有邻点 j
  3. 如果 i 和 j 不在同一个连通分量中,就加一条新边 id[i]->id[j]
for (int i = 1; i <= n; i ++ )for (int j = h[i]; ~j; j = ne[j]){int k = e[j]; // 遍历i的所有邻点kint a = id[i], b = id[k]; // 记录ik所在连通分量编号if (a != b) dout[a] ++ ; // 如果ik不在同一个连通分量 就在两个连通分量之间连一条i指向k的边}

做完tarjon后,连通分量编号递减的顺序一定就是拓扑序

例题

受欢迎的牛

原题链接

每一头牛的愿望就是变成一头最受欢迎的牛。

现在有 N 头牛,编号从 1 到 N,给你 M 对整数 (A,B),表示牛 A 认为牛 B 受欢迎。

这种关系是具有传递性的,如果 A 认为 B 受欢迎,B 认为 C 受欢迎,那么牛 A 也认为牛 C 受欢迎。

你的任务是求出有多少头牛被除自己之外的所有牛认为是受欢迎的。

输入格式

第一行两个数 N,M;

接下来 M 行,每行两个数 A,B,意思是 A 认为 B 是受欢迎的(给出的信息有可能重复,即有可能出现多个 A,B)。

输出格式

输出被除自己之外的所有牛认为是受欢迎的牛的数量。

数据范围

1 ≤ N ≤ 104 , 1≤N≤104, 1N104,
1 ≤ M ≤ 5 × 104 1≤M≤5×104 1M5×104

输入样例

3 3
1 2
2 1
2 3

输出样例

1

样例解释

只有第三头牛被除自己之外的所有牛认为是受欢迎的。

题意

一头牛会欢迎另一头牛,这种欢迎是有传递性的,现给出多对欢迎关系,问有几头牛是被其余所有牛欢迎的

思路

先对连通分量进行缩点操作

之后分析,如果有唯一一个连通分量满足出度为0,那么这个连通分量内的所有点都可以由其余任意点到达,答案就是这个连通分量内点的个数,如果这样的连通分量超过一个就不满足条件

代码

#include <bits/stdc++.h>using namespace std;const int  N = 10010, M = 50010;int n, m;
int h[N], ne[M], e[M], idx;
int dfn[N], low[N], timestamp;
stack<int> stk;
bool in_stk[N]; // 存储点是否入栈
int id[N], scc_cnt, Size[N];
int dout[N]; // 连通分量的出度void add(int a, int b)
{e[idx] = b, ne[idx] = h[a], h[a] = idx ++ ;
}void tarjan(int u)
{dfn[u] = low[u] = ++ timestamp; // 先将dfn和low都初始化为时间戳stk.push(u), in_stk[u] = true; // u加入栈中for (int i = h[u]; ~i; i = ne[i]){int j = e[i]; // 取出u的所有邻点jif (!dfn[j]) // 如果j还没被遍历{tarjan(j);low[u] = min(low[u], low[j]); // 用low[j]更新low[u]}else if (in_stk[j]) low[u] = min(low[u], dfn[j]); // 如果j已入栈 则用dfn[j]更新low[u]}if (dfn[u] == low[u]) // 如果该点是所在强连通分量的最高点{++ scc_cnt; // 强连通分量数量加一int y;do {y = stk.top(); // 取出栈顶元素stk.pop();in_stk[y] = false;id[y] = scc_cnt; // 标记每个点所在的连通分量编号Size[scc_cnt] ++ ; // 更新此连通分量中的点个数} while (y != u); // 直到取到此连通分量的最高点为止}
}int main()
{ios::sync_with_stdio(false);cin.tie(0), cout.tie(0);cin >> n >> m;memset(h, -1, sizeof h);while (m -- ){int a, b;cin >> a >> b;add(a, b);}for (int i = 1; i <= n; i ++ )if (!dfn[i]) tarjan(i);for (int i = 1; i <= n; i ++ )for (int j = h[i]; ~j; j = ne[j]){int k = e[j]; // 遍历i的所有邻点kint a = id[i], b = id[k]; // 记录ik所在连通分量编号if (a != b) dout[a] ++ ; // 如果ik不在同一个连通分量 就在两个连通分量之间连一条i指向k的边}int zeros = 0, sum = 0;for (int i = 1; i <= scc_cnt; i ++ )if (!dout[i]) // 如果当前连通分量出度为0{zeros ++ ; // 出度为0的连通分量个数加一sum += Size[i]; // 更新出度为0的连通分量中点的个数if (zeros > 1) // 如果出度为0的连通分量个数超过一个 说明没有一头牛被所有牛喜欢{sum = 0;break;}}cout << sum << '\n';
}

学校网络

原题链接

一些学校连接在一个计算机网络上,学校之间存在软件支援协议,每个学校都有它应支援的学校名单(学校 A 支援学校 B,并不表示学校 B 一定要支援学校 A)。

当某校获得一个新软件时,无论是直接获得还是通过网络获得,该校都应立即将这个软件通过网络传送给它应支援的学校。

因此,一个新软件若想让所有学校都能使用,只需将其提供给一些学校即可。

现在请问最少需要将一个新软件直接提供给多少个学校,才能使软件能够通过网络被传送到所有学校?

最少需要添加几条新的支援关系,使得将一个新软件提供给任何一个学校,其他所有学校就都可以通过网络获得该软件?

输入格式

第 1 行包含整数 N,表示学校数量。

第 2…N+1 行,每行包含一个或多个整数,第 i+1 行表示学校 i 应该支援的学校名单,每行最后都有一个 0 表示名单结束(只有一个 0 即表示该学校没有需要支援的学校)。

输出格式

输出两个问题的结果,每个结果占一行。

数据范围

2 ≤ N ≤ 100 2≤N≤100 2N100

输入样例

5
2 4 3 0
4 5 0
0
0
1 0

输出样例

1
2

题意

给出一张图:

  • 问题一:至少从多少个点出发能够遍历完图上所有点
  • 问题二:至少加多少条边能让图的强连通分量就是自身

思路

设入度为0的连通分量个数为a,出度为0的连通分量个数为b

问题一就是问a的大小

问题二就是问ab中较大的值(需要特判一下如果只有一个强连通分量,就不需要加边,输出0即可 )
举个栗子 不具体证明了:在这里插入图片描述

代码

#include <bits/stdc++.h>using namespace std;const int  N = 110, M = 10010;int n, m;
int h[N], ne[M], e[M], idx;
int dfn[N], low[N], timestamp;
stack<int> stk;
bool in_stk[N]; // 存储点是否入栈
int id[N], scc_cnt;
int din[N], dout[N]; // 连通分量的入度和出度void add(int a, int b)
{e[idx] = b, ne[idx] = h[a], h[a] = idx ++ ;
}void tarjan(int u)
{dfn[u] = low[u] = ++ timestamp; // 先将dfn和low都初始化为时间戳stk.push(u), in_stk[u] = true; // u加入栈中for (int i = h[u]; ~i; i = ne[i]){int j = e[i]; // 取出u的所有邻点jif (!dfn[j]) // 如果j还没被遍历{tarjan(j);low[u] = min(low[u], low[j]); // 用low[j]更新low[u]}else if (in_stk[j]) low[u] = min(low[u], dfn[j]); // 如果j已入栈 则用dfn[j]更新low[u]}if (dfn[u] == low[u]) // 如果该点是所在强连通分量的最高点{++ scc_cnt; // 强连通分量数量加一int y;do {y = stk.top(); // 取出栈顶元素stk.pop();in_stk[y] = false;id[y] = scc_cnt; // 标记每个点所在的连通分量编号} while (y != u); // 直到取到此连通分量的最高点为止}
}int main()
{ios::sync_with_stdio(false);cin.tie(0), cout.tie(0);cin >> n;memset(h, -1, sizeof h);for (int i = 1; i <= n; i ++ ) // 建图{int t;while (cin >> t, t) add(i, t);}for (int i = 1; i <= n; i ++ )if (!dfn[i]) tarjan(i);for (int i = 1; i <= n; i ++ )for (int j = h[i]; ~j; j = ne[j]){int k = e[j]; // 遍历i的所有邻点kint a = id[i], b = id[k]; // 记录ik所在连通分量编号if (a != b) // 如果ik不在同一个连通分量 就在两个连通分量之间连一条i指向k的边{dout[a] ++ ;din[b] ++ ;}}int a = 0, b = 0;for (int i = 1; i <= scc_cnt; i ++ ){if (!din[i]) a ++ ; // 记录入度为0的点个数if (!dout[i]) b ++ ; // 记录出度为0的点个数}cout << a << '\n';if (scc_cnt == 1) cout << 0 << '\n';else cout << max(a, b) << '\n';
}

最大半连通子图

原题链接

一个有向图 G=(V,E) 称为半连通的 (Semi-Connected),如果满足:∀u,v∈V,满足 u→v 或 v→u,即对于图中任意两点 u,v,存在一条 u 到 v 的有向路径或者从 v 到 u 的有向路径。

若 G′=(V′,E′) 满足,E′ 是 E 中所有和 V′ 有关的边,则称 G′ 是 G 的一个导出子图。

若 G′ 是 G 的导出子图,且 G′ 半连通,则称 G′ 为 G 的半连通子图。

若 G′ 是 G 所有半连通子图中包含节点数最多的,则称 G′ 是 G 的最大半连通子图。

给定一个有向图 G,请求出 G 的最大半连通子图拥有的节点数 K,以及不同的最大半连通子图的数目 C。

由于 C 可能比较大,仅要求输出 C 对 X 的余数。

输入格式

第一行包含三个整数 N,M,X。N,M 分别表示图 G 的点数与边数,X 的意义如上文所述;

接下来 M 行,每行两个正整数 a,b,表示一条有向边 (a,b)。

图中的每个点将编号为 1 到 N,保证输入中同一个 (a,b) 不会出现两次。

输出格式

应包含两行。

第一行包含一个整数 K,第二行包含整数 C mod X。

数据范围

1 ≤ N ≤ 105 , 1≤N≤105, 1N105,
1 ≤ M ≤ 106 , 1≤M≤106, 1M106,
1 ≤ X ≤ 108 1≤X≤108 1X108

输入样例

6 6 20070603
1 2
2 1
1 3
2 4
5 6
6 4

输出样例

3
3

题意

找出最大的半连通子图,输出节点数和子图个数

思路

最大半连通子图就是图上最长的一条链,所以按照以下思路:

  1. tarjan
  2. 缩点、建图、给边判重
  3. 按拓扑序递推

代码

#include <bits/stdc++.h>using namespace std;typedef long long ll;const int  N = 100010, M = 2000010;int n, m, mod;
int h[N], hs[N], ne[M], e[M], idx;
int dfn[N], low[N], timestamp;
stack<int> stk;
bool in_stk[N]; // 存储点是否入栈
int id[N], scc_cnt, scc_size[N];
int f[N], g[N]; // 连通分量的入度和出度void add(int h[], int a, int b)
{e[idx] = b, ne[idx] = h[a], h[a] = idx ++ ;
}void tarjan(int u)
{dfn[u] = low[u] = ++ timestamp; // 先将dfn和low都初始化为时间戳stk.push(u), in_stk[u] = true; // u加入栈中for (int i = h[u]; ~i; i = ne[i]){int j = e[i]; // 取出u的所有邻点jif (!dfn[j]) // 如果j还没被遍历{tarjan(j);low[u] = min(low[u], low[j]); // 用low[j]更新low[u]}else if (in_stk[j]) low[u] = min(low[u], dfn[j]); // 如果j已入栈 则用dfn[j]更新low[u]}if (dfn[u] == low[u]) // 如果该点是所在强连通分量的最高点{++ scc_cnt; // 强连通分量数量加一int y;do {y = stk.top(); // 取出栈顶元素stk.pop();in_stk[y] = false;id[y] = scc_cnt; // 标记每个点所在的连通分量编号scc_size[scc_cnt] ++ ;} while (y != u); // 直到取到此连通分量的最高点为止}
}int main()
{ios::sync_with_stdio(false);cin.tie(0), cout.tie(0);memset(h, -1, sizeof h);memset(hs, -1, sizeof hs);cin >> n >> m >> mod;while (m -- ){int a, b;cin >> a >> b;add(h, a, b);}for (int i = 1; i <= n; i ++ )if (!dfn[i]) tarjan(i);unordered_set<ll> S;for (int i = 1; i <= n; i ++ )for (int j = h[i]; ~j; j = ne[j]){int k = e[j]; // 遍历i的所有邻点kint a = id[i], b = id[k]; // 记录ik所在连通分量编号ll hash = a * 1000000ll + b;if (a != b && !S.count(hash)) // 如果ik不在同一个连通分量 就在两个连通分量之间连一条i指向k的边{add(hs, a, b);S.insert(hash);}}for (int i = scc_cnt; i; i -- ){if (!f[i]) // 起点{f[i] = scc_size[i]; // 更新最大半连通子图内元素个数g[i] = 1; // 更新最大半连通子图个数}for (int j = hs[i]; ~j; j = ne[j]){int k = e[j];if (f[k] < f[i] + scc_size[k]) // 有更优解{f[k] = f[i] + scc_size[k];g[k] = g[i];}else if (f[k] == f[i] + scc_size[k]) // 有结果一样的解g[k] = (g[k] + g[i]) % mod;}}int maxf = 0, sum = 0;for (int i = 1; i <= scc_cnt; i ++ )if (f[i] > maxf){maxf = f[i];sum = g[i];}else if (f[i] == maxf) sum = (sum + g[i]) % mod;cout << maxf << '\n';cout << sum << '\n';
}

银河

原题链接

银河中的恒星浩如烟海,但是我们只关注那些最亮的恒星。

我们用一个正整数来表示恒星的亮度,数值越大则恒星就越亮,恒星的亮度最暗是 1。

现在对于 N 颗我们关注的恒星,有 M 对亮度之间的相对关系已经判明。

你的任务就是求出这 N 颗恒星的亮度值总和至少有多大。

输入格式

第一行给出两个整数 N 和 M。

之后 M 行,每行三个整数 T,A,B,表示一对恒星 (A,B) 之间的亮度关系。恒星的编号从 1 开始。

  • 如果 T=1,说明 A 和 B 亮度相等。
  • 如果 T=2,说明 A 的亮度小于 B 的亮度。
  • 如果 T=3,说明 A 的亮度不小于 B 的亮度。
  • 如果 T=4,说明 A 的亮度大于 B 的亮度。
  • 如果 T=5,说明 A 的亮度不大于 B 的亮度。

输出格式

输出一个整数表示结果。

若无解,则输出 −1。

数据范围

N ≤ 100000 , M ≤ 100000 N≤100000,M≤100000 N100000,M100000

输入样例

5 7 
1 1 2 
2 3 2 
4 4 1 
3 4 5 
5 4 5 
2 3 5 
4 5 1 

输出样例

11

题意

求图中有无正环的问题

思路

  1. 用tarjan求强连通分量
  2. 缩点、根据差分约束建图
  3. 依据拓扑序递推

代码

#include <bits/stdc++.h>using namespace std;typedef long long ll;const int  N = 100010, M = 600010;int n, m;
int h[N], hs[N], w[M], ne[M], e[M], idx;
int dfn[N], low[N], timestamp;
stack<int> stk;
bool in_stk[N]; // 存储点是否入栈
int id[N], scc_cnt, scc_size[N];
int dist[N]; // 连通分量的入度和出度void add(int h[], int a, int b, int c)
{e[idx] = b, w[idx] = c, ne[idx] = h[a], h[a] = idx ++ ;
}void tarjan(int u)
{dfn[u] = low[u] = ++ timestamp; // 先将dfn和low都初始化为时间戳stk.push(u), in_stk[u] = true; // u加入栈中for (int i = h[u]; ~i; i = ne[i]){int j = e[i]; // 取出u的所有邻点jif (!dfn[j]) // 如果j还没被遍历{tarjan(j);low[u] = min(low[u], low[j]); // 用low[j]更新low[u]}else if (in_stk[j]) low[u] = min(low[u], dfn[j]); // 如果j已入栈 则用dfn[j]更新low[u]}if (dfn[u] == low[u]) // 如果该点是所在强连通分量的最高点{++ scc_cnt; // 强连通分量数量加一int y;do {y = stk.top(); // 取出栈顶元素stk.pop();in_stk[y] = false;id[y] = scc_cnt; // 标记每个点所在的连通分量编号scc_size[scc_cnt] ++ ;} while (y != u); // 直到取到此连通分量的最高点为止}
}int main()
{ios::sync_with_stdio(false);cin.tie(0), cout.tie(0);memset(h, -1, sizeof h);memset(hs, -1, sizeof hs);cin >> n >> m;for (int i = 1; i <= n; i ++ ) add(h, 0, i, 1);while (m -- ){int t, a, b;cin >> t >> a >> b;if (t == 1) add(h, b, a, 0), add(h, a, b, 0);else if (t == 2) add(h, a, b, 1);else if (t == 3) add(h, b, a, 0);else if (t == 4) add(h, b, a, 1);else add(h, a, b, 0);}tarjan(0);bool success = true;for (int i = 0; i <= n; i ++ ){for (int j = h[i]; ~j; j = ne[j]){int k = e[j]; // 遍历i的所有邻点kint a = id[i], b = id[k]; // 记录ik所在连通分量编号if (a == b) // ik在同一个强连通分量{if (w[j] > 0) // ik间有权值为正的路径{success = false; // 无解break;}}else add(hs, a, b, w[j]);// 如果ik不在同一个连通分量 就在两个连通分量之间连一条i指向k的边}if (!success) break;}if (!success) cout << "-1\n";else // 有解则求最长路{for (int i = scc_cnt; i; i -- ){for (int j = hs[i]; ~j; j = ne[j]){int k = e[j];dist[k] = max(dist[k], dist[i] + w[j]);}}ll res = 0;for (int i = 1; i <= scc_cnt; i ++ ) res += (ll)dist[i] * scc_size[i];cout << res << '\n';}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/110262.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

结构体成绩排序

任务描述 本关任务&#xff1a;有n个学生的信息(包括学号&#xff0c;姓名&#xff0c;成绩)&#xff0c;要求按照成绩的高低顺序输出学生的信息。 相关知识 参考结构体第1关相关知识 编程要求 在右侧编辑器Begin-End处补充代码&#xff0c;要求按照成绩的高低顺序输出学生…

【深度学习】 Python 和 NumPy 系列教程(廿七):Matplotlib详解:3、多子图和布局:散点矩阵图(Scatter Matrix Plot)

目录 一、前言 二、实验环境 三、Matplotlib详解 1、2d绘图类型 2、3d绘图类型 3、多子图和布局 1. subplot()函数 2. subplots()函数 3. 散点矩阵图&#xff08;Scatter Matrix Plot&#xff09; 一、前言 Python是一种高级编程语言&#xff0c;由Guido van Rossum于…

视屏点播项目

项目背景 大家应该在电脑上刷过视频吧,这个项目就是模拟一下我们刷视频的整个流程,我们要做的是一个类似B站的网页,这里面包含视频的上传修改和观看以及删除,注意我这个是一个简易版本的,在后面我会做一个升级,增加其他的功能. 基本原理 下面我们说一下我们项目的基本原理.我…

如何根据性能需求进行场景设计?

场景设计一 探索 测试环境 客户端: win10 这里可以用linux,但没用,因为想直观查看结果。 被测环境:linux X86 4核CPU16G内存 被测接口:登录接口,没有做数据驱动。 在测试执行前,先使用influxSQL把influxdb的数据清理一下,以防影响结果查看。 有这么一个需求,要求系…

Android 12 源码分析 —— 应用层 五(SystemUI的StatusBar类的启动过程和三个窗口的创建)

Android 12 源码分析 —— 应用层 五&#xff08;SystemUI的StatusBar类的启动过程和三个窗口的创建&#xff09; 在前面的文章中&#xff0c;我们介绍了SystemUI App的基本布局和基本概念。接下来&#xff0c;我们进入SystemUI应用的各个UI是如何被加入屏幕的。那么我们就先从…

SOLIDWORKS Composer反转关键帧实现产品安装过程

SOLIDWORKS Composer 是一款被用来制作交互式产品说明书的工具&#xff0c;可以帮助我们对产品设定精确的机构动画&#xff0c;并能根据材质生成一定细节的渲染图像。 今天我们主要向大家讲解的是&#xff0c;利用SOLIDWORKS Composer关键帧反转实现产品动态的安装。 一般情况下…

微信小程序通过 wxministore 实现类似于vuex的全局装填数据管理

首先 我们打开终端 引入依赖 npm install wxministore --save然后 如果你是新版开发者工具 就 npm i构建一下 如果你是 老版本的 微信开发者工具 就打开右上角详情 选择本地管理 勾选 使用 npm 模块 然后 在根目录下创建一个 store.js 当然建在哪是你自己决定的 反正 后面能…

密集人头检测数据集汇总和格式转换

1、VSCrowd 2022年9月新出的数据集,数据集链接:https://github.com/HopLee6/VSCrowd-Dataset 网盘地址:链接:https://pan.baidu.com/s/17VARxt59y7GnUHIskEGzKw?pwd=m9qo 提取码:m9qo 数据格式: FrameID HeadID x1 y1 x2 y2 p1 p2 HeadID x1 y1 x2 y2 p1 p2 … Fram…

长城网络靶场第三题

关卡描述&#xff1a;1.oa服务器的内网ip是多少&#xff1f; 先进行ip统计&#xff0c;开始逐渐查看前面几个ip 基本上都是b/s&#xff0c;所以大概率是http&#xff0c;过滤一下ip 第一个ip好像和oa没啥关系 第二个ip一点开就是 oa&#xff0c;应该就是他了。 关卡描述&a…

数据仓库模型设计V2.0

一、数仓建模的意义 数据模型就是数据组织和存储方法&#xff0c;它强调从业务、数据存取和使用角度合理存储数据。只有将数据有序的组织和存储起来之后&#xff0c;数据才能得到高性能、低成本、高效率、高质量的使用。 高性能&#xff1a;良好的数据模型能够帮助我们快速查询…

spark6. 如何设置spark 日志

spark yarn日志全解 一.前言二.开启日志聚合是什么样的2.1 开启日志聚合MapReduce history server2.2 如何开启Spark history server 三.不开启日志聚合是什么样的四.正确使用log4j.properties 一.前言 本文只讲解再yarn 模式下的日志配置。 二.开启日志聚合是什么样的 在ya…

基于SSM+Vue的校园教务系统的设计与实现

末尾获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端框架&#xff1a;SSM 前端&#xff1a;采用Vue技术开发 数据库&#xff1a;MySQL5.7和Navicat管理工具结合 服务器&#xff1a;Tomcat8.5 开发软件&#xff1a;IDEA / Eclipse 是否Maven项目&#x…