这里提供两种方法,一种是只交换对应的数据,另一种是通过更改指针来交换节点。而更改指针中又可以分为新建节点与不新建节点的方法。
1.不更改指针
这个没啥好说的,直接将对应的data交换即可。(这里的a,c节点都为被交换节点的上一个节点。因为不更改指针所以遍历到被交换节点即a->next和c->next也可以)
void List::swap(Node* a, Node* c)
{
int tmp;
tmp = a->next->data;
a->next->data = c->next->data;
c->next->data = tmp;
}
2.更改指针
更改指针又可以分为新建节点和不新建节点的方法。
①新建节点
新建节点去保存被交换节点的值,再将其插入到单链表中。相当于删去原先单链表中的两个节点,再将它们插入到对应位置。(这里的a,b节点都为被交换节点的上一个节点。但这里由于需要改变被交换节点的上一个节点的next指针,使next指向你新建的节点,那么就不能遍历到被交换节点。同时不要忘了你新建的节点的next需要指向被交换节点的下一个节点)
void List::swap(Node* a, Node* b)
{
Node* c = new Node();
Node* d = new Node();
c->data = a->next->data;
d->data = b->next->data;
d->next = a->next->next;
c->next = b->next->next;
a->next = d;
b->next = c;
}
②不新建节点
这个可能有点难理解,这里的a,c节点仍然为被交换节点的上一个节点。接下来我们来分析分析这段代码。
void List::swap(Node* a, Node* c)
{
Node* b = a->next;
Node* d = c->next;
a->next = d->next;
c->next = b->next;
b->next = a->next;
d->next = c->next;
a->next = d;
c->next = b;
}
我们需要保存两个被交换节点,同时需要改变其上一个节点的next指针等。那么新建b和d来保存被交换节点。
Node* b = a->next;
Node* d = c->next;
难点主要在这下面这块
a->next = d->next;
c->next = b->next;
b->next = a->next;
d->next = c->next;
a->next = d;
c->next = b;
这里拿图中这个单链表解释。假设我们需要交换第1个节点与第四个节点,括号内的字母为对应指针所在位置。
Head(a) |
11(b) |
22 |
33(c) |
44(d) |
55 |
NULL |
大致思路很简单,想办法把被交换节点b后面的节点(一直到c)接到d的后面,把d之后的节点(直到为NULL)接到b的后面。
那不如单独取出两个节点,将上面两部分的节点分别接到这两个节点之后,再拼接到被交换节点之后不就好了吗。这里用被交换节点的前一个节点来保存对应被交换节点之后的节点。
a->next = d->next;
c->next = b->next;
33 (c) |
22 (b->next) |
33 |
22 |
...(此时c被包含于b->next...->next中,所以循环) |
再将对应节点拼接到被交换节点后
b->next = a->next;
d->next = c->next;
head (a) |
11 (b) |
55 (a->next) |
44 (d) |
22 |
33 (c->next) |
22 |
...(22,33循环) |
最后合并,得到交换后的链表,完成
a->next = d;
c->next = b;
head |
44 |
22 |
33 (c->next) |
22 |
...(22,33循环) |
Head(a) |
11(b) |
22 |
33(c) |
44(d) |
55 |
NULL |