SDUT 2023 summer team contest(for 22) - 14

news/2024/7/20 22:37:29 标签: 深度优先, 算法

A - Amanda Lounges

 题意 有n个机场,m条边,对于每个机场可能需要等候室也可能不需要;如果输入2,代表路线连接的两个机场都需要建立,输入1,代表路线连接的其中一个机场建立(必须),输入0代表路线连接的两个机场都不可以建立,问你最少建立几个等候室;

思路: 染色法+DFS
是否建立等候室看成是否染色,机场就是图论的点
对于输入2时,输入的两点颜色标记为’2’即可
对于输入1时,输入的两点颜色标记为‘1’即可
对于上面两种情况,染色时如果出现矛盾情况,就直接“impossible”
对于输入1时,输入的两点暂时先标记为‘0’,代表颜色待定
并只对输入1时,输入的两点建边(为了方便下面dfs搜索)
然后此时建立完边后,一定会出现多个封闭的图,里面的点会有很多情况。
此时,将图进行分类,分为两类,分成图中点全是0,和图中有点不是0的
先对存在不是0点的图搜索一遍,因为建立的边都是因为输入1,而建立的,所以当知道其中一个点的颜色时,这个封闭图其他所有点的颜色都可以推导出来.例如:一个点是2,那与这个点相连的点都应该是1,因为连接的点只能有1个等候室。一个点是1,那与这个点相连的点都应该是2,原理如上。
在这个搜索+染色的过程中,如果发现某个点的颜色与相连的点颜色一致,那么说明矛盾,直接退出
再对全是0的图进行染色,此时就需要对这个图的某个点设置一个颜色,要么设置为1,要么设置为2,对这两种情况都搜索一遍+染色,最终取染成1的最小值(实际不用代码搜索两遍,染色成2和染色成1最终形成的2

和1的数量一定是相对的,知道一个就知道另一个情况的,所以直接取最小值即可)

具体见代码注释(思路转自)

#include <bits/stdc++.h>
using namespace std;
#define pi acos(-1)
#define xx first
#define yy second
#define endl "\n"
#define lowbit(x) x & (-x)
#define int long long
#define ull unsigned long long
#define pb push_back
typedef pair<int, int> PII;
typedef pair<double, double> PDD;
#define max(a, b) (((a) > (b)) ? (a) : (b))
#define min(a, b) (((a) < (b)) ? (a) : (b))
#define Ysanqian ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);
const int N = 1e6 + 10, M = 1010, inf = 0x3f3f3f3f, mod = 18446, P = 13331;
const double eps = 1e-8;
int n, m;
int color[N];
bool st[N];
vector<int> g[N];
int res, a, b;
bool dfs(int u, int Color, int tag)
{
	color[u] = Color;
	st[u] = 1;
	if (tag) // tag来判断是否是全0染色
	{
		if (Color == 1) // 记录我们染色的数量一个是1,一个是2,取min即可
			a++;
		else
			b++;
	}
	else // 如果是1,2色的扩展染色,我们只会去染没有染过的点,且由于二号色一定有机场,所以扩展染色时染成一号色的就不需要休息室了,
	// 而二号染色是一定链接者没有休息室的1号染色是一定要休息室的,对答案贡献加一
	{
		if (Color == 2)
			res++;
	}
	for (auto ed : g[u]) // 经典的二分染色的版子
	{
		if (color[ed])
		{
			if (color[ed] == Color)
				return false;
		}
		else if (!dfs(ed, 3 - Color, tag))
			return false;
	}
	return true;
}
void solve()
{
	bool flag = 1;
	cin >> n >> m;
	for (int i = 1; i <= m; i++)
	{
		int a, b, c;
		cin >> a >> b >> c;
		if (c == 2) // 都有休息室的染成2
		{
			if (color[a] == 1 || color[b] == 1)
				flag = 0;
			color[a] = color[b] = 2;
		}
		else if (c == 0) // 都没休息室的染成1
		{
			if (color[a] == 2 || color[b] == 2)
				flag = 0;
			color[a] = color[b] = 1;
		}
		else // 待定的建边
		{
			g[a].pb(b);
			g[b].pb(a);
		}
	}
	// 对有1或-1点的图,进行扩散染色
	for (int i = 1; i <= n; i++)
	{
		if (!st[i] && color[i]) // 没有访问过,求以及有色的
		{
			if (!dfs(i, color[i], 0))
				flag = 0;
		}
	}
	// 对于只有0的图,选一个点设置为1,再扩散染色
	for (int i = 1; i <= n; i++)
	{
		if (st[i])
			continue;
		a = 0, b = 0; // 可能有多个全0的联通分量,a,b,代表2种颜色,我们用最小值即可
		if (!dfs(i, 1, 1))
			flag = 0;
		res += min(a, b);
	}
	if (flag)
		cout << res << endl;
	else
		cout << "impossible" << endl;
}
signed main()
{
	Ysanqian;
	int T;
	T = 1;
	// cin >> T;
	while (T--)
		solve();
	return 0;
}

