考虑到可能要移除链表头结点,可以直接在链表上操作,但是这里需要作为一种情况单独考虑;也可以在链表头结点之前再添加一个节点,这样移除头结点也可以有像移除其他节点一样的操作,不需要特判
直接在链表上操作
var removeElements = function(head, val) {
while(head && head.val==val) head=head.next
if(!head) return null
let cur = head.next
let pre = head
while(cur){
if(cur.val==val){
pre.next=cur.next
}else pre = cur
cur=cur.next
}
return head
};
添加一个新的结点
function ListNode(val, next) {
this.val = (val===undefined ? 0 : val)
this.next = (next===undefined ? null : next)
}
var removeElements = function(head, val) {
let node = new ListNode(-1, head)
let cur = node
while(node.next){
if(node.next.val === val){
node.next = node.next.next
}else node = node.next
}
return cur.next
};
var reverseList = function(head) {
let pre = null
let cur = head
let next
while(cur){
next = cur.next
cur.next = pre
pre = cur
cur = next
}
return pre
};
- new 一个虚拟的头结点
node 指向 head,p1 ,p2 为当前需要交换的两个节点,next 作为除去当前交换节点的第一个节点 - while 循环遍历链表,循环结束的条件是
p1 为空或者 p2 为空 p1 为空对应偶数 个节点的链表操作结束,p2 为空对应奇数 个节点的链表操作结束- while循环体:
- 先将后续链表节点用next 存下来
- pre 的下一个指向 p2
- p2 的下一个指向 p1
- p1 的下一个指向 next
- 到这里两个节点的交换就完成了 ,从 pre -> p1 -> p2 -> next 到 pre -> p2 -> p1 -> next,接下来就是后移节点继续定位到下一对需要交换的节点了
- pre 指向 p1
- p1 指向 next,若发现p1 为空,则证明偶数个节点的链表已经交换结束,break 出去
- p2 指向next 的next,若发现p1 为空,则证明奇数个节点的链表已经交换结束
- 最终返回 node.next
function ListNode(val, next) {
this.val = (val===undefined ? 0 : val)
this.next = (next===undefined ? null : next)
}
var swapPairs = function(head) {
if(!head || !head.next) return head
let node = new ListNode(-1,head)
let pre = node
let p1 = node.next
let p2 = p1.next
let next = null
while(p2){
next = p2.next
pre.next = p2
p2.next = p1
p1.next = next
pre = p1
p1 = next
if(p1 === null) break
p2 = next.next
}
return node.next
};
这题可以先找到倒数第 n-1个节点,然后直接删掉它的下一个节点。因为有可能要删掉头结点,所以给这个链表添加了一个虚拟头结点 node。如何遍历一遍找到倒数第 n-1 个节点呢?使用双指针 p1,p2,p1初始化指向 node,p2初始化指向 node.next,p2往后走一步,n 就 --,当 n <= 0的时候 ,p1也往后移动,当 p2 指向null的时候,p1指向倒数第n-1个节点。
function ListNode(val, next) {
this.val = (val===undefined ? 0 : val)
this.next = (next===undefined ? null : next)
}
var removeNthFromEnd = function(head, n) {
if(!head) return head
let node = new ListNode(-1,head)
let p1 = node
let p2 = node.next
while(p2){
p2 = p2.next
if(n <= 0) p1 = p1.next
n --
}
console.log(p1.val)
p1.next = p1.next.next
return node.next
};
两个链表长度不一样如何确定交点起始位置呢?可以先让p1把headA走一遍,p2把headB走一遍,当p1走完时,再去走p2,当p2走完时再去走p1,第二趟如果两者相等,则表白此时出去公共部分的起始位置
- flagA,flagB 表示两个指针是否已经换过链表遍历了
- 循环的结束条件为 p1,p1都不为空,如果有一个为空都表示没有公共部分
- 当两个指针都已经换过链表遍历并且当前相等,则返回当前节点
- 当指针走到链表末尾,判断flagA,flagB是否为0,为0表示还没有换过链表,那就换另外一个链表遍历,不为0表示之前已经更换过了,直接跳出循环
var getIntersectionNode = function(headA, headB) {
let flagA = 0
let flagB = 0
let p1 = headA
let p2 = headB
while(p1&&p2){
if(flagA && flagB && p1 === p2){
return p1
}
if(p1.next){
p1 = p1.next
}
else{
if(flagA == 0){
flagA = 1
p1 = headB
}else break
}
if(p2.next){
p2 = p2.next
}else{
if(flagB == 0){
p2 = headA
flagB = 1
}else break
}
}
return null
};
第二遍做这个题了,还是没有根据快慢指针的路程得出那个需要的等式,又看了一遍题解的推导过程,希望我第三遍遇到这个题我能自己写出来吧,呜呜呜。
var detectCycle = function(head) {
if(!head || !head.next) return null
let slow = head
let fast = head
let p
while(slow && fast){
if(slow.next) slow = slow.next
else return null
if(fast.next && fast.next.next) fast = fast.next.next
else return null
if(slow === fast){
p = head
break
}
}
while(p && slow){
if(p === slow) return p
if(p && p.next) p = p.next
if(slow.next) slow = slow.next
}
return null
};
|