第一篇:数据结构实验报告
浙江师范大学
实 验 报 告
学 院: 数理与信息工程学院 专 业: 计算机科学与技术 姓 名: 杨富生 学 号: 201531910137 课程名称: 数据结构 指导教师: 钟发荣 实验时间: 2016-06-15
2016年6月15日
实验一
1.实验要求
1.1 掌握数据结构中线性表的基本概念。
1.2 熟练掌握线性表的基本操作:创建、插入、删除、查找、输出、求长度及合并并运算在顺序存储结构上的实验。
2.实验内容
2.1 编写一个函数,从一个给定的顺序表A中删除元素值在x到y之间的所有元素,要求以较高效率来实现。
#include
A[i-k]=A[i];i++;} return(n-k);} void main(){ int i,j;int a[maxsize];printf(“输入%d个数:n”,maxsize);for(i=0;i scanf(“%d,”,&a[i]); j=del(a,maxsize,1,3); printf(“输出删除后剩下的数:n”); for(i=0;i ”n,a[i]);} 2.2 试写一个算法,在无头结点的动态单链表上实现线性表插入操作INSERT(L,i,b)。 void Insert(Linklist &L,int i,elemtype x){ if(!L){ } L=(Linklist)malloc(sizeof(Lnode));(*L).data=x;(*L).next=NULL;} else { if(i==1){ s=(Linklist)malloc(sizeof(Lnode)); s->data=x;s->next=L;L=s;} else { p=L;j=1; while(p&&j {j++;p=p->next;} if(p||j>i-1) return error; s=(Linklist)malloc(sizeof(Lnode)); s->data=x;s->next=p->next;p->next=s;} } 2.3 生成两个多项式PA和PB,求他们的和,输出“和多项式”。typedef struct node {int exp;float coef;struct node *next;}polynode;polynode *polyadd(polynode *pa,polynode *pb){ polynode *p,*q,*pre,*r;float x;p=pa->next;q=pb->next;pre=pa;while((p!=NULL)&&(q!=NULL)) if(p->exp>q->exp) { r=q->next; q->next=p; pre->next=q; pre=q; q=r; } } else if(p->exp==q->exp){ x=p->coef+q->coef; if(x!=0) {p->coef=x; s=p; } else {pre->next=p->next; free(p); } p=pre->next; r=p; q=q->next; free(r);} else if(p->exp { pre=p; p=p->next; } if(q!=NULL) pre->next=q; free(pb);2.4 设计一个统计选票的算法,输出每个候选人的得票结果。 typedef int elemtype typedef struct linknode { elemtype data;struct linknode *next;}nodetype;nodetype *create(){ elemtype d;nodetype h=NULL,*s,*t;int i=1;printf(“建立单链表:n”);while(1){ printf(“输入第%d个结点数据域”,i); scanf(“%d”,&d); if(d==0)break; if(i==1) { h=(nodetype *)malloc(sizeof(nodetype)); h->data=d;h->next=NULL;t=h; } else { s=(nodetype *)malloc(sizeof(nodetype)); s->data=d;s->next=NULL;t->next=s; t=s; } i++;} return h;} void sat(nodetype *h,int a[]){ nodetype *p=h;while(p!=NULL){ a[p->data]++; p=p->next;} } void main(){ int a[N+1],i;for(i=0;i a[i]=0;nodetype *head;head=create();sat(head,a);printf(“候选人:”);for(i=1;i<=N;i++)printf(“%3d”,i);printf(“n得票数n”);for(i=1;i<=N;i++) printf(“%3d”,a[i]);printf(“n”);} 3.实验心得体会 线性表是最简单的、最常用的一种数据结构,是实现其他数据结构的基础。 实验二 1.实验要求 1.1 了解栈和队列的特性,以便灵活运用。1.2 熟练掌握栈和有关队列的各种操作和应用。 2.实验内容 2.1 设一个算术表达式包括圆括号,方括号和花括号三种括号,编写一个算法判断其中的括号是否匹配。#include push(*str,L);else if(*str==')'||*str==']'||*str=='}') if(pop(*str,L)) {puts(“错误,请检查!”); puts(“按回车键退出”); getchar();exit(-2); } str++;} if(L->next){puts(“错误,请检查!”);puts(“按任意键退出”);getchar();exit(-2);} } void push(char c,list *L){list *p;p=(list *)malloc(sizeof(list));if(!p){ printf(“错误!”);exit(-2);} p->str=c;p->next=L->next;L->next=p;} #define check(s)if(L->next->str==s){p=l->next;L->next=p->next;free(p);return(0);} int pop(char c,list *L){ list *p;if(L->next==NULL)return 1;switch(c){ case')':check('(')break;case']':check('[')break;case'}':check('{')break;} return 1; 实验三 1.实验要求 1.1 掌握二叉树,二叉树排序数的概念和存储方法。1.2 掌握二叉树的遍历算法。 1.3 熟练掌握编写实现树的各种运算的算法。2.实验内容 2.1 编写程序,求二叉树的结点数和叶子数。#include bt=(struct node *)malloc(sizeof(bnode)); bt->data=ch; bt->lchild=creat(); bt->rchild=creat();} return bt;} int n=0,n1=0;void preorder(blink bt){ if(bt){ n++; if(bt->lchild==NULL&&bt->rchild==NULL) n1++; preorder(bt->lchild); preorder(bt->rchild);} } void main(){ } blink root;root=creat();preorder(root);printf(“此二叉数的接点数有:%dn”,n);printf(“此二叉数的叶子数有:%dn”,n1);2.2 编写递归算法,求二叉树中以元素值为X的结点为根的子数的深度。int get_deep(bitree T,int x){ if(T->data==x){ printf(“%dn”,get_deep(T));exit 1;} else { if(T->lchild)get_deep(T->lchild,x);if(T->rchild)get_deep(T->rchild,x);} int get_depth(bitree T){ if(!T)return 0;else { m=get_depth(T->lchild);n=get_depth(T->rchild);return(m>n?m:n)+1;} } 2.3 编写程序,实现二叉树的先序,中序,后序遍历,并求其深度。#include inorder(bt->lchild);printf(“%c”,bt->data);inorder(bt->rchild);} } void postorder(blink bt){ if(bt){ postorder(bt->lchild);postorder(bt->rchild);printf(“%c”,bt->data);} } int max(int x,int y){ if(x>y)return x;else return y;} int depth(blink bt){ if(bt)return 1+max(depth(bt->lchild),depth(bt->rchild));else return 0;} void main(){ blink root;root=creat();printf(“n”);printf(“按先序排列:”);preorder(root);printf(“n”);printf(“按中序排列:”);inorder(root);printf(“n”);printf(“按后序排列:”);postorder(root);printf(“n”);printf(“此二叉数的深度是:”);printf(“depth=%dn”,depth(root));} 3.实验心得体会 通过本章学习实验,对树有了初步的认识。树就是一种非线性的数据结构,描述了客观世界中事物之间的层次关系。这种结构有着广泛的应用,一切具有层次关系的问题都可以用树来表示。 #include bt=(Treenode *)malloc(len); bt->data=n;bt->ltag=0;bt->rtag=0; bt->lchild=creat(); bt->rchild=creat();} else bt=null;return bt;} void preorder1(Treenode *t){ if(t!=null){ printf(“%4d”,t->data); preorder1(t->lchild); preorder1(t->rchild);} } void preorder2(Treenode *t){ if(t!=null){ preorder2(t->lchild); printf(“%4d”,t->data); preorder2(t->rchild);} } void preorder3(Treenode *t){ if(t!=null){ preorder3(t->lchild); preorder3(t->rchild); printf(“%4d”,t->data);} } void InThread(Treenode *T){ Treenode *p;p=T;if(p){ InThread(p->lchild);if(!p->lchild){ p->ltag=1;p->lchild=pre;} if(!pre->rchild){ pre->rtag=1;pre->rchild=p;} pre=p;InThread(p->rchild);} } Treenode *inorderthreading(Treenode *T){ Treenode *Thre;Thre=(Treenode *)malloc(sizeof(Treenode)Thre->lchild=T;Thre->rchild=Thre;pre=Thre;InThread(T);pre->rtag=1;pre->rchild=Thre;Thre->rchild=pre;return Thre;} void InThrTravel(Treenode *Thre){ Treenode *p;p=Thre->lchild;while(p!=Thre){ while(p->ltag==0)p=p->lchild;printf(“%4d”,p->data);while(p->rtag==1 && p->rchild!=Thre){ p=p->rchild;printf(“%4d”,p->data);} p=p->rchild;}printf(“n”);} void main(){ Treenode *tree;int i;printf(“建立二叉树:n”);tree=creat();printf(“请选择遍历二叉树的方法:nn1.先根序遍历.n2.中根序遍历.n3.后根序遍历.n”);scanf(“%d”,&i);while(i>3 && i<1){ scanf(“%d”,&i);} switch(i){ case 1:preorder1(tree);break;case 2:preorder2(tree);break;case 3:preorder3(tree);break;} printf(“n中根序线索遍历:n”);tree=inorderthreading(tree);InThrTravel(tree) 实验四 #include tree[i].parent=0; tree[i].llink=0; tree[i].rlink=0;} } void inputweight(node tree[]){ int i;printf(“请输入各结点的权值:n”);for(i=1;i<=n;i++){ printf(“第%d个结点的权值:”,i); scanf(“%d”,&tree[i].weight);} } void select(int pre,int *min1,int *min2){ int i;*min1=*min2 = 0;for(i=1;i select(i-1,&x1,&x2); tree[x1].parent=i; tree[x2].parent=i; tree[i].llink=x1; tree[i].rlink=x2; tree[i].weight=tree[x1].weight+tree[x2].weight;} void encode(node tree[],element table[]){ int i,s,f;codetype c;for(i=1;i<=n;i++){ table[i].c=tree[i].weight; c.start=n+1;s=i; while(f=tree[s].parent) { c.bits[--c.start]=(s==tree[f].llink)?'0':'1';s=f;} table[i].code=c;} void main(){ int i;element table[n+1];sethuftree(tree);encode(tree,table);printf(“n输出Huffman编码n”);for(i=1;i<=n;i++){ printf(“n %c :”,table[i].c);puts(table[i].code.bits);} } 实验五 #include for(j=0;j if(i==j)g->edges[i][j]=0; else g->edges[i][j]=MAX; for(k=0;k printf(“Input i,j,w:(eg.0,1,5)n”); scanf(“%d,%d,%d”,&i,&j,&w); g->edges[i][j]=w; printf(“nThe graph is: nn”); printf(“t ”); for(k=0;k printf(“%3c”,g->vexs[k]); if(k==g->n-1)printf(“nn”); } for(i=0;i printf(“t%c ”,g->vexs[i]); for(j=0;j if(g->edges[i][j]==MAX)printf(“%3c”,MAX-923); else printf(“%3d”,g->edges[i][j]); printf(“n”); } void main(){ MGraph *g;g=(MGraph *)malloc(sizeof(MGraph));createmgraph(g);getchar();} 实验六 1.实验要求 1.1 熟悉图的各种存储方法。 1.2 掌握遍历图的递归和非递归的算法。1.3 理解图的有关算法。 2.实验内容 2.1 写出将一个无向图的邻接矩阵转换成邻接表的算法。void mattolist(int a[][],adjlist b[],int n) { for(i=0;i for(i=0;i for(j=n-1;j>=0;j--) if(a[i][j]!=0) {p=(arcnodetp *)malloc(sizeof(arcnodetp)); p->adjvex=j; p->nextare=b[i].firstare; b[i].firstarc=p; } } 2.2 以邻接表作存储结构,给出拓扑排序算法的实现。typedef struct vexnode { VertexType vertex;int in;ArecNodeTp * fristarc;}AdjList[vnum];typedef struct graph { AdjList adjlist;int vexnum,arcnum;}GraphTp;Top_Sort(GraphTp g){ LstackTp *p;int m,i,v;initStack(S);for(i=0;i if(g.adjlist[i].in==0)/*if(w的入度==0)*/ push(S,&v);/*w入S栈*/ } } m=0;whlie(!EmptyStack(S)){ Pop(S,&v)//S出栈->v printf(“%d”,v);/*输出v*/ m++;p=g.adjlist[i].fristarc;/*p=图g中顶点v的第一个邻接点*/ while(p!=NULL){//p存在 (g.adjlist[p->adjvex].in)--;/*p的入度--*/ if(g.adjlist[p->adjvex].in==0)/*if(p的入度==0)*/ Push(S,p->adjvex);/*p入S栈*/ p=p->nextarc;/*p=图g中的顶点v的下一个邻接点*/ } } if(m 实验七 #include if(t->right==NULL) { p=(dnode *)malloc(sizeof(dnode)); p->data=c; p->left=NULL;p->right=NULL; t->right=p; } else sort(t->right,c);if(c if(t->left==NULL) { p=(dnode *)malloc(sizeof(dnode)); p->data=c;p->left=NULL;p->right=NULL;t->left=p; } else sort(t->left,c)dnode *creat(){ dnode *ht;int x;ht=(dnode *)malloc(sizeof(dnode));printf(“创建二叉排序树(以0结束输入):”);scanf(“%d”,&x);ht->data=x;n++;ht->left=NULL;ht->right=NULL;scanf(“%d”,&x);while(x>0){ sort(ht,x);n++; scanf(“%d”,&x);} return ht;} int find(dnode *b,int x,dnode *a[]){ dnode *stack[maxsize],*p;int top;a[1]=NULL; if(b!=NULL){ top=1;stack[top]=b;while(top>0){ p=stack[top];top--;if(p->left->data==x || p->right->data==x)a[1]=p;if(p->data==x){a[0]=p;return 1;} if(p->right!=NULL){ top++;stack[top]=p->right;} if(p->left!=NULL){ top++;stack[top]=p->left;} } } a[0]=p;} dnode *delet(dnode *t){ dnode *p,*q,*s,*f,*a[2];int flag=0,x;p=(dnode *)malloc(sizeof(dnode));f=(dnode *)malloc(sizeof(dnode));printf(“请输入要删除的结点:”);scanf(“%d”,&x);find(t,x,a);p=a[0];f=a[1];if(p==NULL){printf(“NO FIND!n”);exit(0);} if(p->left==NULL)s=p->right;else if(t->right=NULL)s=p->left;else{ q=p;s=p->left;while(s->right!=NULL) {q=s;s=s->right;} if(q==p)q->left=s->left; else q->right=s->left;p->data=s->data;free(s);flag=1;if(flag==0){ if(f==NULL)t=s;else if(f->left==p)f->left=s; else f->right=s;} return t;} void inorder(dnode *t){ if(t!=NULL){ inorder(t->left);printf(“%4d”,t->data);inorder(t->right);} } void main(){ dnode *h;h=creat();printf(“中序遍历二叉排序树:”);inorder(h);printf(“n”);h=delet(h); inorder(h);printf(“n”);} 实验八 1.实验要求 1.1 掌握顺序查找、二分法查找、分块查找和哈希表查找的算法。1.2 能运用线性表的查找方法解决实际问题。2.实验内容 2.1 编写一个算法,利用二分查找算法在一个有序表中插入一个元素X,并保持表的有序性。 #include printf(“%3d”,*(data+insert));getch();} /*********************************************************/ int input(int *data){ int i,m;printf(“nInput the max num:”);scanf(“%d”,&m);printf(“input datan”);for(i=1;i<=m;i++) scanf(“%d”,data+i);return m;} /**********************************************************/ int search(int *data,int low,int high)/*递归查找插入位置*/ { int mid;if(low>high)return low;/*没有找到插入数据,返回low*/ else{ mid=(low+high)/2; if(*(data+mid)==*data)retun mid;/*找到插入数据,返回mid*/ else if(*(data+mid)<*data) else if(*()data+mid)>*data)} search(data,low,high);} /**********************************************************/ void plug(int *data,int insert,int m){ int i;for(i=m;i>insert;i--) *(data+i+1)=*(data+i);*(data+insert)=*data } 2.2 根据给定的数据表,先建立索引表,然后进行分块查找。 #include (p+k)->key=dat a[m*k]; (p+k)->addr=m*k; for(j=m*k;j if(data[j]>(p+k)->key) (p+k)->key=data[j];/*块的最大关键字*/ } return p;} int BlockSearch(index *list,int rectab[],int n,int m,int k)/*分块查找*/ { int low=0,high=m-1,mid,i;int b=n/m;/*每块有b个元素*/ while(low<=high){/*块间折半查找*/ mid=(low+high)/2; if((list+mid)->key>=k) high=mid+1; else low=mid+1;} if(low for(i=(list+low)->addr;i<=(list+low)->adder+b-1&&rectab[i]!=k;i++); if(i<=(list+low)->addr+b-1) return i; else return-1;} return-1;} void main(){ int record[N]={22,12,13,8,9,20,33,42,44,38,24,48,60,58,74,49,86,53};int key;index *list;printf(“please input key:n”);scanf(“%d”,&key);list=CreateList(record,N);printf(“data postion id %dn”,BlockSearch(list,record,N,BlockNum,key));} 3.实验心得体会 通过本章的学习,对排序有较高层次的理解与认识,从平时的练习中可以看出排序是数据处理中经常用到的重要运算。有序的顺序表可以采用查找效率较高的折半查找法,而无序的顺序表只能用效率较低的顺序查找法。 注意:实验结束后提交一份实验报告电子文档 电子文档命名为“学号+姓名”,如:E01214058宋思怡 《数据结构》实验报告 (一)学号:姓名:专业年级: 实验名称:线性表 实验日期:2014年4月14日 实验目的: 1、熟悉线性表的定义及其顺序和链式存储结构; 2、熟练掌握线性表在顺序存储结构上实现基本操作的方法; 3、熟练掌握在各种链表结构中实现线性表基本操作的方法; 4、掌握用 C/C++语言调试程序的基本方法。 实验内容: 一、编写程序实现顺序表的各种基本运算,并在此基础上设计一个主程序完成如下功能: (1)初始化顺序表L; (2)依次在L尾部插入元素-1,21,13,24,8; (3)输出顺序表L; (4)输出顺序表L长度; (5)判断顺序表L是否为空; (6)输出顺序表L的第3个元素; (7)输出元素24的位置; (8)在L的第4个元素前插入元素0; (9)输出顺序表L; (10)删除L的第5个元素; (11)输出顺序表L。 源代码 调试分析(给出运行结果界面) 二、编写程序实现单链表的各种基本运算,并在此基础上设计一个主程序完成如下功能: „„„„ „„„„ 小结或讨论: (1)实验中遇到的问题和解决方法 (2)实验中没有解决的问题 (3)体会和提高 南京信息工程大学实验(实习)报告 实验(实习)名称数据结构实验(实习)日期 2011-11-2得分指导教师周素萍 系公共管理系专业信息管理与信息系统年级10级班次1姓名常玲学号2010230700 3实验一顺序表的基本操作及C语言实现 【实验目的】 1、顺序表的基本操作及 C 语言实现 【实验要求】 1、用 C 语言建立自己的线性表结构的程序库,实现顺序表的基本操作。 2、对线性表表示的集合,集合数据由用户从键盘输入(数据类型为整型),建立相应的顺序表,且使得数据按从小到大的顺序存放,将两个集合的并的结果存储在一个新的线性表集合中,并输出。 【实验内容】 1、根据教材定义的顺序表机构,用 C 语言实现顺序表结构的创建、插入、删除、查找等操作; 2、利用上述顺序表操作实现如下程序:建立两个顺序表表示的集合(集合中无重 复的元素),并求这样的两个集合的并。 【实验结果】 [实验数据、结果、遇到的问题及解决] 一. Status InsertOrderList(SqList &va,ElemType x) { } 二. Status DeleteK(SqList &a,int i,int k) {//在非递减的顺序表va中插入元素x并使其仍成为顺序表的算法 int i;if(va.length==va.listsize)return(OVERFLOW);for(i=va.length;i>0,x } //注意i的编号从0开始 int j;if(i<0||i>a.length-1||k<0||k>a.length-i)return INFEASIBLE;for(j=0;j<=k;j++)a.elem[j+i]=a.elem[j+i+k];a.length=a.length-k;return OK; 三.// 将合并逆置后的结果放在C表中,并删除B表 Status ListMergeOppose_L(LinkList &A,LinkList &B,LinkList &C) { LinkList pa,pb,qa,qb;pa=A;pb=B;qa=pa;qb=pb;// 保存pa的前驱指针 // 保存pb的前驱指针 pa=pa->next;pb=pb->next;A->next=NULL;C=A;while(pa&&pb){} while(pa){} qa=pa;pa=pa->next;qa->next=A->next;A->next=qa;if(pa->data data){} else{} qb=pb;pb=pb->next;qb->next=A->next;//将当前最小结点插入A表表头 A->next=qb;qa=pa;pa=pa->next;qa->next=A->next;//将当前最小结点插入A表表头 A->next=qa; } } pb=B;free(pb);return OK;qb=pb;pb=pb->next;qb->next=A->next;A->next=qb; 顺序表就是把线性表的元素存储在数组中,元素之间的关系直接通过相邻元素的位置来表达。 优点:简单,数据元素的提取速度快; 缺点:(1)静态存储,无法预知问题规模的大小,可能空间不足,或浪费存储空间;(2)插入元素和删除元素时间复杂度高——O(n) 求两个集合的并集 该算法是求两个集合s1和s2的并集,并将结果存入s引用参数所表示的集合中带回。首先把s1集合复制到s中,然后把s2中的每个元素依次插入到集合s中,当然重复的元素不应该被插入,最后在s中就得到了s1和s2的并集,也就是在s所对应的实际参数集合中得到并集。 数据结构实验报告 一. 题目要求 1)编程实现二叉排序树,包括生成、插入,删除; 2)对二叉排序树进行先根、中根、和后根非递归遍历; 3)每次对树的修改操作和遍历操作的显示结果都需要在屏幕上用树的形状表示出来。4)分别用二叉排序树和数组去存储一个班(50人以上)的成员信息(至少包括学号、姓名、成绩3项),对比查找效率,并说明在什么情况下二叉排序树效率高,为什么? 二. 解决方案 对于前三个题目要求,我们用一个程序实现代码如下 #include typedefintElemType; //数据类型 typedefint Status; //返回值类型 //定义二叉树结构 typedefstructBiTNode{ ElemType data; structBiTNode *lChild, *rChild;//左右子树域 }BiTNode, *BiTree;intInsertBST(BiTree&T,int key){//插入二叉树函数 if(T==NULL){ T =(BiTree)malloc(sizeof(BiTNode)); T->data=key; T->lChild=T->rChild=NULL; return 1;} else if(key InsertBST(T->rChild,key);} else return 0;} BiTreeCreateBST(int a[],int n){//创建二叉树函数 BiTreebst=NULL;inti=0;while(i //数据域 InsertBST(bst,a[i]); i++;} returnbst;} int Delete(BiTree&T) { BiTreeq,s; } if(!(T)->rChild){ //右子树为空重接它的左子树 q=T;T=(T)->lChild;free(q);}else{ if(!(T)->lChild){ //若左子树空则重新接它的右子树 q=T;T=(T)->rChild;}else{ q=T;s=(T)->lChild;while(s->rChild){ q=s;s=s->rChild;} (T)->data=s->data;//s指向被删除结点的前驱 if(q!=T) q->rChild=s->lChild; else q->lChild=s->lChild; free(s);} } return 1; //删除函数,在T中删除key元素 intDeleteBST(BiTree&T,int key){ if(!T)return 0;else{ if(key==(T)->data)return Delete(T); else{ if(key<(T)->data) returnDeleteBST(T->lChild,key); else returnDeleteBST(T->rChild,key); } } } intPosttreeDepth(BiTree T){//求深度 inthr,hl,max;if(!T==NULL){ hl=PosttreeDepth(T->lChild);hr=PosttreeDepth(T->rChild);max=hl>hr?hl:hr;return max+1;} else return 0; } void printtree(BiTreeT,intnlayer){//打印二叉树 if(T==NULL)return;printtree(T->rChild,nlayer+1);for(inti=0;i ”);} printf(“%dn”,T->data);printtree(T->lChild,nlayer+1);} void PreOrderNoRec(BiTree root)//先序非递归遍历 { BiTree p=root;BiTreestack[50];intnum=0;while(NULL!=p||num>0){ while(NULL!=p) { printf(“%d ”,p->data); stack[num++]=p; p=p->lChild; } num--; p=stack[num]; p=p->rChild;} printf(“n”);} void InOrderNoRec(BiTree root)//中序非递归遍历 { BiTree p=root; } intnum=0;BiTreestack[50];while(NULL!=p||num>0){ while(NULL!=p){ stack[num++]=p; p=p->lChild;} num--;p=stack[num];printf(“%d ”,p->data);p=p->rChild;} printf(“n”);void PostOrderNoRec(BiTree root)//后序非递归遍历 { BiTree p=root;BiTreestack[50];intnum=0;BiTreehave_visited=NULL; while(NULL!=p||num>0){ while(NULL!=p) { stack[num++]=p; p=p->lChild; } p=stack[num-1]; if(NULL==p->rChild||have_visited==p->rChild) { printf(“%d ”,p->data); num--; have_visited=p; p=NULL; } else { p=p->rChild; } } printf(“n”);} int main(){//主函数 printf(“ ---------------------二叉排序树的实现-------------------”);printf(“n”);int layer;inti;intnum;printf(“输入节点个数:”);scanf(“%d”,&num);printf(“依次输入这些整数(要不相等)”);int *arr=(int*)malloc(num*sizeof(int));for(i=0;i scanf(“%d”,arr+i);} BiTreebst=CreateBST(arr,num);printf(“n”);printf(“二叉树创建成功!”);printf(“n”);layer=PosttreeDepth(bst);printf(“树状图为:n”);printtree(bst,layer);int j;int T;int K;for(;;){ loop: printf(“n”);printf(“ ***********************按提示输入操作符************************:”);printf(“n”);printf(“ 1:插入节点 2:删除节点 3:打印二叉树 4:非递归遍历二叉树 5:退出”);scanf(“%d”,&j); switch(j){ case 1: printf(“输入要插入的节点:”); scanf(“%d”,&T); InsertBST(bst,T); printf(“插入成功!”);printf(“树状图为:n”); printtree(bst,layer); break; case 2: } printf(“输入要删除的节点”);scanf(“%d”,&K);DeleteBST(bst,K);printf(“删除成功!”);printf(“树状图为:n”);printtree(bst,layer);break;case 3: layer=PosttreeDepth(bst);printtree(bst,layer);break;case 4: printf(“非递归遍历二叉树”);printf(“先序遍历:n”);PreOrderNoRec(bst);printf(“中序遍历:n”);InOrderNoRec(bst); printf(“后序遍历:n”); PostOrderNoRec(bst); printf(“树状图为:n”); printtree(bst,layer); break;case 5: printf(“程序执行完毕!”); return 0;} goto loop;} return 0;对于第四小问,要储存学生的三个信息,需要把上面程序修改一下,二叉树结构变为 typedefintElemType; //数据类型 typedefstring SlemType; typedefint Status; //返回值类型 //定义二叉树结构 typedefstructBiTNode{ SlemType name;ElemType score;ElemType no; //数据域 structBiTNode *lChild, *rChild;//左右子树域 }BiTNode, *BiTree;参数不是key,而是另外三个 intInsertBST(BiTree&T,intno,intscore,string name){//插入二叉树函数 if(T==NULL){ T =(BiTree)malloc(sizeof(BiTNode)); T->no=no;T->name=name;T->score=score; T->lChild=T->rChild=NULL; return 1;} else if(no InsertBST(T->rChild,no,score,name);} else return 0;} 其他含参函数也类似 即可完成50个信息存储 用数组存储50个信息,查看以往代码 #include int main(){ cout<<“ 欢迎来到学生管理系统”< cout<<“该学号信息已经存在,添加失败”< break;} cout<<“重新输入添加的学号”< for(int n=m+1;n<20;n++){ if(ptr[m].average() student a; a=ptr[m]; ptr[m]=ptr[n]; ptr[n]=a; }} ptr[m].show();} break;case 4: cout<<“谢谢使用”< 二叉排序树储存数据界面(储存学生信息略) 创建二叉树: 插入节点: 删除节点: 非递归遍历: 退出: 数组储存学生信息界面 分析查找效率: 因为二叉树查找要创建二叉树,而数组查找只创建一个数组,二叉树的创建时间比较长,所以对于数据量较少的情况下数组的查找效率比较高。但当数据量增加时,二叉树的查找优势就显现出来。所以数据量越大的时候,二叉树的查找效率越高。 四. 总结与改进 这个实验工作量还是很大的,做了很久。树状图形输出还是不美观,还需要改进。 一开始打算用栈实现非递归,但是根据书里面的伪代码发现部分是在C++编译器里运行不了的(即使补充了头文件和数据的定义),所以之后参考了网上的数组非递归,发现其功能和栈相似。 递归遍历的实现比非递归的遍历真的简单很多。 开始时只看到前三问,所以没有写到储存学生数据的代码,里面还可以用clock()函数加一个计算查找所要数据时间的代码,让二叉树查找与数组查找到效率比较更加直观。 实验报告4 排序 一、实验目的 1、掌握常用的排序方法,并掌握用高级语言实现排序算法的方法。 2、深刻理解排序的定义和各种排序方法的特点,并能加以灵活应用。 3、了解各种方法的排序过程及其依据的原则,并掌握各种排序方法的时间复杂度的分析方法。 二、实验要求及内容 要求编写的程序所能实现的功能包括: 1、从键盘输入要排序的一组元素的总个数 2、从键盘依次输入要排序的元素值 3、对输入的元素进行快速排序 4、对输入的元素进行折半插入排序 三、实验代码及相关注释 #include typedef struct { int key;}RedType; typedef struct { RedType r[100];int length;}SqList; //1 快速排序的结构体 typedef struct { int data[100]; int last;}Sequenlist;//2 折半插入排序的结构体 int Partition(SqList &L, int low, int high) //1 寻找基准 { L.r[0]=L.r[low];//子表的第一个记录作基准对象 int pivotkey = L.r[low].key;//基准对象关键字 while(low while(low L.r[low] = L.r[high];//小于基准对象的移到区间的左侧 while(low L.r[high] = L.r[low];//大于基准对象的移到区间的右侧 } L.r[low] = L.r[0];return low;} void QuickSort(SqList &L, int low, int high) //1 快速排序 { //在序列low-high中递归地进行快速排序 if(low < high) { int pivotloc= Partition(L, low, high); //寻找基准 QuickSort(L, low, pivotloc-1);//对左序列同样递归处理 QuickSort(L, pivotloc+1, high);//对右序列同样递归处理 } } Sequenlist *Sqlset() //2 输入要折半插入排序的一组元素 { Sequenlist *L; int i; L=(Sequenlist *)malloc(sizeof(Sequenlist)); L->last=0; cout<<“请输入要排序的所有元素的总个数:”; cin>>i; cout< cout<<“请依次输入所有元素的值:”; if(i>0) { for(L->last=1;L->last<=i;L->last++) cin>>L->data[L->last]; L->last--; } return(L);} middlesort(Sequenlist *L) //2 折半插入排序 { int i,j,low,high,mid;for(i=1;i<=L->last;i++){ L->data[0]=L->data[i]; low=1; high=i-1; while(low<=high) { mid=(low+high)/2; if(L->data[0] high=mid-1;//插入点在前半区 else low=mid+1;//插入点在后半区 } for(j=i;j>high+1;j--){ L->data[j]=L->data[j-1];} //后移 L->data[high+1]=L->data[0];//插入 } return 0;} int main(){ gg: cout<<“请选择功能(1.快速排序 2.折半插入排序 3.退出程序):”;int m;cin>>m;cout< if(m==1){ SqList L;int n;cout<<“请输入要排序的所有元素的总个数:”;cin>>n;cout< cin>>L.r[i].key; } cout< QuickSort(L,1,L.length); for(int j=1;j<=L.length;j++) { cout< } cout< cout< } if(m==2){ Sequenlist *L; int i; L=Sqlset(); cout< middlesort(L); cout<<“折半插入排序后为:”; for(i=1;i<=L->last;i++) { cout< } cout< cout< goto gg;} if(m==3){ exit(0); cout< 四、重要函数功能说明 1、Sequenlist *Sqlset() 输入要折半插入排序的一组元素 2、int Partition(SqList &L, int low, int high) 寻找快速排序的基准 3、void QuickSort(SqList &L, int low, int high) 快速排序 4、middlesort(Sequenlist *L) 折半插入排序 五、程序运行结果 下图仅为分别排序一次,可多次排序,后面有相关截图: 六、实验中遇到的问题、解决及体会 1、起初编写快速排序的程序时,我是完全按照老师PPT上的算法敲上去的,然后建立了一个SqList的结构体,调试运行时出现错误,仔细查看才意识到Partition函数中L中应该包含元素key,而我建立结构体时没有注意,然后我将key这个元素补充进去,继续调试,又出现错误,提示我Partition没有定义,我就觉得很奇怪,我明明已经写了函数定义,为什么会这样,当我又回过头来阅读程序时,我发现QuickSort函数中调用了Partition函数,但是我的Partition函数的定义在QuickSort函数的后面,于是我将Partition函数放到了QuickSort函数的前面,再次调试运行,就可以正常运行,得出结果了。这让我懂得,编程一定要认真仔细,不可大意马虎,否则又会花很多时间回过头来检查修改程序,得不偿失。 运行程序错误截图: 2、本来我是编写了两个程序,分别实现快速排序和折半插入排序的功能,但我后来想我是否可以将其合二为一,于是我想到用if选择语句用来实现不同的功能,从键盘输入功能选项m,if(m==1),可以进行快速排序,if(m==2),可以进行折半插入排序,于是我继续思考,我是否可以在一次运行程序中,多次对含有不同元素的序列进行排序,于是我用了goto语句,每次排序一次后,自动循环到选择语句,当不需要在排序的时候,可以从键盘输入3,退出程序,这样一来,程序变得更加实用和清晰明朗。这让我懂得,想要编出好的程序,要善于思考,在实现所需功能的前提下,多想问题,看是否能使程序更加实用简便。 修改程序前两个运行结果截图 (两个程序,调试运行两次,每次只能进行一次排序) 1、快速排序程序运行结果截图: 2、折半插入排序程序结果截图: 程序重要模块修改截图: 修改程序后运行截图: (一个程序,调试运行一次,可多次进行不同序列的不同排序)第二篇:数据结构实验报告
第三篇:数据结构实验报告
第四篇:数据结构实验报告
第五篇:数据结构实验报告