随机查找(一切看命)
  EC73qDgVNYjz 2023年11月05日 50 0

  对于一个给定的数组,若要查找当中是否包含某个值,传统的方法是遍历数组中的每一个元素,如果找到则返回。如果学习过数据结构,也可以立马想到用哈希表来存储,哈希表的查找性能优异,一般可以达到O(1)的时间复杂度,在最坏情况下也有可能达到O(n)的复杂度。但是今天,我将带来一种有意思的查找方式,也就是通过随机数来对数组进行划分。这种方式运用到了分治的思想,每次随机生成一个下标,然后判断该下标的元素是否等于目标值,如果不等于,则对该下标两边的元素进行递归查找,之后每一次划分的下标也是基于随机数生成。说起来比较抽象,我们来看看代码。

bool RandomFind(const std::vector<int>& v, int l, int r, const int& value) {
    if (l > r)
        return false;
    if (l == r) {
        if (v[l] == value)
            return true;
        return false;
    }

    std::uniform_int_distribution dis(l, r);
    int RandomIndex = dis(g);

    bool LeftPart = RandomFind(v, 0, RandomIndex, value);
    bool RightPart = RandomFind(v, RandomIndex + 1, r, value);

    return LeftPart || RightPart;
}

 注意递归的终止条件,需要防止数组最小下标l大于最大的下标,还要注意处理好数组只剩下一个元素的情况。由于我们只需要判断是否找到某个元素,因此LeftPart和RightPart只需要有其中一部分找到即可。不过我们还可以对上述代码进行一定的“剪枝”优化,换句话说就是如果随机生成的下标对应的元素是目标值,我们直接return,没必要再做无意义的搜索了。对上面代码稍作改动即可。

bool RandomFind(const std::vector<int>& v, int l, int r, const int& value) {
    if (l > r)
        return false;
    if (l == r) {
        if (v[l] == value)
            return true;
        return false;
    }

    std::uniform_int_distribution dis(l, r);
    int RandomIndex = dis(g);
    if (v[RandomIndex] == value)
        return true;

    bool LeftPart = RandomFind(v, 0, RandomIndex - 1, value);
    bool RightPart = RandomFind(v, RandomIndex + 1, r, value);

    return LeftPart || RightPart;
}

  贴个Python版的。

import random 

def randomFind(v: list, l: int, r: int, target: int) -> bool:
    if l > r:
        return False
    if l == r:
        if v[l] == target:
            return True
        return False

    dis = random.randint(l, r)
    if v[dis] == target:
        return True 
    
    LeftPart = randomFind(v, 0, dis - 1, target)
    RightPart = randomFind(v, dis + 1, r, target)

    return LeftPart or RightPart

  不过话说回来,这种查找方式容易递归较深,性能也不高,仅仅当随机生成的下标对应的元素刚好等于目标值的时候可能稍微快点,所以说一切看命。

【版权声明】本文内容来自摩杜云社区用户原创、第三方投稿、转载,内容版权归原作者所有。本网站的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@moduyun.com

  1. 分享:
最后一次编辑于 2023年11月08日 0

暂无评论

推荐阅读
EC73qDgVNYjz