拓扑排序 C++实现

描述:

学校领导在大学生培养计划的制订中,涉及到课程的安排问题,由于课程较多,现在要求编程高手的你帮忙。假定课程之间的先修关系已确定,现在要求你根据先修关系,通过编程确定各门课程的先后关系,并生成一张课程先后安排顺序表,以帮助学校设置各年度课程。

输入:

输入只包括一个测试用例,第一行为一个自然数n,表示课程数量,第二行为n个单词,分别表示n门课程,一个单词表示一门课程,单词全为小写状态,各单词之间用一个空格隔开。

接下来为n*n行0和1构成的矩阵,表示各门课程之间的先修关系。假设i行j列为1,表示第i课程为第j课程的先修课。否则表示不存在先修关系。

输出

要求根据各课程之间的先修关系,用一行输出课程的拓扑排序结果。注意:如果两门课程的访问顺序相同,则根据课程名的字典顺序进行访问。

样例输入

4

db c computer math

0 0 0 0

1 0 1 0

0 0 0 0

0 0 1 0

 

样例输出:

c db math computer

代码:

#include<iostream>
#include <string>
using namespace std;
#define MAX 1024

class Stack      //自定义栈
{
 string str[10000];
 int statop;            //最顶元素下标加1,我习惯这样做
public:
 Stack();
    void push(string &);  //  进栈
    void  pop();
 int getsize(){return statop;}  //得到元素个数 
};
Stack::Stack()
{
 statop=0;
}
void Stack::pop()
{
 if(statop!=0)
 {
  statop--;
  cout<<str[statop];
 }
}
void Stack::push(string & te)
{
 if(statop!=10000)
 {
  str[statop]=te;
  statop++;
 }
}

Stack mystack;
int indegree[MAX];   //外面关联图中的入度

struct node
{
    int adjvex;
 string data;
    node* first;      //前驱个数(入度)
 node* next;        //后继,用来减少后面顶点的入度
}adj[MAX];           

void Create(node adj[],int n)  //建邻接表,n点数
{
    int i,j;
    for(i=1;i<=n;i++)
    {
        adj[i].adjvex=i;
  cin>>adj[i].data;
        adj[i].first=NULL;
  adj[i].next=NULL;
    }
 int u,v,t;
 node *p,*pp;
    for(i=1;i<=n;i++)
  for(j=1;j<=n;j++)
  {
   cin>>t;
   if(t==1)
   {
    u=j;
    v=i;
  
   pp=new node;
          pp->adjvex=u;
    pp->data=adj[u].data;
    pp->next=adj[v].next;
       adj[v].next=pp;
      
    p=new node;
    p->adjvex=v;
    p->data=adj[v].data;
    p->first=adj[u].first;
    adj[u].first=p;
           
   }
  }
}

void print(int n)
{
    int i;
    node *p;

cout<<"前驱:";
    for(i=1;i<=n;i++)
    {
        p=&adj[i];
        while(p!=NULL)
        {
            cout<<p->data<<' ';
            p=p->first;
        }
        cout<<endl;
    }
 cout<<"后继:";
  for(i=1;i<=n;i++)
    {
        p=&adj[i];
        while(p!=NULL)
        {
            cout<<p->data<<' ';
            p=p->next;
        }
        cout<<endl;
    }
 
}

void topsort(node adj[],int n)    //拓扑排序
{
 
    int i,pkm;
    node *p,*q;
    memset(indegree,0,sizeof(indegree));
    for(i=1;i<=n;i++)
    {
  q=&adj[i];
        p=q->first;
        while(p!=NULL)
        {
            indegree[q->adjvex]++;
            p=p->first;
        }
    }

    for(i=1;i<=n;i++)
    {
  
        if(indegree[i]==0)
  {
            mystack.push(adj[i].data);
   pkm=adj[i].adjvex;
   indegree[i]--;
   break;
  }
    }
    int count=0;

    while(mystack.getsize()!=0)
    {
        mystack.pop();
  cout<<' ';
 
        count++;
        for(p=adj[pkm].next;p!=NULL;p=p->next)
            indegree[p->adjvex]--;

  for(i=1;i<=n;i++)
   if(indegree[i]==0)
   {
    mystack.push(adj[i].data);
    pkm=adj[i].adjvex;
    indegree[i]--;
    break;
   }
    }
    cout<<endl;
 if(count<n) cout<<"有回路"<<endl;
}

int main()
{
    int n;

    cin>>n;
    Create(adj,n);
// cout<<"输入的邻接表为:"<<endl;
// print(n);
 // cout<<"拓扑排序结果为:"<<endl;
    topsort(adj,n);
    return 0;
}

运行结果:

敲了好久才敲好,不过弄懂了就是高兴的,呵呵

里面可能还少了按字典排序一个小问题,改一下就行,大概思路就是这样了。

时间: 2024-08-17 19:39:24

拓扑排序 C++实现的相关文章

拓扑排序模板

