return(head);
}
182写一函数insert,用来向动态链表插入一结点
Student*insert(student*head,student*stud)
{
student*p0,*p1,*p2;
p1=head;
p0=stud;
if(head==NULL)
{
head=p0;
p0->next=NULL;
}
else
{
while((p0->num>p1->num)&&(p1->next!
=NULL))
{
p2=p1;
p1=p1->next;
}
if(p0->num<=p1->num)
{
if(head==p1)
head=p0;
else
p2->next=p0;
p0->next=p1;
}
else
{
p1->next=p0;
p0->next=NULL;
}
}
n=n+1;
return(head);
}
183链表题:
一个链表的结点结构
structNode
{
intdata;
Node*next;
};
typedefstructNodeNode;
(1)已知链表的头结点head,写一个函数把这个链表逆序(Intel)
Node*ReverseList(Node*head)//链表逆序
{
if(head==NULL||head->next==NULL)
returnhead;
Node*p1=head;
Node*p2=p1->next;
Node*p3=p2->next;
p1->next=NULL;
while(p3!
=NULL)
{
p2->next=p1;
p1=p2;
p2=p3;
p3=p3->next;
}
p2->next=p1;
head=p2;
returnhead;
}
(2)已知两个链表head1和head2各自有序,请把它们合并成一个链表依然有序。
(保留所有结点,即便大小相同)
Node*Merge(Node*head1,Node*head2)
{
if(head1==NULL)
returnhead2;
if(head2==NULL)
returnhead1;
Node*head=NULL;
Node*p1=NULL;
Node*p2=NULL;
if(head1->datadata)
{
head=head1;
p1=head1->next;
p2=head2;
}
else
{
head=head2;
p2=head2->next;
p1=head1;
}
Node*pcurrent=head;
while(p1!
=NULL&&p2!
=NULL)
{
if(p1->data<=p2->data)
{
pcurrent->next=p1;
pcurrent=p1;
p1=p1->next;
}
else
{
pcurrent->next=p2;
pcurrent=p2;
p2=p2->next;
}
}
if(p1!
=NULL)
pcurrent->next=p1;
if(p2!
=NULL)
pcurrent->next=p2;
returnhead;
}
(3)已知两个链表head1和head2各自有序,请把它们合并成一个链表依然有序,这次要求用递归方法进行。
(Autodesk)
答案:
Node*MergeRecursive(Node*head1,Node*head2)
{
if(head1==NULL)
returnhead2;
if(head2==NULL)
returnhead1;
Node*head=NULL;
if(head1->datadata)
{
head=head1;
head->next=MergeRecursive(head1->next,head2);
}
else
{
head=head2;
head->next=MergeRecursive(head1,head2->next);
}
returnhead;
}
184.利用链表实现将两个有序队列A和B合并到有序队列H中,不准增加其他空间。
请提供全一点的程序
以升序为例:
while(a!
=NULL&&b!
=NULL)
{
if(a->datadata)
{
h->data=a->data;
a=a->next;
}
elseif(a->data==b->data)
{
h->data=a->data;
a=a->next;
b=b->next;
}
else
{
h->data=b->data;
b=b->next
}
h=h->next;
}
if(a==NULL)
{
while(b!
=NULL)
{
h->data=b->data;
h=h->next;
b=b->next;
}
}
else
{
while(a!
=NULL)
{
h->data=a->next;
h=h->next;
a=a->next;
}
}
185单向链表的反转是一个经常被问到的一个面试题,也是一个非常基础的问题。
比如一个链表是这样的:
1->2->3->4->5通过反转后成为5->4->3->2->1。
最容易想到的方法遍历一遍链表,利用一个辅助指针,存储遍历过程中当前指针指向的下一个元素,然后将当前节点元素的指针反转后,利用已经存储的指针往后面继续遍历。
源代码如下:
structlinka{
intdata;
linka*next;
};
voidreverse(linka*&head)
{
if(head==NULL)
return;
linka*pre,*cur,*ne;
pre=head;
cur=head->next;
while(cur)
{
ne=cur->next;
cur->next=pre;
pre=cur;
cur=ne;
}
head->next=NULL;
head=pre;
}
还有一种利用递归的方法。
这种方法的基本思想是在反转当前节点之前先调用递归函数反转后续节点。
源代码如下。
不过这个方法有一个缺点,就是在反转后的最后一个结点会形成一个环,所以必须将函数的返回的节点的next域置为NULL。
因为要改变head指针,所以我用了引用。
算法的源代码如下:
linka*reverse(linka*p,linka*&head)
{
if(p==NULL||p->next==NULL)
{
head=p;
returnp;
}
else
{
linka*tmp=reverse(p->next,head);
tmp->next=p;
returnp;
}
}
186对如下双链表
typedefstruct_node
{
intiData;
struct_node*pPrev;
struct_node*pNext;
}node;
a.请写出代码,将node*n插入到node*p后。
b.如果多线程同时访问此链表,需要加锁,请说明以下步骤
(a)申请内存给n.
(b)N数据初始化。
(c)插入
注意加锁和解锁的时机。
node*insert(node*p,node*n)
{
if((p==NULL)||(n==NULL))
{
returnNULL;
}
if(p->pNext!
=NULL)
{
p->pNext->pPrev=n;
}
n->pPrev=p;
n->pNext=p->pNext;
p->pNext=n;
returnn;
}
187、试创建二叉数,并写出常见的几种遍历方式?
#include"stdio.h"
#include"string.h"
#include
#defineNULL0
typedefstructBiTNode{
chardata;
structBiTNode*lchild,*rchild;
}BiTNode,*BiTree;
BiTreeCreate(BiTreeT){
charch;
ch=getchar();
if(ch=='0')
T=NULL;
else{
if(!
(T=(BiTNode*)malloc(sizeof(BiTNode))))
printf("Error!
");
T->data=ch;
T->lchild=Create(T->lchild);
T->rchild=Create(T->rchild);
}
returnT;
}
voidPreorder(BiTreeT){
if(T){
printf("%c",T->data);
Preorder(T->lchild);
Preorder(T->rchild);
}
}//先序遍历
voidInorder(BiTreeT){
if(T){
Inorder(T->lchild);
printf("%c",T->data);
Inorder(T->rchild);
}
}//中序遍历
voidPostorder(BiTreeT){
if(T){
Postorder(T->lchild);
Postorder(T->rchild);
printf("%c",T->data);
}
}//后序遍历
188、前序遍历输入,如图所示,写出后序遍历输出结果?
例如二叉树:
输入序列ABD..EH...CF.I..G..
输出结果为:
?
答案:
输出结果为:
DHEBIFGCA