k个一组翻转链表js

k个一组翻转链表js_第1张图片

输入:head = [1,2,3,4,5], k = 2
输出:[2,1,4,3,5]

如果我设法把前 2 个节点反转,那么后面的那些节点怎么处理?后面的这些节点也是一条链表,而且规模(长度)比原来这条链表小,这就叫子问题

因为子问题和原问题的结构完全相同,这就是所谓的递归性质。

k个一组翻转链表js_第2张图片

输入:head = [1,2,3,4,5], k = 3
输出:[3,2,1,4,5]

思路:

有很多组个k需要反转,例如图1,有两组k个值需要反转;图2有一组k个值需要反转。

设置每组头节点为a,第k个为b

那我们不如设置一个for循环,从每组的a开始,到第k个节点结束,完成每一个区间的分别反转。这样有很多个分开的 被反转的组了。他们是单独的,但我们需要链表连接到一起,所以还有一个连接的过程。所以a.next连接到下一组反转之后的头结点中。至此,完成反转。形成一条反转并连接成功的链表。

那么在反转时需要每次调用反转方法:

设置3个指针,pre指针初始为null,cur与nxt初始等于a(每一组的头节点)

反转操作:另nxt等于cur.next,在让cur.next指向pre(指针反转),pre向后移动一位等于cur,cur向后移动一位等于nxt。

/**
 * Definition for singly-linked list.
 * function ListNode(val, next) {
 *     this.val = (val===undefined ? 0 : val)
 *     this.next = (next===undefined ? null : next)
 * }
 */
/**
 * @param {ListNode} head
 * @param {number} k
 * @return {ListNode}
 */

var reverseKGroup = function (head, k) {
    var a = head, b = head
    for (let i = 0; i < k; i++) {
        if (b == null) return head  //不足k个剩下的不需要反转
        b = b.next    //分组:每一组k个被反转的节点
    }
    var newHead = reverse(a, b)  //调用反转函数
    a.next = reverseKGroup(b, k)  //将每组被反转的组,连接到一起
    return newHead  //返回最后的链表
};
function reverse(a, b) {
    var pre = null, cur = a, nxt = a
    while (cur != b) {
        nxt = cur.next
        cur.next = pre
        pre = cur
        cur = nxt
    }
    return pre
}

 

你可能感兴趣的:(链表,数据结构,javascript)