题目大意:

有 \(n\) 名员工,一共有 \(m\) 种语言,每名员工都会其中 \(k_i\) 种语言(\(m \ge \boldsymbol{k_i \ge 0}\)),现规定两名员工可以交流的条件如下:

  1. 两名员工会一种及以上共同的语言。
  2. 有一名员工可以当这两名员工的翻译(即有一名员工会这两名员工会的语言中的各一种)。

每位员工所需要学习的语言数的和为多少。

题目分析:

我们可以对能够交流的员工建图,然后不难发现以下性质:

  1. 若 \(A\) 可以和 \(B\) 交流,则 \(B\) 也可以和 \(A\) 交流(即无向性)。
  2. 若 \(A\) 可以和 \(B\) 交流,\(B\) 可以和 \(C\) 交流,则 \(A\) 也可以和 \(C\) 交流(即传递性)。

所以,此时的问题就变成了给定一张无向图,问加几条边能够使其成为一张连通图。

可能有更好的做法,但我的做法如下:

  1. 若该点表的员工一种语言也不会,则 \(cnt\) 加一。
  2. 若该点没有被访问过,且该点表示的员工至少会一种语言,则 \(cnt\) 加一,然后以该点为起点进行 \(\operatorname{dfs}\)。
  3. 若该点被访问过,则跳过。

此时答案是 \(cnt - 1\)。

注意:若没有员工会语言,则此时的答案是 \(cnt\) 而不是 \(cnt-1\),因为每一名员工都要学一种语言才能交流。

综上,时间复杂度 \(O(n \times m)\),主要取决于建图部分的时间复杂度。

代码实现:

#include <bits/stdc++.h>
using namespace std;
#define int long long
#define TIME_LIMIT (time_t)1e3
#define dbg(x) cerr<<#x<<": "<<x<<endl;
#define MAX_SIZE (int)2.1e4
vector <int> hashtable[MAX_SIZE];
unordered_map <int, bool> edges;
int head[MAX_SIZE];
int ver[MAX_SIZE];
int Next[MAX_SIZE];
int tot = 0;
void add(int u,int v){
ver[++tot] = v;
Next[tot] = head[u];
head[u] = tot;
}
bitset<MAX_SIZE>vis;
bitset<MAX_SIZE>pass;
void dfs(int step, int u, int fa){
assert(step<=114);
vis[u] = 1;
for(int i=head[u];i;i=Next[i]){
int v = ver[i];
if(v==fa)
continue;
if(v==u)
continue;
if(!vis[v])
dfs(step+1,v,u);
}
}
signed main() {
ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
#ifdef LOCAL
#undef cout
#undef cin
ifstream cin("in.in");
ofstream cout("out.out");
assert(cin.is_open());
assert(cout.is_open());
freopen("in.in","r",stdin);
freopen("out.out","w",stdin);
time_t cs = clock();
#endif
//========================================
int n,m;
cin>>n>>m;
for(int i=1;i<=n;i++){
int k;
cin>>k;
for(int j=1;j<=k;j++){
int lang;
cin>>lang;
hashtable[lang].push_back(i);
}
}
for(int i=1;i<=m;i++){
auto &vec = hashtable[i];
if(vec.empty())
continue;
int u = vec[vec.size()-1];
pass[u] = 1;
vec.pop_back();
for(auto v:vec){
if(!edges[min(u,v)*100+max(v,u)]){
add(u,v);
add(v,u);
pass[v] = 1;
edges[min(u,v)*100+max(v,u)] = 1;
}
}
}
int cnt = 0;
for(int i=1;i<=n;i++) {
if(!vis[i]&&pass[i]){
dfs(1,i,i);
++cnt;
}
}
int nothing = 0;
for(int i=1;i<=n;i++)
if(!pass[i])
++nothing;
if(nothing==n)
cout<<nothing;
else
cout<<nothing+cnt-1;
//========================================
#ifdef LOCAL
cin.close();
cout.close();
time_t ce = clock();
cerr<< "Used Time: " << ce-cs << " ms."<<endl;
if(TIME_LIMIT<ce-cs)
cerr<< "Warning!! Time exceeded limit!!"<<endl;
#endif
return 0;
}

