随笔 - 251  文章 - 504  trackbacks - 0
<2006年10月>
24252627282930
1234567
891011121314
15161718192021
22232425262728
2930311234

本博客系个人收集材料及学习记录之用,各类“大侠”勿扰!

留言簿(14)

随笔分类

收藏夹

My Favorite Web Sites

名Bloger

非著名Bloger

搜索

  •  

积分与排名

  • 积分 - 200061
  • 排名 - 285

最新评论

学生选课系统

大学里实行学分制。每门课都有一定的学 。每个学生均需要修满规定数量的课程才能毕业。其中有些课程可以直接修读,有些课程需要一定的基础知识,必须在选了其他一些课程的基础上才能修读。例如,《数据结构》必须在选修了《高级语言程序设计》之后才能选修。我们称《高级语言程序设计》是《数据结构》的“先修课”。在我们的大学里,假定每门课的直接先修课至多只有一门,两门课可能存在相同的先修课。例如:

课号

先修课号

学分

1

0

1

2

1

1

3

2

3

4

0

3

5

2

4

 

上例中,12的先修课,即如果要选修2,则1必定被选。

学生不可能学完大学里开设的所有课程,因此每个学生必须在入学时选定自己要学的课程。每个学生可选课程的总数目是给定的。现在请你们小组编写一个“学生选课系统”,任意给定一种课程体系(总课程数,课程之间的修读先后制约关系,学生毕业要求修的课程数目),该系统能帮助学生找出一种选课方案,使得他能得到的学分最多,并且必须满足先修课程优先的原则

具体实现如下:   

1.          课程体系存放为课程体系文件 CourseHierarchy.txt

2.          课程体系文件 CourseHierarchy.txt 转换左孩子右兄弟二叉树表示

3.          在此基础上对二叉树进行先序、中序、后序遍历。

4.       给出最多学分选课方案。

将课程体系文件转换为二叉树

思想:从 CourseHierarchy.txt 头部开始扫描,第一个先修课为 0 (即没有先修课)的课程为整棵二叉树的根节点 i 。从 CourseHierarchy.txt 头部开始扫描,所有 i 的后续课程作为 i 的左子树;课程体系中,其他先修课为 0 的节点,及其后续课程构成 i 的右子树。第一个先修课为 i 的节点作为 i 的左子树的根节点 ,从 CourseHierarchy.txt 头部开始扫描,其他以 i 为先修课的课程构成 的右子树。如此低归构造出课程体系二叉树

步骤 1 :从文件 CourseHierarchy.txt 生成课程链表 Course_Slist (参数表)。

步骤 2 :将指针 CousreSystem_root 指引的 课程链表递归转换为课程体系二叉树
步骤 3 :先序输出二叉树,以验证生成的二叉树是否正确
实现代码
tree_binary.cpp
// tree_binary.cpp : 定义控制台应用程序的入口点。
//
#pragma once


#include <iostream>
#include <tchar.h>

//#include<fstream>
//#include<iostream>
//#include<cstdlib>
#include"tree_binaryHF.h"
using namespace std;
int _tmain(int argc, _TCHAR* argv[])
{
 char file_name[20]="CourseHierarchy.txt";
 LinkList L;//链表的头指针
 LinkList T;//树的根
 cout<<"\t本程序的作用是从文件(文件内容可按需求改写)"<<endl
  <<"        中读取信息,放到链表中,最后生成二叉树,"<<endl
  <<"        然后先序中序,后续遍历二叉树."<<endl;
 InitList(L);
 cout<<"读出的文件内容:"<<endl;
 OpenFile(L,file_name);
 cout<<"生成链表后的结果:"<<endl;
 PrintLinkLIst(L);
    CreateBiTree(L,T,0);//生成二叉树
 cout<<"先序遍历二叉树的结果:";
 PreOrder(T);
 cout<<endl<<"中序遍历二叉树的结果:";
 InOrder(T);
 cout<<endl<<"后序遍历二叉树的结果:";
    BackOrder(T);
 cout<<endl;
 return 0;
}

 


