由于到达关系具有传递性,可以考虑不断将若干个可以相互到达的点缩点,并且当两个点只能单向到达时,能到达另一个点的点一定不是最小值

由此,我们来考虑dfs,即不断从一个节点开始,遍历其可以到达的点,当发现了环即将这些点合并(启发式合并),当发现了已经完全搜过(不在栈中)的点即一定不能作为最小值

在实现上,有一些困难,注意以下细节:

1.对每一个点还要维护一个钥匙集合,也参与启发式合并

2.启发式合并时可能有新的边可以到达,在加入时判定即可

3.由于要弹出元素,且不像强连通分量一样保证必然弹出,因此建议手动模拟栈来实现

4.在搜过一条边后将其弹出边集,以避免合并后被其他点重复搜索

5.判定是否遍历到只能单向到达的点,可以在搜索结束后再判定

  1 #include<bits/stdc++.h>
2 #include"keys.h"
3 using namespace std;
4 #define N 300005
5 #define vi vector<int>
6 #define fi first
7 #define se second
8 stack<int>st;
9 set<int>key[N];
10 set<int>::iterator it1;
11 vector<int>ans,G0[N];
12 set<pair<int,int> >G1[N];
13 set<pair<int,int> >::iterator it2;
14 int n,m,mn,dfn[N],vis[N],f[N],sz[N];
15 int find(int k){
16 if (k==f[k])return k;
17 return f[k]=find(f[k]);
18 }
19 int get_sum(int k){
20 return key[k].size()+G0[k].size()+G1[k].size();
21 }
22 void add(int x,int y,int z){
23 x=find(x);
24 if (key[x].find(z)!=key[x].end())G0[x].push_back(y);
25 else G1[x].insert(make_pair(z,y));
26 }
27 void merge(int x,int y){
28 x=find(x),y=find(y);
29 if (x==y)return;
30 if (get_sum(x)<get_sum(y))swap(x,y);
31 f[y]=x;
32 sz[x]+=sz[y];
33 for(it1=key[y].begin();it1!=key[y].end();it1++){
34 int k=(*it1);
35 key[x].insert(k);
36 while (1){
37 it2=G1[x].lower_bound(make_pair(k,0));
38 if ((it2==G1[x].end())||((*it2).fi!=k))break;
39 G0[x].push_back((*it2).se);
40 G1[x].erase(it2);
41 }
42 }
43 key[y].clear();
44 for(int i=0;i<G0[y].size();i++)G0[x].push_back(G0[y][i]);
45 G0[y].clear();
46 for(it2=G1[y].begin();it2!=G1[y].end();it2++)add(x,(*it2).se,(*it2).fi);
47 G1[y].clear();
48 }
49 void dfs(int k){
50 st.push(k);
51 while (!st.empty()){
52 k=st.top();
53 dfn[k]=1;
54 if (G0[k].empty()){
55 vis[k]=1;
56 st.pop();
57 continue;
58 }
59 int x=find(G0[k].back());
60 G0[k].pop_back();
61 if (!dfn[x]){
62 st.push(x);
63 continue;
64 }
65 if (vis[x])continue;
66 while (st.top()!=x){
67 merge(st.top(),k);
68 st.pop();
69 }
70 merge(st.top(),k);
71 st.pop();
72 st.push(find(k));
73 }
74 }
75 void check(int x,int y,int z){
76 x=find(x),y=find(y);
77 if (x!=y){
78 if (key[x].find(z)!=key[x].end())vis[x]=1;
79 if (key[y].find(z)!=key[y].end())vis[y]=1;
80 }
81 }
82 vi find_reachable(vi r,vi u,vi v,vi c){
83 n=r.size(),m=u.size();
84 for(int i=1;i<=n;i++){
85 f[i]=i,sz[i]=1;
86 key[i].insert(r[i-1]);
87 }
88 for(int i=0;i<m;i++){
89 u[i]++,v[i]++;
90 add(u[i],v[i],c[i]);
91 add(v[i],u[i],c[i]);
92 }
93 for(int i=1;i<=n;i++)
94 if (!dfn[i])dfs(i);
95 memset(vis,0,sizeof(vis));
96 for(int i=0;i<m;i++)check(u[i],v[i],c[i]);
97 mn=0x3f3f3f3f;
98 for(int i=1;i<=n;i++)
99 if (!vis[find(i)])mn=min(mn,sz[find(i)]);
100 for(int i=1;i<=n;i++)
101 if ((!vis[find(i)])&&(sz[find(i)]==mn))ans.push_back(1);
102 else ans.push_back(0);
103 return ans;
104 }

