个人解题思路:
先求出整个链表的表长,再除以2,得到链表的中间位置。
此时我就卡在了如何获取这个数组??(cai本cai
下面会提到,其实通过指针的移动来将数据存入到新建的一个数组里就可以了
借鉴他人解决方法:
方法一:数组法
由于链表无法用下标来获取数据,所以只能通过指针的移动来将数据存入到新建的一个数组A里;此时结束循环的条件是数组的最后一个数据的next指针
指向的数据为空。接着算出数组A的长度,即中间节点为A.length /2
,即获取到中间结点的值为 A[A.length/2]
若遇到长度不能被2整除,则需要用到Math.trunc()
方法
它是JavaScript数学库中的函数,用于提取给定浮点数的整数部分。 它将删除小数点及其后的所有数字(右边)。
var middleNode = function(head) {
let A = [head];//新建一个数组A,它的第一个值为非空单链表的头结点
//在数组A中插值的同时,不断地延长数组的长度以及判断数组的最后一个值的next指针是否为空
while(A[A.length-1].next!=null){
A.push(A[A.length-1].next)
}
return A[Math.trunc(A.length / 2)]
};
方法二:单指针法
空间优化:省去新建的数组A
方法思路:对链表进行两次遍历。
第一次遍历时,我们统计链表中的元素个数 N;
第二次遍历时,我们遍历到第 N/2 个元素(链表的首节点为第 0 个元素)时,将该元素返回即可。
var middleNode = function(head) {
n = 0;//用于记录链表的元素个数
cur = head;
while (cur != null) {
++n;
cur = cur.next;
}
k = 0;//用于记录第n/2的元素位置,即链表中间结点位置
cur = head;
while (k < Math.trunc(n / 2)) {
++k;
cur = cur.next;
}
return cur;
};
方法三:快慢指针法
这种方法可能得找到规律才可以总结出这种方法出来的吧,绝绝子!
继续优化方法二:用两个指针 slow 与 fast 一起遍历链表。slow 一次走一步,fast 一次走两步。那么当 fast 到达链表的末尾时,slow 必然位于中间。
var middleNode = function(head) {
slow = fast = head;
while (fast && fast.next) {
slow = slow.next;
fast = fast.next.next;
}
return slow;
};