C - BLG POJ - 1417 种类并查集加dp(背包)
思路:刚看这道题感觉什么都不清楚,人物之间的关系一点也看不出来,都不知道怎么写,连并查集都没看出来,但是你可以仔细分析一下,当输入字符串为“yes”的时候,我们设输入的值为x和y,当x为天使是则由题可知y也为天使;当x为魔鬼的时候,则y也为魔鬼,所以输入“yes”的时候就相当于说他们是同类。
当输入字符串为“no”的时候,如果x为天使,则y为魔鬼;x为魔鬼的时候,y就是天使,所以当输入字符串为“no”的时候他们为异类。。这不就是种类并查集的
0 (同类) 1(异类)并查集嘛!
再接着想,通过并查集我们可以把它们分成两类,一种是和他们的根节点同类,一种不是同类。。这一点要注意了,所有的他们并不是一定是一个父节点,假如
(1 2)(3 4) 那他们就不在一棵树上。。
用完并查集我们就要用一个数组来记录他们根节点的数目,而且还要记录每一个根节点与其同类和异类的两种情况的数目,然后我们只要用这些数可以组合出来
天使或魔鬼的数目就可以了,但是要保证只有一种方法可以,如果有好几种方法,那就无法确定。。还要注意的是当去组成天使或魔鬼的数目的时候,一个根节点只能选根节点的同类或异类来往上加,不能都选。。。
上一点是不是有点像01背包,用一些数来组成一个数,看有几种方法
最后就是输出路径(路径方法我之前写过,你也可以在网上搜一下)
接着就上代码吧!
//https://www.cnblogs.com/geloutingyu/p/6142473.html 我是看这个网址才懂得,可以看一下
#include<stdio.h>
#include<string.h>
#include<iostream>
#include<algorithm>
using namespace std;
int v[],w[],n,a,b,gg[],jj[][],tag[][],dp[][],cc[][];
struct shudui
{
int start,last;
char q;
}m[];
int finds(int x)
{
if(x!=v[x])
{
int y=finds(v[x]);
w[x]=(w[x]+w[v[x]])%;
v[x]=y;
return y;
}
return x;
}
void join(int x,int y,int z)
{
int fx=finds(x);
int fy=finds(y);
if(fx==fy)
{
return;
}
else
{
v[fx]=fy;
w[fx]=(-w[x]+z-+w[y])%;
}
}
int main()
{
while(~scanf("%d%d%d",&n,&a,&b))
{
memset(w,,sizeof(w));
if(n+a+b==) break;
for(int i=;i<=a+b;++i)
{
v[i]=i;
}
while(n--)
{
char q[];
int x,y;
scanf("%d%d%s",&x,&y,q);
if(x==y)
continue;
if(strcmp("no",q)==)
{
join(x,y,);
}
else join(x,y,);
}
//int w1[1005],w2[1005];
memset(gg,,sizeof(gg)); //存根节点
memset(jj,,sizeof(jj));
memset(tag,,sizeof(tag)); //来记录根节点同来和异类的数目
memset(dp,,sizeof(dp)); //背包dp,看看有几种方法
memset(cc,,sizeof(cc)); //标记,做最后输出
int cnt=;
for(int i=;i<=a+b;++i) //统计集合个数
{
if(finds(i)==i)
{
gg[i]=++cnt;
}
}
for(int i=;i<=a+b;++i)
{
tag[gg[finds(i)]][w[i]]++;
}
dp[][]=;
for(int i=;i<=cnt;++i)
{
for(int j=;j<=a;++j)//dp[i][j]表示到第i个集合,人数达到j的方法数
{
if(j-tag[i][]>= && dp[i-][j-tag[i][]])
{
dp[i][j]+=dp[i-][j-tag[i][]];
jj[i][j]=tag[i][]; //jj是记录路径的
}
if(j-tag[i][]>= && dp[i-][j-tag[i][]])
{
dp[i][j]+=dp[i-][j-tag[i][]];
jj[i][j]=tag[i][];
}
}
}
if(dp[cnt][a]!=)
{
printf("no\n");
}
else
{
for(int i=cnt,j=a;j> && i>; i--)
{
if(jj[i][j]==tag[i][])
cc[i][]=;
else cc[i][]=;
j-=jj[i][j];
}
for(int i=;i<=a+b;++i)
{
if(cc[gg[finds(i)]][w[i]]) printf("%d\n",i);
}
printf("end\n");
}
}
return ;
}
C - BLG POJ - 1417 种类并查集加dp(背包)的更多相关文章
- Poj(1182),种类并查集
题目链接:http://poj.org/problem?id=1182 再次熟练种类并查集,又积累点经验,和技巧,rank 0 2 1 先计算father[x] ,再更新rank[x]; #inclu ...
- Poj(1703),种类并查集
题目链接:http://poj.org/problem?id=1703 已经不是第一次接触种类并查集了,直到今天才搞懂. 感谢红黑联盟,感谢杰哥!!! 每个节点只要关系确定,不管是不是同一个集合里面, ...
- POJ - 2492 种类并查集
思路:保存每个点与其父节点的关系,注意合并和路径压缩即可. AC代码 #include <cstdio> #include <cmath> #include <cctyp ...
- poj 1182:食物链(种类并查集,食物链问题)
食物链 Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 44168 Accepted: 12878 Description ...
- POJ 1733 Parity game(种类并查集)
http://poj.org/problem?id=1733 题意: 给出一个01串,有多次询问,每次回答[l,r]这个区间内1的个数的奇偶性,但是其中有一些回答是错误的,问到第几个回答时与前面的回答 ...
- 种类并查集,Poj(1703)
题目链接:http://poj.org/problem?id=1703 第一次做种类并查集,有的地方还不是很清楚,想了一上午,有点明白了,这里记录一下. 这里我参考的红黑联盟的题解. 关键:种类并查集 ...
- poj 1182 食物链(种类并查集 ‘初心者’)
题目链接:http://poj.org/problem?id=1182 借着这题可以好好理解一下种类并查集,这题比较简单但挺经典的. 题意就不解释了,中问题. 关于种类并查集结局方法也是挺多的 1扩增 ...
- 【POJ】2492 A bug's life ——种类并查集
A Bug's Life Time Limit: 10000MS Memory Limit: 65536K Total Submissions: 28211 Accepted: 9177 De ...
- poj 2492 A Bug's Life 二分图染色 || 种类并查集
题目链接 题意 有一种\(bug\),所有的交往只在异性间发生.现给出所有的交往列表,问是否有可疑的\(bug\)(进行同性交往). 思路 法一:种类并查集 参考:https://www.2cto.c ...
随机推荐
- redis session 共享 测试案列
下载 spring redis session demo 2.分别在不同的服务器上启动 3.nginx 安装 测试
- Manifest merger failed : Attribute application@icon value=(@mipmap/ic_launcher) from AndroidManifest
情况是这样子的,导入一个比较老的项目(两年前),它依赖于一个 Libraray,已经先导入了 library,现在导入项目的时候出了错 (1) Android Studio 目前提供将 SDK包成 . ...
- jsonp原理,封装,应用(vue项目)
jsonp原理 JSON是一种轻量级的数据传输格式. JSONP(JSON with Padding)是JSON的一种“使用模式”,可用于解决主流浏览器的跨域数据访问的问题.由于同源策略,一般来说位于 ...
- 【DeepLearning】深入理解dropout正则化
本文为转载,作者:Microstrong0305 来源:CSDN 原文:https://blog.csdn.net/program_developer/article/details/80737724 ...
- [NOI 2018] 归程
Description 传送门 Solution 65分做法 先求出每个点到\(1\)号点的最短路,记为\(d[i]\).然后按照海拔从大到小依次加边,并查集维护每个连通块中\(d[i]\)的最小值, ...
- golang中使用ETCD
安装 下载ETCD https://github.com/etcd-io/etcd/releases/ 安装 我下载的是window版,直接解压就可以了,解压后有以下目录 点击etcd.exe运行 然 ...
- java的集合:List、Set和Map
虚线是接口,实线是实现类: 集合能够解决的问题:集合可以丽杰为是一种更高级的数组,可以保存多条数据 本质:java官方开发人员基于java的一些基础内容(数组等等)创建了一些接口和类,然后使用这些接口 ...
- NFV-Based Scalable Guaranteed-Bandwidth Multicast Service for Software Defined ISP Networks
文章名称:NFV-Based Scalable Guaranteed-Bandwidth Multicast Service for Software Defined ISP Networks 发表时 ...
- pandas技巧两则——列内元素统计和列内元素排序
更新:后来忽然发现有个cumcount()函数,支持正排倒排,所以以下说的那些基本都没啥用了. 最近做比赛线上无甚进展,所以先小小地总结遇到的一些困难和解决的方法,以防之后忘记.毕竟总是忙着大步赶路的 ...
- STDIN_FILENO和stdin
title: STDIN_FILENO和stdin date: 2019/03/15 22:03:22 toc: true --- STDIN_FILENO和stdin https://blog.cs ...