G - Outing

题意:一辆车容量为V,现有n个人,每个人都有一个对应a[i],表示只有a[i]上车 i才肯上车,问这辆车最多承载多少人。

由于a[i]和i的对应关系 可能存在  或者 环 或者 环连着链->树 三种情况,我们可以得知如果存在环 这个环必然是所在树的树根

为什么环一定是树根呢?

根据本题题意,每一个点最多有一个依赖点,所以每一个点的出度为1,如果联通分量中出现了环,那么环中每一个点的出边必然指向环内的一个点,不会指向环外的点,所以这个环在缩点后出度必然为0。如果该连通分量中存在两个环,那么这两个环因为都没有出度,所以不可能出现由一个环出发走到另一个环的可能(除非相交,但相交就相当于一个环)因为所有结点的出度为1,该联通分量的的任意一个结点都有只有一个路径可走,假设路径A能走到第一个环,B能走到第二个环,路径A和路径B之间必然不连通,所以这两个环必然位于两个联通分量之中。

tarjan缩点之后,我们根据缩点后的图用并查集来合并同一联通分量的点

因此对图求强连通分量,每个树所采纳的人数的【下限——树根的那个强连通分量,上限——全部】,然后对这些树做分组背包求容量V以内的最大人数。

#include <bits/stdc++.h>
using namespace std;
#define pi acos(-1)
#define xx first
#define yy second
#define endl "\n"
#define lowbit(x) x & (-x)
#define int long long
#define ull unsigned long long
#define pb push_back
typedef pair<int, int> PII;
typedef pair<double, double> PDD;
// #define max(a, b) (((a) > (b)) ? (a) : (b))
// #define min(a, b) (((a) < (b)) ? (a) : (b))
#define Ysanqian ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);
const int N = 1e6 + 10, M = 1010, inf = 0x3f3f3f3f, mod = 18446, P = 13331;
const double eps = 1e-8;
int f[N], p[N], a[N], stk[N], id[N], siz[N];
int dfn[N], low[N];
int timestamp, top, scc_cnt;
bool is_stk[N];
vector<int> g[N];
map<int, PII> mp;
void tarjan(int u) // 缩点,将强连通分量缩成一个点
{
    dfn[u] = low[u] = ++timestamp;
    stk[++top] = u;
    is_stk[u] = true;
    for (auto j : g[u])
    {
        if (!dfn[j])
        {
            tarjan(j);
            low[u] = min(low[u], low[j]);
        }
        else if (is_stk[j])
            low[u] = min(low[u], low[j]);
    }
    if (dfn[u] == low[u])
    {
        int y;
        ++scc_cnt;
        do
        {
            y = stk[top--];
            is_stk[y] = false;
            id[y] = scc_cnt; // 记录该点位于那个强联通分量
            siz[scc_cnt]++;  // 维护强联通分量的大小
        } while (y != u);
    }
}
int find(int x)
{
    if (x != p[x])
        p[x] = find(p[x]);
    return p[x];
}
void merge(int x, int y)
{
    int a = find(x), b = find(y);
    if (a == b)
        return;
    p[b] = a;
}
void solve()
{
    int n, m;
    cin >> n >> m;
    for (int i = 1; i <= n; i++)
    {
        cin >> a[i];
        g[i].pb(a[i]); // 建边
    }
    for (int i = 1; i <= n; i++)
    {
        if (!dfn[i]) // 把环进行缩点
            tarjan(i);
    }
    for (int i = 1; i <= n; i++)
        p[i] = i;
    for (int i = 1; i <= n; i++)
    {
        int x = id[i], y = id[a[i]]; // 是否在一个联通分量里
        merge(x, y);                 // 合并联通分量,就相当于组分组背包,
        /*例如将以下合成一个联通分量
              o       o
             /      /   \   (这是个环,那么正如我们所说下限—就是树根的那个强连通分量,上限就是全部的和,这样就形成一个分组背包)
            o      o-----o
           /      /
          o      o

       */
    }
    for (int i = 1; i <= scc_cnt; i++) // 统计该组的大小和最小选择人数
    {
        mp[find(i)].xx = max(mp[find(i)].xx, siz[i]); // mp的int存的强连通分量的编号,pair第一维表示该联通分量的最大体积,第二维表示最大体积
        mp[find(i)].yy += siz[i];
    }
    for (auto ed : mp)
    {
        for (int j = m; j >= 0; j--)
        {
            for (int k = ed.yy.xx; k <= ed.yy.yy; k++)
            {
                if (j >= k)
                    f[j] = max(f[j], f[j - k] + k);
            }
        }
    }
    cout << f[m] << endl;
}
signed main()
{
    Ysanqian;
    int T;
    T = 1;
    // cin >> T;
    while (T--)
        solve();
    return 0;
}


