POJ 3687 Labeling Balls(反向拓扑+贪心思想!!!非常棒的一道题)
|
Labeling Balls
Description Windy has N balls of distinct weights from 1 unit to N units. Now he tries to label them with 1 to N in such a way that:
Can you help windy to find a solution? Input The first line of input is the number of test case. The first line of each test case contains two integers, N (1 ≤ N ≤ 200) and M (0 ≤ M ≤ 40,000). The next M line each contain two integers a and b indicating the ball labeled with a must be lighter than the one labeled with b. (1 ≤ a, b ≤ N) There is a blank line before each test case. Output For each test case output on a single line the balls' weights from label 1 to label N. If several solutions exist, you should output the one with the smallest weight for label 1, then with the smallest weight for label 2, then with the smallest weight for label 3 and so on... If no solution exists, output -1 instead. Sample Input 5 4 0 4 1 Sample Output 1 2 3 4 Source POJ Founder Monthly Contest – 2008.08.31, windy7926778
|
题目意思:
给你n个球和m组关系,关系a b表示a比b轻,在符合这些要求的条件下,给他们贴标签
要求轻的球标签也要小,最后要你输出这些球的重量顺序
第一次的做法:
一开始的思路是用一个优先队列,从而做到从小到大取标签,
保证标签号小的在序列的尽可能前面,这样对于比较小的标签,
就可以得到比较小的重量了。
然后妥妥的wa
因为我输出的整个标签的序列而不是每个标签所对应的球的重量。
改完以后交了,还是错,想了好久,
看了别人的思路都说这道题要反向建图,
在从大往小的取,将大的数尽量放到答案的后面,
这样就能够保证小重量的球尽可能的放到了前面。
题目意思一开始就看错了两个地方
1.要求输出的不是标签顺序,而是重量的顺序
2.没有理解轻的球其标签也要小
正解:
分析;
给了你m组关系,a b表示标签为a比标签为b轻
我们可以先得到标签的顺序,然后再变化一下得到每个标签对应的重量
求标签的顺序:
要使得轻的球其标签小,也就是轻的放标签序列的前面
如果按照我一开始的做法,正向拓扑,然后每次选标签小的放前面
这样是不行的,因为即使该点最小,但是它后面连接的点未必是最小的啊
比如:
6->4->1
\
3->9->2 ->0
/
5->7->8
1,2,8都是指向0的,不好画
如果取入度为0且最小的话,那么是先取3,而我们的目的是让标号小的尽量靠前,即让1靠前,这与先取3是相违背的
所以我一开始的做法的确不行
因为题目是要满足轻的球标签小,所以正确做法:
每次让标签号大的尽量完后排,因为从后往前看,位置是越来越重要的!
贪心的思想
所以我们反向建边就是b比a重,每次选择标签号大的从最后面开始
因为越到前面,是越来越重要的,我们先从不重要的开始放,最后再放最重要的(同时最后放的也是在满足约束的条件下,标签号最小的)
样例:
5 3
1 4
4 2
3 5
反向建图:
2->4->1
5->3
每次选择标签号大的排(从后往前排)
那么拓扑得到的标签顺序就是:1 4 2 3 5
再求其重量排序:(重量为2的排在第3个,重量为3的排在第四个)
1 3 4 2 5
#include<stdio.h>
#include<iostream>
#include<math.h>
#include<string.h>
#include<set>
#include<map>
#include<list>
#include<queue>
#include<algorithm>
using namespace std;
typedef long long LL;
int mon1[]= {,,,,,,,,,,,,};
int mon2[]= {,,,,,,,,,,,,};
int dir[][]= {{,},{,-},{,},{-,}}; int getval()
{
int ret();
char c;
while((c=getchar())==' '||c=='\n'||c=='\r');
ret=c-'';
while((c=getchar())!=' '&&c!='\n'&&c!='\r')
ret=ret*+c-'';
return ret;
} #define max_v 205
int indgree[max_v];
vector<int> vv[max_v];
priority_queue<int,vector<int>,less<int> > q;
int n,m;
int a[max_v];
int tpsort()
{
for(int i=;i<=n;i++)
{
if(indgree[i]==)
q.push(i);
}
int temp;
int cnt=;
int k=n;
while(!q.empty())
{
temp=q.top();
q.pop();
a[temp]=k--;
cnt++;
for(int i=;i<vv[temp].size();i++)
{
indgree[vv[temp][i]]--;
if(indgree[vv[temp][i]]==)
q.push(vv[temp][i]);
}
}
if(cnt!=n)
return ;
else
return ;
}
int main()
{
int t;
int x,y;
scanf("%d",&t);
while(t--)
{
scanf("%d %d",&n,&m);
if(m==)
{
for(int i=;i<=n;i++)
{
if(i==)
printf("%d",i);
else
printf(" %d",i);
}
printf("\n");
continue;
}
memset(indgree,,sizeof(indgree));
for(int i=;i<=n;i++)
vv[i].clear();
while(!q.empty())
q.pop();
int flag1=;//环
for(int i=;i<=m;i++)
{
scanf("%d %d",&y,&x);
if(flag1==)
{
if(x==y)
{
flag1=;
continue;
}
if(count(vv[x].begin(),vv[x].end(),y)==)
{
vv[x].push_back(y);
indgree[y]++;
}
}
}
if(flag1==)
flag1=tpsort();
if(flag1)
{
printf("-1\n");
continue;
}
for(int i=;i<=n;i++)
{
if(i==)
printf("%d",a[i]);
else
printf(" %d",a[i]);
}
printf("\n");
}
return ;
} /*
题目意思:
给你n个球和m组关系,关系a b表示a比b轻,在符合这些要求的条件下,给他们贴标签
要求轻的球标签也要小,最后要你输出这些球的重量顺序 第一次的做法:
一开始的思路是用一个优先队列,从而做到从小到大取标签,
保证标签号小的在序列的尽可能前面,这样对于比较小的标签,
就可以得到比较小的重量了。
然后妥妥的wa
因为我输出的整个标签的序列而不是每个标签所对应的球的重量。
改完以后交了,还是错,想了好久,
看了别人的思路都说这道题要反向建图,
在从大往小的取,将大的数尽量放到答案的后面,
这样就能够保证小重量的球尽可能的放到了前面。 题目意思一开始就看错了两个地方
1.要求输出的不是标签顺序,而是重量的顺序
2.没有理解轻的球其标签也要小 正解:
分析;
给了你m组关系,a b表示标签为a比标签为b轻
我们可以先得到标签的顺序,然后再变化一下得到每个标签对应的重量 求标签的顺序:
要使得轻的球其标签小,也就是轻的放标签序列的前面
如果按照我一开始的做法,正向拓扑,然后每次选标签小的放前面
这样是不行的,因为即使该点最小,但是它后面连接的点未必是最小的啊
比如:
6->4->1
\
3->9->2 -0
/
5->7->8 1,2,8都是指向0的,不好画 如果取入度为0且最小的话,那么是先取3,而我们的目的是让标号小的尽量靠前,即让1靠前,这与先取3是相违背的
所以我一开始的做法的确不行 因为题目是要满足轻的球标签小,所以正确做法:
每次让标签号大的尽量完后排,因为从后往前看,位置是越来越重要的!
贪心的思想
所以我们反向建边就是b比a重,每次选择标签号大的从最后面开始
因为越到前面,是越来越重要的,我们先从不重要的开始放,最后再放最重要的(同时最后放的也是在满足约束的条件下,标签号最小的)
样例:
5 3
1 4
4 2
3 5 反向建图:
2->4->1
5->3 每次选择标签号大的排(从后往前排)
那么拓扑得到的标签顺序就是:1 4 2 3 5
再求其重量排序:(重量为2的排在第3个,重量为3的排在第四个)
1 3 4 2 5
*/
POJ 3687 Labeling Balls(反向拓扑+贪心思想!!!非常棒的一道题)的更多相关文章
- POJ 3687 Labeling Balls【拓扑排序 优先队列】
题意:给出n个人,m个轻重关系,求满足给出的轻重关系的并且满足编号小的尽量在前面的序列 因为输入的是a比b重,但是我们要找的是更轻的,所以需要逆向建图 逆向建图参看的这一篇http://blog.cs ...
- POJ 3687 Labeling Balls(拓扑排序)题解
Description Windy has N balls of distinct weights from 1 unit to N units. Now he tries to label them ...
- POJ - 3687 Labeling Balls (拓扑)
(点击此处查看原题) 题意 此处有n盏灯,编号为1~n,每盏灯的亮度都是唯一的,且在1~n范围之间,现已知m对灯之间的关系:a b ,说明灯a的亮度比灯b小,求出每盏灯的亮度,要求字典序最小(编号小的 ...
- [ACM] POJ 3687 Labeling Balls (拓扑排序,反向生成端)
Labeling Balls Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 10161 Accepted: 2810 D ...
- poj 3687 Labeling Balls【反向拓扑】
Labeling Balls Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 12246 Accepted: 3508 D ...
- poj——3687 Labeling Balls
Labeling Balls Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 14835 Accepted: 4346 D ...
- POJ 3687 Labeling Balls()
Labeling Balls Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 9641 Accepted: 2636 Descri ...
- POJ 3687 Labeling Balls (top 排序)
Labeling Balls Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 15792 Accepted: 4630 D ...
- poj 3687 Labeling Balls - 贪心 - 拓扑排序
Windy has N balls of distinct weights from 1 unit to N units. Now he tries to label them with 1 to N ...
随机推荐
- mysql小试题
1. 用户登录日志表 xes_user_login_logs 如下: (1) 检索登录超过两次的用户ID(sql语句) select user_id from vvt_ceshi group by u ...
- Software-Defined Networking之搬砖的故事
在很久很久以前,有一个村子. 村里的每一户,都有一个男人和一个女人. 每一户,都以搬砖为生. 从不同的地方,搬到不同的地方. 男人负责搬砖,女人负责告诉男人往哪搬. 每个家庭,都服从村委会的指挥. 村 ...
- Echarts图表常用功能配置,Demo示例
先看下效果图: 就如上图所示,都是些常用的基本配置. Legend分页,X轴设置,Y轴设置,底部缩放条设置, 数值显示样式设置,工具箱设置,自定义工具按钮, 绑定点击事件等等.这些配置代码中都做了简单 ...
- 活字格Web应用平台学习笔记 7 - 导出 Excel
活字格一直强调和Excel的兼容,可以导入导出Excel,今天终于学到这一课了. 课程目标: 好吧,就是这么快,已经加了一个“导出到Excel”的按钮了 我以为多高深呢,原来人家都给写好逻辑了,直接选 ...
- Android 状态栏开发
又好久没写了...还是记个笔记吧.这次关于Android手机App状态栏的各种处理做一个笔记. 场景一:需要做全屏,不看到手机状态栏信息(手机电量,信号等) 这种需求一般用的比较多的地方是App的Sp ...
- 禅道项目管理软件 为提交Bug页面添加“优先级”字段
为提交Bug页面添加“优先级”字段 by:授客 QQ:1033553122 测试环境: 禅道项目管理软件7.1.stable版本 备注:仅适合windows版本,linux下,直接在页面管理后台安装官 ...
- Jaguar_websocket结合Flutter搭建简单聊天室
1.定义消息 在开始建立webSocket之前,我们需要定义消息,如:发送人,发送时间,发送人id等.. import 'dart:convert'; class ChatMessageData { ...
- Python 3前言
Python具有简单.易学.免费.开源.可移植.可扩展.可嵌入.面向对象等优点,它的面向对象甚至比java和C#.net更彻底. 作为一种通用语言,Python几乎可以用在任何领域和场合,角色几乎是无 ...
- LeetCode 题解之Rotate List
1.题目描述 2.题目分析 首先数出链表中的元素个数count,然后对k = k % count 求余数.接着将链表形成一个环,从最后一个元素开始,运动 count - k 步,此时节点的下一个节点 ...
- UIAutomator环境搭建
目录 下载.安装JDK&配置Java环境变量 下载.安装SDK.ADT&配置Android环境变量 下载.安装ANT&配置ANT环境变量 创建UIAutomator工程 UIA ...