Search in Rotated Sorted Array

给定一个已按升序排列 且在某个未知点旋转的数组,在数组中搜索目标值。如果找到返回其索引,否则返回 -1。解决方案包括分段和优化的二分查找策略。

Search in Rotated Sorted Array

给定旋转数组,找到目标数字。

Suppose a sorted array is rotated at some pivot unknown to you beforehand.

(i.e., 0 1 2 4 5 6 7 might become 4 5 6 7 0 1 2).

You are given a target value to search. If found in the array return its index, otherwise return -1.

You may assume no duplicate exists in the array.

Example

For [4, 5, 1, 2, 3] and target=1, return 2

For [4, 5,1, 2, 3] and target=0, return -1

Solution:

分段:

    public int search(int[] A, int target) {
        if (A == null || A.length == 0) {
            return -1;
        }
        int mid = 0;
        for (int i = 0; i < A.length - 1; i++) {
            if (A[i] > A[i + 1]) {
                mid = i;
                break;
            }
        }
        int result ;
        result = bs(A, 0, mid, target);
        if (result != -1) {
            return result;
        }
        result = bs(A, mid + 1, A.length - 1, target);
        return result;
    }
    private int bs(int[] A, int start, int end, int target) {
        int mid;
        while (start + 1 < end) {
            mid = start + ((end - start) >> 1);
            if (A[mid] == target) {
                return mid;
            } else if (A[mid] > target) {
                end = mid - 1;
            } else {
                start = mid + 1;
            }
        }
        if (A[start] == target) {
            return start;
        }
        if (A[end] == target) {
            return end;
        }
        return -1;
    }

首先用遍历的方法找到分割点,然后在在第一段用二分查找。如果没找到,则在第二段里找。


优化:

public int search(int[] A, int target) {
    if(A == null || A.length == 0) {
        return -1;
    }
    int pivot = A.length - 1;
    for(int i = 0; i < A.length - 1; i++) {
        if(A[i] > A[i + 1]) {
            pivot = i;
            break;
        }
    }
    if(target >= A[0]) {
        return bs(A, target, 0, pivot);
    } else if (pivot + 1 < A.length) {
        return bs(A, target, pivot + 1, A.length - 1);
    } else {
        return -1;
    }
}
   
private int bs(int[] A, int target, int start, int end) {
    int mid;
    while(start + 1 < end) {
        mid = start + ((end - start) >> 1);
        if(target < A[mid]) {
            end = mid - 1;
        } else if(target == A[mid]) {
            return mid;
        } else {
            start = mid + 1;
        }
    }
    if(A[start] == target) {
        return start;
    } else if(A[end] == target) {
        return end;
    } else {
        return -1;
    }
}

首先用遍历的方法找到分割点,然后根据target选择二分查找段。index在加1的时候注意溢出。注意pivot的初始值。


不分段:

    public int search(int[] A, int target) {
        if (A == null || A.length == 0) {
            return -1;
        }
        int start = 0;
        int end = A.length - 1;
        int mid;
        while (start + 1 < end) {
            mid = start + ((end - start) >> 1);
            if (A[mid] == target) {
                return mid;
            }
            if (A[start] < A[mid]) {
                if (A[start] <= target && target < A[mid]) {
                    end = mid - 1;
                } else {
                    start = mid + 1;
                }
            } else {
                if (A[mid] < target && target <= A[end]) {
                    start = mid + 1;
                } else {
                    end = mid - 1;
                }
            }
        }
        if (A[start] == target) {
            return start;
        }
        if (A[end] == target) {
            return end;
        }
        return -1;
    }

思路:

1. 找到转折点,将原array分成两段,分别进行二分查找。要注意转折点可能在起始点,也可能在最末尾(相当于没有转折)。

2. array的两段,一段的值比另一段的值都大。mid要么落在第一段,要么落在第二段。

如果落在前一段,则判断是否在start到mid之间,排除一半。

如果落在后一段,则判断是否在mid到end之间,排除一半。



评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值