习题3.5 求链表的倒数第m个元素 (20 point(s))
原题链接
这道题不难。但我想记录一下这个比较聪明的算法。
下面给出accept代码。
ElementType Find(List L, int m) {
List p = L;
List q = L;
int len = 0;
/*while (p->Next != NULL) {
len++;
p = p->Next;
}
if (len == 0)
return ERROR;
int key = len - m +1;
p = L;
for (int i = 0; i < key; i++) {
if (i == key - 1) {
return p->Next->Data;
}
else {
p = p->Next;
}
}*///历遍2N-m+1次算法
while (p->Next != NULL) {
if (len != m) {
p = p->Next;
len++;
}
else {
p = p->Next;
q = q->Next;
}
}
if (len == 0)
return ERROR;
return q->Next->Data;//同时移动的算法,比较聪明
}
我当时用的第一种比较笨的算法,看到了书上那个同时移动的算法觉得好聪明,于是记录一下,我好菜啊
基本思路
- 先讲讲我的笨算法。很简单的思路,先历遍链表,得到链表长度(记作len),再历遍到len-m+1(m是倒数的位置)的位置,然后返回该位置的data值。该算法会进行2*len-m+1次扫描。
- 比较聪明的算法是代码中给出的第二种:
- 建立两个操作指针,都指向链表的头节点。(一个指针称为p,另一个指针称为q。)
- 声明一个距离计数器,称为len。
- p开始向前移动,len自加。
- 直至len等于m时,p与q开始共同移动。
- 当p移动到链表尾部,q结点的位置就是倒数第m个的位置,返回该点的数据。
- 记得,当表为空时,返回ERROR。需要特判空表。
心得
积累比较巧妙的算法。