http://www.niftyadmin.cn/n/4930275.html

相关文章

Nim游戏:取石头

&#xff08;一&#xff09;一堆取石头 背景&#xff1a; 在博弈论中&#xff0c;有一种称为Nim游戏的经典问题&#xff0c;它涉及到取石子的问题&#xff0c;其中有许多变种。Nim游戏是一种零和博弈&#xff0c;即两名玩家交替行动&#xff0c;每次只能从一堆物品中取走一定数…

leetcode 96. 不同的二叉搜索树

2023.8.10 本题可以先在草稿上画出各种不同的二叉搜索树&#xff0c;从中寻找规律&#xff0c;找出递推关系&#xff1a; 例如&#xff1a;当n3时&#xff0c;分三种情况&#xff1a; ①1作为根节点&#xff0c;此时左子树有0个节点&#xff0c;右子树有2两个节点&#xff0c;…

电脑麦克风没声音?

这3招就可以解决&#xff01; 在我们使用电脑录制视频时&#xff0c;有时会遇到一个令人头疼的问题&#xff1a;麦克风没有声音。那么&#xff0c;为什么会出现这种情况呢&#xff1f;更重要的是&#xff0c;我们应该如何解决这个问题呢&#xff1f;本文将介绍3种方法&#xf…

“深入探究JVM:揭秘Java虚拟机的工作原理“

标题&#xff1a;深入探究JVM&#xff1a;揭秘Java虚拟机的工作原理 摘要&#xff1a;本文将深入探究Java虚拟机&#xff08;JVM&#xff09;的工作原理&#xff0c;包括JVM的架构、内存管理、垃圾回收机制以及即时编译等关键概念。通过详细解释这些概念&#xff0c;读者将能够…

批量翻译多个文件夹,让你的文件管理更智能高效!

大家好&#xff01;对于经常需要管理大量文件夹的你来说&#xff0c;每次手动逐个改名实在是一项繁琐且易出错的工作。现在&#xff0c;我们为你带来一款强大的文件夹批量改名工具&#xff0c;让你能够轻松实现多个文件夹的批量翻译&#xff0c;让你的文件管理更智能高效 第一…

安卓如何快速定位native内存泄露。

步骤1&#xff09;cat /proc/pid/status,观察下面俩个指标 RssAnon: 5300 kB //一直增大说明匿名映射的内存增大&#xff0c;malloc本质就是调用匿名映射分 配内存 RssFile: 26884 kB //文件句柄泄露&#…

谷歌云 | BigQuery 现在支持用于查询开放表格式的清单文件

Cloud Ace 是谷歌云全球战略合作伙伴&#xff0c;拥有 300 多名工程师&#xff0c;也是谷歌最高级别合作伙伴&#xff0c;多次获得 Google Cloud 合作伙伴奖。作为谷歌托管服务商&#xff0c;我们提供谷歌云、谷歌地图、谷歌办公套件、谷歌云认证培训服务。 开放表格式依赖嵌…

跨境多语言商城源码搭建--定制代码+源码开源

搭建一个跨境多语言商城需要以下步骤&#xff1a; 1. 确定需求&#xff1a;首先&#xff0c;需要明确商城的功能和需求&#xff0c;比如支持哪些语言、支持哪些支付方式、支持哪些货币等。根据需求来决定使用的开发语言和技术栈。 2. 寻找源码&#xff1a;可以在互联网上搜索…