Search in Rotated Sorted Array II

Question

Problem Statement

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.

题解

仔细分析此题和之前一题的不同之处,前一题我们利用A[start] < A[mid]这一关键信息,而在此题中由于有重复元素的存在,在A[start] == A[mid]时无法确定有序数组,此时只能依次递增start/递减end以缩小搜索范围,时间复杂度最差变为O(n)。

C++

  1. class Solution {
  2. /**
  3. * param A : an integer ratated sorted array and duplicates are allowed
  4. * param target : an integer to be search
  5. * return : a boolean
  6. */
  7. public:
  8. bool search(vector<int> &A, int target) {
  9. if (A.empty()) {
  10. return false;
  11. }
  12. vector<int>::size_type start = 0;
  13. vector<int>::size_type end = A.size() - 1;
  14. vector<int>::size_type mid;
  15. while (start + 1 < end) {
  16. mid = start + (end - start) / 2;
  17. if (target == A[mid]) {
  18. return true;
  19. }
  20. if (A[start] < A[mid]) {
  21. // situation 1, numbers between start and mid are sorted
  22. if (A[start] <= target && target < A[mid]) {
  23. end = mid;
  24. } else {
  25. start = mid;
  26. }
  27. } else if (A[start] > A[mid]) {
  28. // situation 2, numbers between mid and end are sorted
  29. if (A[mid] < target && target <= A[end]) {
  30. start = mid;
  31. } else {
  32. end = mid;
  33. }
  34. } else {
  35. // increment start
  36. ++start;
  37. }
  38. }
  39. if (A[start] == target || A[end] == target) {
  40. return true;
  41. }
  42. return false;
  43. }
  44. };

Java

  1. public class Solution {
  2. /**
  3. * param A : an integer ratated sorted array and duplicates are allowed
  4. * param target : an integer to be search
  5. * return : a boolean
  6. */
  7. public boolean search(int[] A, int target) {
  8. if (A == null || A.length == 0) return false;
  9. int lb = 0, ub = A.length - 1;
  10. while (lb + 1 < ub) {
  11. int mid = lb + (ub - lb) / 2;
  12. if (A[mid] == target) return true;
  13. if (A[mid] > A[lb]) {
  14. // case1: numbers between lb and mid are sorted
  15. if (A[lb] <= target && target <= A[mid]) {
  16. ub = mid;
  17. } else {
  18. lb = mid;
  19. }
  20. } else if (A[mid] < A[lb]) {
  21. // case2: numbers between mid and ub are sorted
  22. if (A[mid] <= target && target <= A[ub]) {
  23. lb = mid;
  24. } else {
  25. ub = mid;
  26. }
  27. } else {
  28. // case3: A[mid] == target
  29. lb++;
  30. }
  31. }
  32. if (target == A[lb] || target == A[ub]) {
  33. return true;
  34. }
  35. return false;
  36. }
  37. }

源码分析

A[start] == A[mid]时递增start序号即可。

复杂度分析

最差情况下 O(n), 平均情况下 O(\log n).