[loj3524]钥匙的更多相关文章

  1. DAY6 使用ping钥匙临时开启SSH:22端口,实现远程安全SSH登录管理就这么简单

    设置防火墙策略时,关于SSH:22访问权限,我们常常会设置服务器只接受某个固定IP(如公司IP)访问,但是当我们出差或在家情况需要登录服务器怎么办呢? 常用两种解决方案:1.通过VPN操作登录主机: ...

  2. wifi万能钥匙自媒体平台开放注册(付注册流程)

    12月13日,有网友爆料,wifi万能钥匙自媒体开放注册,看来自媒体还没有达到饱和阶段,也印证了自媒体时代才刚刚到来.现在这个自媒体的时代,几乎大多互联网企业都开通了自己的自媒体,比较知名的像今日头条 ...

  3. 使用ping钥匙临时开启SSH:22端口,实现远程安全SSH登录管理就这么简单

    设置防火墙策略时,关于SSH:22访问权限,我们常常会设置服务器只接受某个固定IP(如公司IP)访问,但是当我们出差或在家情况需要登录服务器怎么办呢? 常用两种解决方案:1.通过VPN操作登录主机: ...

  4. 使用HackRF+GNU Radio 破解吉普车钥匙信号

    引文 我最近对软件定义的无线电技术(SDR)产生了浓厚的兴趣,而我对其中一款流行的SDR平台(HackRF)也产生了兴趣,而其频率接收的范围也在1MHz ~6GHz之间(范围较广).而这里也需要提及一 ...

  5. HDU-1438 钥匙计数之一

    http://acm.hdu.edu.cn/showproblem.php?pid=1438                                钥匙计数之一 Time Limit: 200 ...

  6. 手机NFC通信的安全车钥匙

    SmartKeys for Cyber-Cars:Secure Smartphone-based NFC-enabled Car Immobicizer 手机NFC通信的安全车钥匙 1概述 如今,智能 ...

  7. 钥匙计数之一 - HDU 1438(状态压缩打表)

    分析:首先想到每个钥匙的结尾有4种状态,不过题目还需要判断有三种不同的钥匙深度,所以每种深度结尾后有2^4种状态,0000->1111,不过题目还需需要有相邻的钥匙深度大于等于3,所以需要两种不 ...

  8. [转]使用ping钥匙临时开启SSH:22端口,实现远程安全SSH登录管理就这么简单

    原文链接:使用ping钥匙临时开启SSH:22端口,实现远程安全SSH登录管理就这么简单 这个留待后面玩一下,还是有安全隐患,非核心业务 临时用一下可以. 设置防火墙策略时,关于SSH:22访问权限, ...

  9. Wi-Fi万能钥匙:说是破解,其实有危险(转)

    Wi-Fi 万能钥匙如此危险,怎样做才能让这种可能严重侵害公众利益的 app 在中国消失? 这个“钥匙”为什么能够破解 Wi-Fi?它真的是“破解” Wi-Fi 吗?两年前我就有这个疑问了,原谅我对一 ...

随机推荐

  1. oracle查看和修改session和最大连接数

    第一步,在cmd命令行,输入sqlplus 第二步,根据提示输入用户名与密码 1. 查看processes和sessions参数 SQL> show parameter processes NA ...

  2. 网络基础--简单理解什么是DNS? TCP? UDP? Http? Socket?

    什么是IP 协议?  协议就是为了实现网络通信而创建的一系列规范.  通常我们的网络模型从上到下共分为4层: 应用层, 传输层, 网络层 和数据链路层. IP协议属于网络层协议,它精确定义了网络通信中 ...

  3. js 改变this指向的三种方法 bind call apply

    先了解下bind call apply 的注意点 bind 需要手动调用 第一个参数 this 要指向的对象,后面是 散列的参数 call 不需要手动调用 第一个参数 this 要指向的对象,后面是 ...

  4. v72.01 鸿蒙内核源码分析(Shell解析) | 应用窥伺内核的窗口 | 百篇博客分析OpenHarmony源码

    子曰:"苟正其身矣,于从政乎何有?不能正其身,如正人何?" <论语>:子路篇 百篇博客系列篇.本篇为: v72.xx 鸿蒙内核源码分析(Shell解析篇) | 应用窥视 ...

  5. C#开发BIMFACE系列52 CS客户端集成BIMFACE应用的技术方案

    BIMFACE二次开发系列目录     [已更新最新开发文章,点击查看详细] 在我的博客<C#开发BIMFACE系列49 Web网页集成BIMFACE应用的技术方案>.<C#开发BI ...

  6. Solon 框架如何方便获取每个请求的响应时间?

    经常会有同学问 Solon 怎样才能获取每个请求的响应时间?要求是不需要给每个函数加注解.故此,整理了一下. 不给每个函数加注解,主要有两种方式可以获取请求响应时间: 方式1:基于全局过滤器 Solo ...

  7. 4.19——数组双指针——26. 删除有序数组中的重复项 & 27. 删除有序数组中的重复项II & 80. 删除有序数组中的重复项 II

    第一次做到数组双指针的题目是80: 因为python的List是可以用以下代码来删除元素的: del List[index] 所以当时的我直接用了暴力删除第三个重复元素的做法,大概代码如下: n = ...

  8. 封装一个的toast弹出框(vue项目)

    逆风的方向,更适合飞翔 实现效果 实现步骤 先写出一个toast组件 // Toast.vue <template> <div id="toast" :class ...

  9. python png图片生成gif

    有时候写代码就是这样别人把代码写好你在后面加一个句号就行了 我很懒不想写成函数,你自己来吧.有注释就不错了 这个依赖一个图像处理库pillow,轮子就是轮他不是车 import imageio imp ...

  10. 「总结」$dp1$

    大概就是做点题. 先列一下要做的题目列表,从\(UOJ\)上找的. 129寿司晚宴 348州区划分 370滑稽树上滑稽果 457数树 22外星人 37主旋律 300吉夫特 196线段树 311积劳成疾 ...