tree_binaryHF.h
#include"cao.hpp"
//*************************************//
//打开文件,并读取里面的所有内容
Status OpenFile(LinkList&L,char file_name[])//由于这里需要把数据插入链表中
{
 int CNum,CScore,PreScore;
 ifstream in_stream;
 in_stream.open(file_name);
 if(in_stream.fail())//if the file is not opened successfully
 {
  cout<<"can't open the file of "<<file_name<<endl;
  exit(0);
 }
 //i
 cout<<"CNum"<<"\t"<<"PreScore"<<"\t"<<"CScore"<<endl;
 while(in_stream>>CNum>>PreScore>>CScore)//读取三个数据(课程编码,学分,先修课)
 {
  cout<<CNum<<"\t"<<PreScore<<"\t"<<"\t"<<CScore<<endl; //放到节点中
  LinkInsert(L,CNum,PreScore,CScore);//insert the node to the linklist L
 }
 in_stream.close();
 return OK;
}
Status InitList(LinkList &L)
{
 L= (LinkList)malloc(sizeof(LNode));
 if(!L)
  return ERROR;
 L->lchild=NULL;
 L->rchild=NULL;
 L->CNumber=0;
 L->CScore=0;
 L->PreCourse=0;
 return OK;
}
Status LinkInsert(LinkList &L,Status CNum,Status PreCourse,Status CScore)//把新得到的结点插入链表,从头结点处插入
{
 LinkList p;
 p=(LinkList)malloc(sizeof(LNode));
 if(!p)
  return ERROR;
 p->CNumber=CNum;
 p->CScore=CScore;
 p->PreCourse=PreCourse;
 p->lchild=NULL;
 p->rchild=L->rchild;
 L->rchild=p;
 return OK;
}
Status PrintLinkLIst(LinkList &L)//打印整个链表,右指针指向后继结点,左指针空
{
 LinkList p;
 p=L->rchild;
 while(p)
 {
  cout<<p->CNumber<<"\t"<<p->PreCourse<<"\t"<<"\t"<<p->CScore<<endl;
  p=p->rchild;
 }
 return OK;
}
Status CreateBiTree(LinkList &L,LinkList&T,Status c)
{
 LinkList p;
 //while(!EmptyLinkList(L))
 //{
 if(!(p=Serach(L,c)))
  T=NULL;
 else
 {
  T=p;//生成根节点
  delet(L,c);//删除p结点
  CreateBiTree(L,T->lchild,T->CNumber);//构造左子树
  CreateBiTree(L,T->rchild,T->PreCourse);//构造右子树
 }
 //}
 return OK;
}
LinkList Serach(LinkList &L,Status c)
{
 LinkList head,next;
 head=L;
 next=head->rchild;
 while(next&&(next->PreCourse!=c))
 {
  head=next;
  next=next->rchild;
 }
 if(next==NULL)
  return NULL;//没有找到
 else//找到了
  return next;
}
void delet(LinkList &L,Status c)
{
 LinkList head,next;
 head=L;
 next=head->rchild;
 while(next&&(next->PreCourse!=c))
 {
  head=next;
  next=next->rchild;
 }
 head->rchild=next->rchild;
 //free(next);
}
Status EmptyLinkList(LinkList L)
{
 if(L->rchild=NULL)
  return OK;
 else
  return ERROR;
}
//先序遍历的递归
void PreOrder(LinkList T)
{
 if(T)
 {
   //访问结点
  cout<<T->CNumber<<" ";
  PreOrder(T->lchild);   //遍历左子树
  PreOrder(T->rchild);   //遍历右子树
 //cout<<endl;
 }

//中序遍历的递归
void InOrder(LinkList T)
{
 if(T)
 {
  InOrder(T->lchild);   //遍历左子树
   //访问结点
  cout<<T->CNumber<<" ";
  InOrder(T->rchild);   //遍历右子树
 }

//后序遍历的递归
void BackOrder(LinkList T)
{
 if(T)
 {
  BackOrder(T->lchild);   //遍历左子树
  BackOrder(T->rchild);   //遍历右子树
  cout<<T->CNumber<<" ";
}

 


cao.hpp
#include<fstream>
#include<iostream>
#include<cstdlib>
using namespace std;
typedef int Status;
#define OK   1
#define ERROR 0
typedef struct LNode            
{
 struct LNode *lchild;
 struct LNode *rchild; 
 Status CNumber;
 Status CScore;   
 Status PreCourse;
}LNode,*LinkList;
Status OpenFile(LinkList&L,char file_name[]);
//打开文件,并且插入节点
//*************************//
LinkInsert(LinkList &L,Status CNum,Status CScore,Status PreScore);
//insert the node to the linklist L
//*************************//
Status InitList(LinkList &L);//初始化链表
//****************************************//
Status PrintLinkList(LinkList &L);//输出链表
//********************************//
LinkList Serach(LinkList &L,Status c);//查找节点
//*********************************************//
void delet(LinkList&L,Status c);//删除节点
//*************************************//
Status EmptyLinkList(LinkList L);//检查链表是否为空(没用到)
//***********************************//
//先序遍历二叉树
void PreOrder(LinkList T);
//***********************************//
void InOrder(LinkList T);//中序遍历二叉树
//*****************************//
void BackOrder(LinkList T);//后续遍历二叉树

CourceSource.txt
1 2 2
2 0 1
3 0 4
4 2 1
5 7 1
6 7 6
7 2 2


运行结果
先序遍历二叉树:3 2 7 6 5 4 1
中序遍历二叉树:3 6 5 7 4 1 2
后序遍历二叉树:5 6 1 4 7 2 3
posted on 2006-10-29 14:57 matthew 阅读(1372) 评论(0)  编辑  收藏 所属分类: 数据结构与算法设计

只有注册用户登录后才能发表评论。


网站导航: