forked from fishercoder1534/Leetcode
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path_81.java
174 lines (166 loc) · 6.6 KB
/
_81.java
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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
package com.fishercoder.solutions;
/**
* 81. Search in Rotated Sorted Array II
*
* Follow up for "Search in Rotated Sorted Array":
* What if duplicates are allowed?
* Would this affect the run-time complexity? How and why?
* Write a function to determine if a given target is in the array.
*/
public class _81 {
public static class Solution1 {
public boolean search(int[] A, int target) {
int len = A.length;
if (len == 0) {
return false;
}
if (len == 1) {
if (A[0] == target) {
return true;
} else {
return false;
}
}
int watershed = A[0];
int watershedIndex = 0;
for (int i = 0; i < len - 1; i++) {
if (A[i] > A[i + 1]) {
watershed = A[i];
watershedIndex = i;
System.out.println("Place 1: watershed = " + watershed
+ "\twatershedIndex = " + watershedIndex);
for (int j = i + 1; j < len; j++) {
if (A[j] == A[i]) {
watershed = A[j];
watershedIndex = j;
System.out.println("Place 2: watershed = " + watershed
+ "\twatershedIndex = " + watershedIndex);
} else {
break;
}
}
}
}
System.out.println("watershed = " + watershed + "\twatershedIndex = "
+ watershedIndex);
if (target == watershed) {
return true;
} else if (target > watershed) {
/*
* here is the tricky part: when target is greater than watershed,
* it's also possible that this list is ZERO rotated, i.e. it didn't
* rotate at all! Then at this moment, watershed is not the largest
* element int this array, so we need to binary search this whole
* array.
*/
if (watershedIndex == 0) {
int start = 0;
int end = len - 1;
int mid = (start + end) / 2;
while (start <= end) {
if (target > A[mid]) {
start = mid + 1;
mid = (start + end) / 2;
} else if (target < A[mid]) {
end = mid - 1;
mid = (start + end) / 2;
} else if (target == A[mid]) {
return true;
}
}
return false;
} else {
return false;
}
} else if (target < watershed) {
/*
* target could be in either part of this sorted array, then we
* check if target is greater than A[0], if so, then search in the
* first part, if not, then check if it is greater than A[len - 1],
* if so, return -1, if not, search in the second part
*/
if (target == A[0]) {
return true;
} else if (target > A[0]) {
int start = 1;
int end = watershedIndex - 1;
int mid = (start + end) / 2;
while (start <= end) {
if (target > A[mid]) {
start = mid + 1;
mid = (start + end) / 2;
} else if (target < A[mid]) {
end = mid - 1;
mid = (start + end) / 2;
} else if (target == A[mid]) {
return true;
}
}
return false;
} else if (target < A[0]) {
if (target == A[len - 1]) {
return true;
} else if (target > A[len - 1]) {
return false;
} else if (target < A[len - 1]) {
int start = watershedIndex + 1;
int end = len - 2;
int mid = (start + end) / 2;
while (start <= end) {
if (target > A[mid]) {
start = mid + 1;
mid = (start + end) / 2;
} else if (target < A[mid]) {
end = mid - 1;
mid = (start + end) / 2;
} else if (target == A[mid]) {
return true;
}
}
return false;
}
}
}
return false;
}
}
public static class Solution2 {
public boolean search(int[] nums, int target) {
int start = 0;
int end = nums.length - 1;
//check each num so we will check start == end
//We always get a sorted part and a half part
//we can check sorted part to decide where to go next
while (start <= end) {
int mid = start + (end - start) / 2;
if (nums[mid] == target) {
return true;
}
//if left part is sorted
if (nums[start] < nums[mid]) {
if (target < nums[start] || target > nums[mid]) {
//target is in rotated part
start = mid + 1;
} else {
end = mid - 1;
}
} else if (nums[start] > nums[mid]) {
//right part is rotated
//target is in rotated part
if (target < nums[mid] || target > nums[end]) {
end = mid - 1;
} else {
start = mid + 1;
}
} else {
//duplicates, we know nums[mid] != target, so nums[start] != target
//based on current information, we can only move left pointer to skip one cell
//thus in the worst case, we would have target: 2, and array like 11111111, then
//the running time would be O(n)
start++;
}
}
return false;
}
}
}