-
Notifications
You must be signed in to change notification settings - Fork 155
/
Copy pathinsert-into-a-cyclic-sorted-list.js
79 lines (71 loc) · 2.22 KB
/
insert-into-a-cyclic-sorted-list.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
/**
* Insert into a Cyclic Sorted List
*
* Given a node from a cyclic linked list which is sorted in ascending order, write a function to insert a value into
* the list such that it remains a cyclic sorted list. The given node can be a reference to any single node in the list,
* and may not be necessarily the smallest value in the cyclic list.
*
* If there are multiple suitable places for insertion, you may choose any place to insert the new value. After the
* insertion, the cyclic list should remain sorted.
*
* If the list is empty (i.e., given node is null), you should create a new single cyclic list and return the reference
* to that single node. Otherwise, you should return the original given node.
*/
/**
* Definition for singly-linked list.
* function ListNode(val, next = null) {
* this.val = val;
* this.next = next;
* }
*/
import ListNode from 'common/list-node';
/**
* @param {ListNode} start
* @param {number} x
*/
const insert = (start, x) => {
// if start is null, create a node pointing to itself and return
if (start === null) {
const node = new ListNode(x, null);
node.next = node;
return node;
}
// is start is NOT null, try to insert it into correct position
let cur = start;
while (true) {
// case 1A: has a tipping point, still climbing
if (cur.val < cur.next.val) {
if (cur.val <= x && x <= cur.next.val) {
// x in between cur and next
insertAfter(cur, x);
break;
}
// case 1B: has a tipping point, about to return back to min node
} else if (cur.val > cur.next.val) {
if (cur.val <= x || x <= cur.next.val) {
// cur is the tipping point, x is max or min val
insertAfter(cur, x);
break;
}
// case 2: NO tipping point, all flat
} else {
if (cur.next === start) {
// insert x before we traverse all nodes back to start
insertAfter(cur, x);
break;
}
}
// None of the above three cases met, go to next node
cur = cur.next;
}
return start;
};
/**
* Insert value x after Node node
* @param {ListNode} node
* @param {number} x
*/
const insertAfter = (node, x) => {
node.next = new ListNode(x, node.next);
};
export { insert };