第二章
2.1开始结点是指链表中的第一个结点,也就是没有直接前趋的那个结点。
链表的头指针是一指向链表开始结点的指针(没有头结点时),单链表由头指针唯一确定,因此单链表可以用头指针的名字来命名。
头结点是在链表的开始结点之前附加的一个结点。
有了头结点之后,头指针指向头结点,不论链表否为空,头指针总是非空。
而且头指针的设置使得对链表的第一个位置上的操作与在表其他位置上的操作一致(都是在某一结点之后)。
2.3在等概率情况下,顺序表中插入一个结点需平均移动n/2个结点。
删除一个结点需平均移动(n-1)/2个结点。
具体的移动次数取决于顺序表的长度n以及需插入或删除的位置i。
i越接近n则所需移动的结点数越少。
2.4尾指针是指向终端结点的指针,用它来表示单循环链表可以使得查找链表的开始结点和终端结点都很方便,设一带头结点的单循环链表,其尾指针为rear,则开始结点和终端结点的位置分别是rear->next->next和rear,查找时间都是O
(1)。
若用头指针来表示该链表,则查找终端结点的时间为O(n)。
2.5下面分别讨论三种链表的情况。
1.单链表。
若指针p指向某结点时,能够根据该指针找到其直接后继,能够顺后继指针链找到*p结点后的结点。
但是由于不知道其头指针,所以无法访问到p指针指向的结点的直接前趋。
因此无法删去该结点。
2.双链表。
由于这样的链表提供双向指针,根据*p结点的前趋指针和后继指针可以查找到其直接前趋和直接后继,从而可以删除该结点。
其时间复杂度为O
(1)。
3.单循环链表。
根据已知结点位置,可以直接得到其后相邻的结点位置(直接后继),又因为是循环链表,所以我们可以通过查找,得到p结点的直接前趋。
因此可以删去p所指结点。
其时间复杂度应为O(n)。
2.6该算法的功能是:
将开始结点摘下链接到终端结点之后成为新的终端结点,而原来的第二个结点成为新的开始结点,返回新链表的头指针。
2.7算法如下:
#defineListSize100//假定表空间大小为100
typedefintDataType;//假定DataType的类型为int型
typedefstruct{
DataTypedata[ListSize];//向量data用于存放表结点
intlength;//当前的表长度
}Seqlist;
//以上为定义表结构
voidInsertList(Seqlist*L,Datatypex,inti)
{
//将新结点x插入L所指的顺序表的第i个结点ai的位置上,即插入的合法位置为:
0<=i<=L->length
intj;
if(i<0||i>L->length)
Error("positionerror");//非法位置,退出,该函数定义见教材P7.
if(L->length>=ListSize)
Error(“overflow");
for(j=L->length-1;j>=i;j--)
L->data[j+1]=L->data[j];
L->data[i]=x;
L->length++;
}
voidDeleteList(Seqlist*L,inti)
{//从L所指的顺序表中删除第i个结点ai,合法的删除位置为0<=i<=L->length-1
intj;
if(i<0||i>=L->length)
Error("positionerror");
for(j=i;jlength;j++)
L->data[j]=L->data[j+1];//结点前移
L->length--;//表长减小
}
2.8
1.顺序表:
要将该表逆置,可以将表中的开始结点与终端结点互换,第二个结点与倒数第二个结点互换,如此反复,就可将整个表逆置了。
算法如下:
//顺序表结构定义同上题
voidReverseList(Seqlist*L)
{
DataTypetemp;//设置临时空间用于存放data
inti;
for(i=0;i<=L->length/2;i++)//L->length/2为整除运算
{temp=L->data[i];//交换数据
L->data[i]=L->data[L->length-1-i];
L->data[L->length-1-i]=temp;
}
}
2.链表:
可以用交换数据的方式来达到逆置的目的。
但是由于是单链表,数据的存取不是随机的,因此算法效率太低。
可以利用指针改指来达到表逆置的目的。
具体情况入下:
(1)当链表为空表或只有一个结点时,该链表的逆置链表与原表相同。
(2)当链表含2个以上结点时,可将该链表处理成只含第一结点的带头结点链表和一个无头结点的包含该链表剩余结点的链表。
然后,将该无头结点链表中的所有结点顺着链表指针,由前往后将每个结点依次从无头结点链表中摘下,作为第一个结点插入到带头结点链表中。
这样就可以得到逆置的链表。
算法是这样的:
结点结构定义如下:
typedefcharDataType;//假设结点的数据域类型的字符
typedefstructnode{//结点类型定义
DataTypedata;//结点的数据域
structnode*next;//结点的指针域
}ListNode;
typedefListNode*LinkList;
ListNode*p;
LinkListhead;
LinkListReverseList(LinkListhead)
{//将head所指的单链表(带头结点)逆置
ListNode*p,*q;//设置两个临时指针变量
if(head->next&&head->next->next)
{ //当链表不是空表或单结点时
p=head->next;
q=p->next;
p->next=NULL;//将开始结点变成终端结点
while(q)
{//每次循环将后一个结点变成开始结点
p=q;
q=q->next;
p->next=head->next;
head->next=p;
}
returnhead;
}
returnhead;//如是空表或单结点表,直接返回head
}
2.9因已知顺序表L是递增有序表,所以只要从顺序表终端结点(设为i位置元素)开始向前寻找到第一个小于或等于x的元素位置i后插入该位置即可。
在寻找过程中,由于大于x的元素都应放在x之后,所以可边寻找,边后移元素,当找到第一个小于或等于x的元素位置i时,该位置也空出来了。
算法如下:
//顺序表存储结构如题2.7
voidInsertIncreaseList(Seqlist*L,Datatypex)
{
inti;
if(L->length>=ListSize)
Error(“overflow");
for(i=L->length;i>0&&L->data[i-1]>x;i--)
L->data[i]=L->data[i];//比较并移动元素
L->data[i]=x;
L->length++;
}
2.10与上题相类似,只要从终端结点开始往前找到第一个比x大(或相等)的结点数据,在这个位置插入就可以了。
(边寻找,边移动)算法如下:
voidInsertDecreaseList(Seqlist*L,Datatypex)
{
inti;
if(L->length>=ListSize)
Error(“overflow");
for(i=L->length;i>0&&L->data[i-1] L->data[i]=L->data[i];//比较并移动元素
L->data[i]=x;
L->length++;
}
2.11由于在单链表中只给出一个头指针,所以只能用遍历的方法来数单链表中的结点个数了。
算法如下:
intListLength(LinkListL)
{
intlen=0;
ListNode*p;
p=L;//设该表有头结点
while(p->next)
{
p=p->next;
len++;
}
returnlen;
}
2.12分析:
由于要进行的是两单链表的连接,所以应找到放在前面的那张表的表尾结点,再将后表的开始结点链接到前表的终端结点后即可。
该算法的主要时间消耗是用在寻找第一张表的终端尾结点上。
这两张单链表的连接顺序无要求,并且已知两表的表长,则为了提高算法效率,可选表长小的单链表在前的方式连接。
具体算法如下:
LinkListLink(LinkListL1,LinkListL2,intm,intn)
{//将两个单链表连接在一起
ListNode*p,*q,*s;
//s指向短表的头结点,q指向长表的开始结点,回收长表头结点空间
if(m<=n)
{s=L1;q=L2->next;free(L2);}
else{s=L2;q=L1->next;free(L1);}
p=s;
while(p->next)p=p->next;//查找短表终端结点
p->next=q;//将长表的开始结点链接在短表终端结点后
returns;
}
本算法的主要操作时间花费在查找短表的终端结点上,所以本算的法时间复杂度为:
O(min(m,n))
2.13根据已知条件,A和B是两个递增有序表,所以可以先取A表的表头建立空的C表。
然后同时扫描A表和B表,将两表中最大的结点从对应表中摘下,并作为开始结点插入C表中。
如此反复,直到A表或B表为空。
最后将不为空的A表或B表中的结点依次摘下并作为开始结点插入C表中。
这时,得到的C表就是由A表和B表归并成的一个按元素值递减有序的单链表C。
并且辅助空间为O
(1)。
算法如下:
LinkListMergeSort(LinkListA,LinkListB)
{//归并两个带头结点的递增有序表为一个带头结点递减有序表
ListNode*pa,*pb,*q,*C;
pa=A->next;//pa指向A表开始结点
C=A;C->next=NULL;//取A表的表头建立空的C表
pb=B->next;//pb指向B表开始结点
free(B);//回收B表的头结点空间
while(pa&&pb)
{
if(pb->data<=pa->data)
{//当B中的元素小于等于A中当前元素时,将pa表的开始结点摘下
q=pa;pa=pa->next;
}
else
{//当B中的元素大于A中当前元素时,将pb表的开始结点摘下
q=pb;pb=pb->next;}
q->next=C->next;C->next=q;//将摘下的结点q作为开始结点插入C表
}
//若pa表非空,则处理pa表
while(pa){
q=pa;pa=pa->next;
q->next=C->next;C->next=q;}
//若pb表非空,则处理pb表
while(pb){
q=pb;pa=pb->next;
q->next=C->next;C->next=q;}
return(C);
}
该算法的时间复杂度分析如下:
算法中有三个while循环,其中第二个和第三个循环只执行一个。
每个循环做的工作都是对链表中结点扫描处理。
整个算法完成后,A表和B表中的每个结点都被处理了一遍。
所以若A表和B表的表长分别是m和n,则该算法的时间复杂度O(m+n)
2.14要解这样的问题,我们首先想到的是拿链表中的元素一个个地与max和min比较,然后删除这个结点。
由于为已知其是有序链表,则介于min和max之间的结点必为连续的一段元素序列。
所以我们只要先找到所有大于min结点中的最小结点的直接前趋结点*p后,依次删除小于max的结点,直到第一个大于等于max结点*q位置,然后将*p结点的直接后继指针指向*q结点。
算法如下:
voidDeleteList(LinkListL,DataTypemin,DataTypemax)
{
ListNode*p,*q,*s;
p=L;
while(p->next&&p->next->data<=min)
//找比min大的前一个元素位置
p=p->next;
q=p->next;//p指向第一个不大于min结点的直接前趋,q指向第一个大于min的结点
while(q&&q->data {s=q;q=q->next;
free(s);//删除结点,释放空间
}
p->next=q;//将*p结点的直接后继指针指向*q结点
}
2.15本题可以这样考虑,先取开始结点中的值,将它与其后的所有结点值一一比较,发现相同的就删除掉,然后再取第二结点的值,重复上述过程直到最后一个结点。
具体算法:
voidDeleteList(LinkListL)
{
ListNode*p,*q,*s;
p=L-next;
while(p->next&&p->next->next)
{
q=p;//由于要做删除操作,所以q指针指向要删除元素的直接前趋
while(q->next)
if(p->data==q->next->data)
{s=q->next;q->next=s->next;free(s);//删除与*p的值相同的结点
}
elseq=q->next;
p=p->next;
}
}
第三章
3.