ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

[HNOI2015]菜肴制作 题解(贪心+拓扑)

2019-07-11 13:50:50  阅读:257  来源: 互联网

标签:菜肴 顺序 限制 题解 HNOI2015 尽量 include 制作 贪心


Description

知名美食家小 A被邀请至ATM 大酒店,为其品评菜肴。 

ATM 酒店为小 A 准备了 N 道菜肴,酒店按照为菜肴预估的质量从高到低给予 1到N的顺序编号,预估质量最高的菜肴编号为1。由于菜肴之间口味搭配的问题, 某些菜肴必须在另一些菜肴之前制作,具体的,一共有 M 条形如“i 号菜肴‘必须’ 先于 j 号菜肴制作”的限制,我们将这样的限制简写为<i,j>。现在,酒店希望能求 出一个最优的菜肴的制作顺序,使得小 A能尽量先吃到质量高的菜肴:也就是说, (1)在满足所有限制的前提下,1 号菜肴“尽量”优先制作;(2)在满足所有限制,1 号菜肴“尽量”优先制作的前提下,2号菜肴“尽量”优先制作;(3)在满足所有限 制,1号和2号菜肴“尽量”优先的前提下,3号菜肴“尽量”优先制作;(4)在满 足所有限制,1 号和 2 号和 3 号菜肴“尽量”优先的前提下,4 号菜肴“尽量”优 先制作;(5)以此类推。  例1:共4 道菜肴,两条限制<3,1>、<4,1>,那么制作顺序是 3,4,1,2。例2:共 5道菜肴,两条限制<5,2>、 <4,3>,那么制作顺序是 1,5,2,4,3。例1里,首先考虑 1, 因为有限制<3,1>和<4,1>,所以只有制作完 3 和 4 后才能制作 1,而根据(3),3 号 又应“尽量”比 4 号优先,所以当前可确定前三道菜的制作顺序是 3,4,1;接下来 考虑2,确定最终的制作顺序是 3,4,1,2。例 2里,首先制作 1是不违背限制的;接 下来考虑 2 时有<5,2>的限制,所以接下来先制作 5 再制作 2;接下来考虑 3 时有 <4,3>的限制,所以接下来先制作 4再制作 3,从而最终的顺序是 1,5,2,4,3。  现在你需要求出这个最优的菜肴制作顺序。无解输出“Impossible!” (不含引号, 首字母大写,其余字母小写) 

Input

 第一行是一个正整数D,表示数据组数。 

接下来是D组数据。  对于每组数据:  第一行两个用空格分开的正整数N和M,分别表示菜肴数目和制作顺序限 制的条目数。  接下来M行,每行两个正整数x,y,表示“x号菜肴必须先于y号菜肴制作” 的限制。(注意:M条限制中可能存在完全相同的限制) 

Output

 输出文件仅包含 D 行,每行 N 个整数,表示最优的菜肴制作顺序,或

者”Impossible!”表示无解(不含引号)。 

Sample Input

3
5 4
5 4
5 3
4 2
3 2
3 3
1 2
2 3
3 1
5 2
5 2
4 3

Sample Output

1 5 3 4 2
Impossible!
1 5 2 4 3  

拓扑水题

注意小号尽量考前!=字典序最小

显然建反图

我一开始搞了一个小根堆,然后每次取出能拓扑的就dfs拓扑,

但是这样无法保证内部拓扑序最优

于是把所有点扔进去用大根堆直接跑拓扑,

反向输出即可

 

#include<cstdio>
#include<iostream>
#include<cstring>
#include<queue>
#include<vector>
using namespace std;
priority_queue<int,vector<int>,less<int> >q;
const int N=100005;
vector<int> ans;
int T,n,m,to[N],nxt[N],head[N],tot=0,link[N],vis[N],deg[N],num=0;
void ini()
{
    for(int i=1;i<=n;i++)
        to[i]=nxt[i]=head[i]=vis[i]=link[i]=deg[i]=0;
    tot=num=0;ans.clear();
}
void add(int x,int y)
{
    to[++tot]=y;
    nxt[tot]=head[x];
    head[x]=tot;
}
bool topsort()
{
    for(int i=1;i<=n;i++)
        if(!deg[i])q.push(i);
    while(!q.empty())
    {
        int u=q.top();
        ans.push_back(u);
        q.pop();
        for(int i=head[u];i;i=nxt[i])
            if(!(--deg[to[i]]))q.push(to[i]);
    }
    if(ans.size()!=n)return false;
    return true;
}
void work()
{
    ini();
    scanf("%d%d",&n,&m);
    for(int i=1;i<=m;i++)
    {
        int x,y;
        scanf("%d%d",&x,&y);
        add(y,x);deg[x]++;
    }
    if(!topsort())
    {
        puts("Impossible!");
        return ;
    }
    for(int i=ans.size()-1;i>=0;i--)
        printf("%d ",ans[i]);
    puts(" ");
}
int main()
{
    scanf("%d",&T);
    while(T--)work();
    return 0;
}

 

标签:菜肴,顺序,限制,题解,HNOI2015,尽量,include,制作,贪心
来源: https://www.cnblogs.com/Rorschach-XR/p/11169499.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有