题解 CF277A的更多相关文章

  1. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

  2. noip2016十连测题解

    以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...

  3. BZOJ-2561-最小生成树 题解(最小割)

    2561: 最小生成树(题解) Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1628  Solved: 786 传送门:http://www.lyd ...

  4. Codeforces Round #353 (Div. 2) ABCDE 题解 python

    Problems     # Name     A Infinite Sequence standard input/output 1 s, 256 MB    x3509 B Restoring P ...

  5. 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解

    题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...

  6. 2016ACM青岛区域赛题解

    A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Jav ...

  7. poj1399 hoj1037 Direct Visibility 题解 (宽搜)

    http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...

  8. 网络流n题 题解

    学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...

  9. CF100965C题解..

    求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...

  10. JSOI2016R3 瞎BB题解

    题意请看absi大爷的blog http://absi2011.is-programmer.com/posts/200920.html http://absi2011.is-programmer.co ...

随机推荐

  1. 图技术在 LLM 下的应用:知识图谱驱动的大语言模型 Llama Index

    LLM 如火如荼地发展了大半年,各类大模型和相关框架也逐步成型,可被大家应用到业务实际中.在这个过程中,我们可能会遇到一类问题是:现有的哪些数据,如何更好地与 LLM 对接上.像是大家都在用的知识图谱 ...

  2. 交换机:ToR、EoR

    参考链接: 交换机:ToR.EoR ToR:(Top of Rack)接入方式就是在服务器机柜的最上面安装接入交换机. EoR:(End of Row)接入交换机集中安装在一列机柜端部的机柜内,通过水 ...

  3. JVM篇(一) 什么是JVM,它有什么用

    一.JVM的组成 1. JVM由那些部分组成,运行流程是什么? 从图中可以看出 JVM 的主要组成部分 ClassLoader(类加载器) Runtime Data Area(运行时数据区,内存分区) ...

  4. 如何在CMD窗口运行python文件

    进入文件所在的路径输入: python  文件名

  5. 2.0 Python 数据结构与类型

    数据类型是编程语言中的一个重要概念,它定义了数据的类型和提供了特定的操作和方法.在 python 中,数据类型的作用是将不同类型的数据进行分类和定义,例如数字.字符串.列表.元组.集合.字典等.这些数 ...

  6. 【Unity3D】激光雷达特效

    1 由深度纹理重构世界坐标 ​ 屏幕深度和法线纹理简介中对深度和法线纹理的来源.使用及推导过程进行了讲解,本文将介绍使用深度纹理重构世界坐标的方法,并使用重构后的世界坐标模拟激光雷达特效. ​ 本文完 ...

  7. MIT 6.828 Lab实验记录 —— lab1 Booting PC

    实验参考信息 MIT 6.828 lab1 讲义地址 MIT 6.828 课程 Schedule MIT 6.828 lab 环境搭建参考 MIT 6.828 lab 工具guide Brennan' ...

  8. 不关闭Tamper Protection(篡改保护)下强制卸载Windows Defender和安全中心所有组件

    个人博客: xzajyjs.cn 背景介绍 由于微软不再更新arm版本的win10系统,因此只能通过安装insider preview的镜像来使用.而能找到的win10 on arm最新版镜像在安装之 ...

  9. ps aux 命令使用

    转载请注明出处: ps aux​​命令是一个常用的Linux/Unix系统命令,它用于查看正在运行的进程信息.下面是该命令使用的参数介绍.命令响应结果参数介绍以及一些使用实例的总结: 命令使用参数介绍 ...

  10. 浅入深出的微前端MicroApp

    前言: 本文是由最近做的一个项目有感而发,因为之前做了一些技术栈的统一,为了用ant Design的pro-table,PC统一使用react,但是我们有一些老的项目是vue的,本次新页面较多,老页面 ...