G - 二分法+时间复杂度

蒜头君手上有个长度为n的数组A。由于数组实在太大了,所以蒜头君也不知道数组里面有什么数字,所以蒜头君会经常询问整数x是否在数组A中。

输入格式

第一行输入两个整数n和m,分别表示数组的长度和查询的次数。

接下来一行有n个整数ai​。

接下来m行,每行有1个整数x,表示蒜头君询问的整数。

输出格式

对于每次查询,如果可以找到,输出"YES",否则输出"NO"

数据范围

10^61≤n,m≤105,0≤x≤106。

输入样例

10 5
1 1 1 2 3 5 5 7 8 9
0
1
4
9
10

输出样例

NO
YES
NO
YES
NO

二分法

分为二的方法. 设 [a,b]为R的闭区间. 逐次二分法就是造出如下的区间序列 ( [an,bn]):a0=a,b0=b,且对任一自然数n, [an+1,bn+1]或者等于 [an,cn],或者等于 [cn,bn],其中cn表示 [an,bn]的中点. [2]

(1)将数据有序排列:先将一个数据集进行有序排列(可根据某种数值的大小降序或升序<当然排序的规则可根据业务规则自定义>,前提是需要查找的数据具备该规则同样的属性);

(2)数据分半:就是将排序好的数据集切分成大致相等的两份数据;

(3)查找数据:把排序好的数据拆分为个数大致相等的两半,因为有排序,查找的时候先和其中一半数据种的最大或者最小的数进行比较来断定要查找的数据是否会包含被分割后的一半数据种,然后在满足判定条件的数据集中一次获取数据进行比对直到找到数据或者比较完所有数据返回没有该数据,

时间复杂度

时间复杂度就是用来方便开发者估算出程序的运行时间

我们该如何估计程序运行时间呢,我们通常会估计算法的操作单元数量,来代表程序消耗的时间, 这里我们默认CPU的每个单元运行消耗的时间都是相同的。

假设算法的问题规模为n,那么操作单元数量便用函数f(n)来表示

随着数据规模n的增大,算法执行时间的增长率和f(n)的增长率相同,这称作为算法的渐近时间复杂度,简称时间复杂度,记为 O(f(n))

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<iostream>
using namespace std;

const int maxn = 100000 + 5;
int a[maxn];
int n,q;

bool bsearch(int x){
    int l = 0;
    int r = n - 1;
    while(l <= r){
        if(a[r] < x || a[l] > x) return false;
        int mid = (l + r) >> 1;
        if(a[mid] == x) return true;
        else if(a[mid] > x) r = mid - 1;
        else l = mid + 1;
    }
}

int main(){
    while(scanf("%d%d",&n,&q)!=EOF){
        for(int i = 0; i < n; i ++){
            scanf("%d",&a[i]);
        }
        sort(a,a + n);
        while(q--){
            int x;
            scanf("%d",&x);
            bool flag = bsearch(x);
            if(flag) printf("YES\n");//cout<<"YES"<<endl;
            else printf("NO\n"); //cout<<"NO"<<endl;
        }
    }
    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值