3355: [Usaco2004 Jan]有序奶牛

Time Limit: 10 Sec  Memory Limit: 128 MB
Submit: 37  Solved: 19
[Submit][Status][Discuss]

Description

    约翰的N(1≤N≤1500)头牛排成一行挤奶时,有确定的顺序.牛被编成连续的号码1..N,他拥有L条关于奶牛顺序的信息,所有的信息都写成“A在B的前面”这样的形式,而且他知道最后一条是多余的.他觉得,有些冗余信息可以由其他信息推出,可以对这些信息进行精减.请帮助约翰删除尽可能多的冗余信息,但要保证能推出原有的顺序.可以保证的是,答案唯一,且最初的信息没有矛盾,如A在B前面,B在A前面.

Input

 
    第1行:两个整数N和L.
    第2到L+1行:每行两个整数X和Y(1≤X,y≤N),表示X在Y前.无重复.

Output

 
    第1行:整数U.
    第2到U+I行:输出精减后的信息,每行2个数字,按第1列数字排序.

Sample Input

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

Sample Output

4
2 1
3 5
4 2
5 2

HINT

3在1前,4在1前可推.输出的每一行,不能被其他推出.

分析:对于一条边x->y,若去掉之后x不能到达y,那么它是必需的。我们首先需要求出拓扑序列,然后我们按照拓扑序列反向加边,这样的目的就是为了在已知顺序的情况下维护哪些边可以互相到达,然后用一个bieset去判断一下哪些边是必须的。感谢claris和付大神对我的帮助

 #include "iostream"
#include "cstdio"
#include "cstring"
#include "bitset"
#include "algorithm"
using namespace std;
const int maxn=+;
const int maxm=+;
int n,m,ed,ed2;
int ind[maxn]; //入度
int adj[maxm]; //栈顶边对应的顶点
int next1[maxm]; //上一条边的编号
int tail[maxm]; //栈顶边的编号
int vis[maxn]; //存储拓扑排序序列
bitset<maxn>f[maxn];
void add(int x,int y){
ind[y]++;
adj[++ed]=y;
next1[ed]=tail[x];
tail[x]=ed;
}
int adj2[maxm];
int next2[maxm];
int tail2[maxm];
void add2(int x,int y){
adj2[++ed2]=y;
next2[ed2]=tail2[x];
tail2[x]=ed2;
}
typedef pair<int,int> P;
P p[maxm];
int main()
{
//freopen("a.txt", "r", stdin);
//freopen("b.txt", "w", stdout);
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++) f[i][i]=;
ed=;
while(m--){ //数组式邻接表建图
int x,y;
scanf("%d%d",&x,&y);
add(x,y);
} //构造拓扑排序序列
int cnt=;
for(int i=;i<=n;i++)
if(!(ind[i]))
vis[++cnt]=i; int h=;
ed2=;
while(h<=cnt){
int a;
for(int i=tail[a=vis[h++]];i;add2(adj[i],a),i=next1[i])
if(!(--ind[adj[i]]))
vis[++cnt]=adj[i];
} int ans=;
for(int i=;i<=n;i++){
int a;
for(int j=tail2[a=vis[i]];j;f[a]|=f[adj2[j]],j=next2[j])
if(!f[a][adj2[j]])
p[++ans]=P(adj2[j],a);
}
sort(p+,p++ans);
printf("%d\n",ans);
for(int i=;i<=ans;i++)
printf("%d %d\n",p[i].first,p[i].second);
return ;
}

BZOJ3355的更多相关文章

  1. BZOJ3355 : [Usaco2004 Jan]有序奶牛

    对于一条边x->y,若去掉之后x不能到达y,那么它是必需的. 首先拓扑排序求出拓扑序,然后按照终点拓扑序为第一关键字,起点拓扑序为第二关键字从小到大加边. 对于每个点,维护一个bitset,表示 ...

随机推荐

  1. mysql 建立utf8字符集数据库

    CREATE DATABASE `evaluate` DEFAULT CHARACTER SET utf8 COLLATE utf8_general_ci;

  2. [Unit Testing] Fundamentals of Testing in Javascript

    In this lesson, we’ll get the most fundamental understanding of what an automated test is in JavaScr ...

  3. 关于android 使用bitmap的OOM心得和解决方式

    android开发,从2010年開始学习到如今的独立完毕一个app,这漫长的四年,已经经历了非常多次bug的折磨.无数次的加班训练.然而,自以为自己已经比較了解android了,却近期在一个项目上.由 ...

  4. weexapp 开发流程(一)开发环境配置

    1.创建项目 weexpack create weexapp 2.安装必要插件 npm i jwt-simple vue-resource vue-router vuex vuex-router-sy ...

  5. php闭包实例

    php闭包函数,一个典型的实例 function getMoney() { $rmb = 1; $dollar = 6; $func = function($dollar) use (&$rm ...

  6. 聊聊高并发(二十四)解析java.util.concurrent各个组件(六) 深入理解AQS(四)

    近期总体过了下AQS的结构.也在网上看了一些讲AQS的文章,大部分的文章都是泛泛而谈.又一次看了下AQS的代码,把一些新的要点拿出来说一说. AQS是一个管程.提供了一个主要的同步器的能力,包括了一个 ...

  7. Codeforces 558C Amr and Chemistry

    题意: n个数.每次能够选一个数 让其 *=2 或者 /=2 问至少操作多少次使得全部数相等. 思路: 对于每一个数,计算出这个数能够变成哪些数,以及变成那个数的最小步数,用两个数组保存 cnt[i] ...

  8. 解决编译twrp3.0.3遇到的问题

    1. 问题: ninja: error: '/home/jessie/OMNI/out/target/product/m1/obj/SHARED_LIBRARIES/libcryptfs_hw_int ...

  9. Object类及其常用方法简介

    https://www.cnblogs.com/wxywxy/p/6740277.html Object类是一个特殊的类,是所有类的父类,如果一个类没有用extends明确指出继承于某个类,那么它默认 ...

  10. Canvas坐标轴中的Y轴距离是X轴的两倍

    如题,相信很多人在初次玩canvas的时候会出现这样的情况,跟着教程走的情况下,诶 怎么画出来的东西,不怎么对劲啊,,,ԾㅂԾ,,!!!!!先上代码 <!DOCTYPE html> < ...