#include <stdio.h> #include <string.h> #include <iostream> #include <algorithm> #include <queue> using namespace std; const int maxn = 20010; //ip表示第几条边 //indeg表示入度 int head[maxn], ip, indeg[maxn]; int n, m, seq[maxn];//seg表示

【算法小总结】拓扑排序+例题解析

题目1449:确定比赛名次 时间限制:1 秒内存限制:128 兆特殊判题:否提交:669解决:293 题目描述: 有N个比赛队(1<=N<=500),编号依次为1,2,3,....,N进行比赛,比赛结束后,裁判委员会要将所有参赛队伍从前往后依次排名,但现在裁判委员会不能直接获得每个队的比赛成绩,只知道每场比赛的结果,即P1赢P2,用P1,P2表示,排名时P1在P2之前.现在请你编程序确定排名. 输入: 输入有若干组,每组中的第一行为二个数N(1<=N<=500),M:其中N表示队伍

拓扑排序(三) Java详解

拓扑排序介绍 拓扑排序(Topological Order)是指,将一个有向无环图(Directed Acyclic Graph简称DAG)进行排序进而得到一个有序的线性序列. 这样说,可能理解起来比较抽象.下面通过简单的例子进行说明! 例如,一个项目包括A.B.C.D四个子部分来完成,并且A依赖于B和D,C依赖于D.现在要制定一个计划,写出A.B.C.D的执行顺序.这时,就可以利用到拓扑排序,它就是用来确定事物发生的顺序的. 在拓扑排序中,如果存在一条从顶点A到顶点B的路径,那么在排序结果中B

拓扑排序(二) C++详解

拓扑排序介绍 拓扑排序(Topological Order)是指,将一个有向无环图(Directed Acyclic Graph简称DAG)进行排序进而得到一个有序的线性序列. 这样说,可能理解起来比较抽象.下面通过简单的例子进行说明! 例如,一个项目包括A.B.C.D四个子部分来完成,并且A依赖于B和D,C依赖于D.现在要制定一个计划,写出A.B.C.D的执行顺序.这时,就可以利用到拓扑排序,它就是用来确定事物发生的顺序的. 在拓扑排序中,如果存在一条从顶点A到顶点B的路径,那么在排序结果中B

拓扑排序(一) C语言详解

拓扑排序介绍 拓扑排序(Topological Order)是指,将一个有向无环图(Directed Acyclic Graph简称DAG)进行排序进而得到一个有序的线性序列. 这样说,可能理解起来比较抽象.下面通过简单的例子进行说明! 例如,一个项目包括A.B.C.D四个子部分来完成,并且A依赖于B和D,C依赖于D.现在要制定一个计划,写出A.B.C.D的执行顺序.这时,就可以利用到拓扑排序,它就是用来确定事物发生的顺序的. 在拓扑排序中,如果存在一条从顶点A到顶点B的路径,那么在排序结果中B

poj 1094 Sorting It All Out(拓扑排序)

链接: http://poj.org/problem?id=1094 题目: Sorting It All Out Time Limit: 1000MS     Memory Limit: 10000K Total Submissions: 21532     Accepted: 7403 Description An ascending sorted sequence of distinct values is one in which some form of a less-than ope

HDU 1811 Rank of Tetris(拓扑排序+并查集)

链接: http://acm.hdu.edu.cn/showproblem.php?pid=1811 题目: Problem Description 自从Lele开发了Rating系统,他的Tetris事业更是如虎添翼,不久他遍把这个游戏推向了全球. 为了更好的符合那些爱好者的喜好,Lele又想了一个新点子:他将制作一个全球Tetris高手排行榜,定时更新,名堂要比福布斯富豪榜还响.关于如何排名,这个不用说都知道是根据Rating从高到低来排,如果两个人具有相同的Rating,那就按这几个人的R

UVa 10305:Ordering Tasks , 经典的拓扑排序

题目链接: http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=105&page=show_problem&problem=1246 题目类型: 拓扑排序 题目: John has n tasks to do. Unfortunately, the tasks are not independent and the execution of one task is onl

UVa 200 Rare Order:拓扑排序

200 - Rare Order Time limit: 3.000 seconds http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=24&page=show_problem&problem=136 题意:给一列单词,这些单词是按一种未知的字典顺序排列的,要求输出这些字母的顺序. 思路:很明显,拓扑排序. 注意最后要多输出一个换行,否则会WA! 完整代码: /*0.0

算法研究:AOV网与拓扑排序

在一个表示工程的有向图中,用顶点表示活动,用弧表示活动之间的优先关系,这样的有向图为顶点表示活动的网,我 们称之为AOV网(Activity on Vextex Network).AOV网中的弧表示活动之间存在的某种制约关系,AOV网中不能存在回路, 让某个活动的开始要以自己完成作为先决条件,显然是不可以的. 设G= { V, E }是一个具有n个顶点的有向图,V中 的顶点序列v1, v2, ...,vn,满足若从顶点vi到vj有一条路径,则在顶点序列中顶点vi必在vj之前,则我们称这样的顶点