2017-09-12 19:50:58

writer:pprp

最近刚开始接触拓扑排序,拓扑排序适用于:无圈图的顶点的一种排序,

用来解决有优先级别的排序问题,比如课程先修后修,排名等。

主要实现:用矩阵来储存图,用indegree数组记录每个顶点的入度,

从入度为0的开始,每次删除该入度为0的点,然后修改其他顶点的入度,

在进行查找入度为0的顶点,循环下去就可以

题意如下:给你n个队伍,m个优先顺序,让你输出总的排名的优先顺序。

代码如下:

/*
@theme:拓扑排序 hdu 1285
@writer:pprp
@declare: accepted
@begin:19:00
@end:19:47
@error:
*/ #include <iostream>
#include <cstring>
#include <cstdio> using namespace std;
const int maxn = ;
bool G[maxn][maxn];
int ans[maxn];
int indegree[maxn];//存取入度
int n, m; void init()
{
memset(indegree,,sizeof(indegree));
memset(ans,,sizeof(ans));
memset(G,,sizeof(G));
} void toposort()
{
for(int i = ; i <= n ; i++)
{
for(int j = ;j <= n ;j++)
{
if(G[i][j])
indegree[j]++;
}
}
//应该从入度为0的开始
for(int i = ; i <= n;i++)
{
int k = ;
while(indegree[k]!=)
k++;
ans[i] = k;
//删除根节点
indegree[k] = -;
for(int j = ; j <= n; j++)
{
if(G[k][j])
indegree[j]--;
}
} } int main()
{
//freopen("in.txt","r",stdin);
int x, y;
while(cin >> n >> m)
{
init();
for(int i = ; i < m ;i++)
{
cin >> x >> y;
G[x][y] = ;
} toposort(); for(int i = ; i < n; i++)
cout << ans[i] << " ";
cout << ans[n] << endl;
} return ;
}

分析当前的:

每次删掉一个点以后就要遍历全部的点找到入度为0的点,其实入度为0的点只可能出现在改动之后的点,

所以采用了优先队列,记录下来当前的入度为0的点,将其输出或者储存下来以后,改变由该点指向的点的入度

从这些改动的点中看看有没有入度为0的点,如果有的话,那就将其入队列,直到队列为空,或者记录的已经被删除的点的个数是所有的点

那么退出完成程序

优化代码如下:

/*
@theme:拓扑排序 hdu 1285
@writer:pprp
@declare: accepted
@begin:19:59
@end:20:38
@error:注意保持头脑清醒
*/ #include <iostream>
#include <cstring>
#include <algorithm>
#include <queue>
#include <cstdio> using namespace std;
const int maxn = ;
int indegree[maxn];
int n, m, x, y; bool G[maxn][maxn]; void topsort()
{
priority_queue<int,vector<int>,greater<int> > pq; for(int i = ; i <= n; i++)
for(int j = ; j <= n; j++)
if(G[i][j])
indegree[j]++; for(int i = ; i <= n; i++)
if(indegree[i] == )
pq.push(i),indegree[i] = -1; int counter = ; while(!pq.empty())
{
int v = pq.top();
pq.pop(); if(counter != n)
{
cout << v << " ";
counter++;
}
else
{
cout << v << endl;
break;
} for(int i = ; i <= n;i++)
{
if(G[v][i] == )
{
indegree[i]--;
if(indegree[i] == )
pq.push(i),indegree[i] = -1;
}
}
}
if(counter != n)
cerr << "graph has a cycle." << endl;
} void init()
{
memset(indegree,,sizeof(indegree));
memset(G,,sizeof(G));
} int main()
{
freopen("in.txt","r",stdin);
while(cin >> n >> m)
{
init();
for(int i = ; i < m ; i++)
{
cin >> x >> y;
G[x][y] = ;
}
topsort();
} return ;
}

拓扑排序 - hdu 1285(普通和优先队列优化)的更多相关文章

  1. 拓扑排序 HDU - 5695

    众所周知,度度熊喜欢各类体育活动. 今天,它终于当上了梦寐以求的体育课老师.第一次课上,它发现一个有趣的事情.在上课之前,所有同学要排成一列, 假设最开始每个人有一个唯一的ID,从1到NN,在排好队之 ...

  2. 拓扑排序 --- hdu 4948 : Kingdom

    Kingdom Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Sub ...

  3. hdu1532 用BFS求拓扑排序

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1285 题目给出一些点对之间的先后顺序,要求给出一个字典序最小的拓扑排列.对于拓扑排序的问题,我们有DF ...

  4. bzoj4383 [POI2015]Pustynia 拓扑排序+差分约束+线段树优化建图

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=4383 题解 暴力的做法显然是把所有的条件拆分以后暴力建一条有向边表示小于关系. 因为不存在零环 ...

  5. hdu 1285 确定比赛名次 拓扑排序

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1285 有N个比赛队(1<=N<=500),编号依次为1,2,3,....,N进行比赛,比赛 ...

  6. 图论之初,拓扑排序、前向星(通过存储边来存储图)加优先队列对拓扑的优化-----hdu1285

    确定比赛名次 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Sub ...

  7. ACM: HDU 1285 确定比赛名次 - 拓扑排序

     HDU 1285 确定比赛名次 Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u De ...

  8. HDU 5638 拓扑排序+优先队列

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5638 题意: 给你一个DAG图,删除k条边,使得能个得到字典序尽可能小的拓扑排序 题解: 把拓扑排序 ...

  9. 正向与反向拓扑排序的区别(hdu 1285 确定比赛名次和hdu 4857 逃生)

    确定比赛名次 Time Limit : 2000/1000ms (Java/Other)   Memory Limit : 65536/32768K (Java/Other) Total Submis ...

随机推荐

  1. Spark源码分析 -- SchedulableBuilder

    SchedulableBuilder就是对Scheduleable tree的封装, 在Pool层面(中间节点), 完成对TaskSet的调度(FIFO, FAIR) 在TaskSetManager ...

  2. 在任何mac上用u盘安装OSX和Windows10双系统的方法(支持老电脑、不用Bootcamp)

    Win10是微软主推的,兼容性做的还不错,安装工具做的适应性好. 而且很多Mac机上的Bootcamp不支持u盘安装. 1.先安装OSX,一般电脑自带(建议升级到最新版).如果装了新的ssd,重新安装 ...

  3. JS获取当前网页大小以及屏幕分辨率等

    网页可见区域宽:document.body.clientWidth 网页可见区域高:document.body.clientHeight 网页可见区域宽:document.body.offsetWid ...

  4. 前端构建工具gulpjs的使用介绍及技巧(一)

    原文链接:http://www.cnblogs.com/2050/p/4198792.html gulpjs是一个前端构建工具,与gruntjs相比,gulpjs无需写一大堆繁杂的配置参数,API也非 ...

  5. JQuery能够高效地操作页面元素

    关于DOM和CSS的页面元素选择: $("span");     //全部span元素 $("#elem");  //id为elem的元素 $(".c ...

  6. Mybatis的MapperRegistry错误

    1,如果mabtis的配置文件里mapper用的不是包扫描,而是: <mapper resource="com/vmpjin/mapper/OrdersMapper.xml" ...

  7. mysql监控优化(三)慢查询

    顾名思义,慢查询日志中记录的是执行时间较长的query,也就是我们常说的slowquery,通过设--log-slow-queries[=file_name]来打开该功能并设置记录位置和文件名.慢查询 ...

  8. list_for_each_entry解析

    双向链表及链表头: 建立一个双向链表通常有一个独立的用于管理链表的链表头,链表头一般是不含有实体数据的,必须用INIT_LIST_HEAD()进行初始化,表头建立以后,就可以将带有数据结构的实体链表成 ...

  9. (转)spring mvc forward与redirect

    forward 转发,如return "forward:/hello"; 浏览器的地址栏不会变,但是有视图返回来 redirect 重定向,如return "redire ...

  10. Oracle获取当前session ID的方法

    1.使用v$mystat视图获取当前session的ID select sid from v$mystat; 2.使用userenv内部函数获取当前session的ID select userenv( ...