刷题 ------ 二分枚举(查找)

news2024/9/24 23:10:46

文章目录

  • 1.x 的平方根
  • 2.第一个错误的版本
  • 3.有效的完全平方数
  • 4.猜数字大小
  • 5.排列硬币
  • 6. 寻找比目标字母大的最小字母
  • 7. 二分查找
  • 8.检查整数以及其两倍数是否存在
  • 9. 两个数组间的距离值
  • 10.特殊的数组的特征值
  • 11.找出数组排序后的目标下标
  • 12.和有限的最长子序列
  • 13.正整数和负数的最大计数
  • 14.最小公共值
  • 15.统计和小于目标的下标对数目
  • 16.LCP.早餐组合
  • 16.LCP.采购方案
  • 17.LCR.两数之和 || - 输入有序数组
  • 18. LCR 搜索插入位置
  • 19.LCR 山脉素组的峰顶索引
  • 19.寻找旋转排序数组中的最小值
  • 20.LCR.统计目标成绩出现的次数
  • 21.LCR.点名
  • 22.LCR.查找总价格为目标值的两个商品
  • 23.魔术索引

本篇是二分枚举,其中所有题都可以用线性枚举做出来,还有的双指针也可以。
建议先线性枚举做一遍,然后知道求什么后,再去想着二分去求。
本篇的解法中只有二分枚举,没有线性枚举。

1.x 的平方根

在这里插入图片描述

  • 不适用库函数哈。
  • 我使用二分法,举两个例子,8和9。
  • 使left 和 right 分别等于 1 和 x。
  • 然后取其中间值mid,
  • 如果mid*mid > x,right = mid;
  • 如果mid*mid < x, left = mid;
  • mid*mid = x ,return mid;
  • 还有一种情况就是如果left + 1 == right 也就是说 left 和mid相邻的时候,直接返回left就好了

在这里插入图片描述

int mySqrt(int x)
{
    long long left = 1,right = x,mid;
    while(left + 1 != right)
    {
        mid = (left + right) / 2;
        if(mid * mid > x)
        {
            right = mid;
        }
        else if(mid * mid < x)
        {
            left = mid;
        }
        else
        {
            return mid;
        }
    }

    return left;
}

2.第一个错误的版本

在这里插入图片描述

  • 首先得知道这个isBadVersion函数使质量合格返回false,不合格返回true。
  • 然后对其进行二分查找就好了

int firstBadVersion(int n)
{
    int left = 1, right = n;
    while(left < right)
    {
        //防止溢出
        int mid = left + (right - left) / 2;
        if(isBadVersion(mid))
        {
            //不合格  [left,mid]
            right = mid;
        }
        else
        {
            //[mid+1,right]
            left = mid + 1;
        }
    }

    return left;
}

3.有效的完全平方数

在这里插入图片描述

  • 上面第一题ok了,下面这道题也就ok了
bool isPerfectSquare(int num)
{
    long long left = 1,right = num;
    while(left + 1 != right)
    {
        long long mid = (left + right) / 2;
        if(mid * mid > num)
        {
            right = mid;
        }
        else if (mid * mid < num)
        {
            left = mid;
        }
        else
        {
            return true;
        }
    }        


    return false;
}

4.猜数字大小

在这里插入图片描述

  • 嗯。。。。。。
int guessNumber(int n)
{
    int left = 1,right = n;
    while(left != right)
    {
        int mid = left + (right - left) / 2;
        int flag = guess(mid);
        if( flag == -1)
        {
            //比mid小
            right = mid - 1;
        }
        else if(flag == 1)
        {
            //比mid大
            left = mid + 1;
        }
        else 
        {
            return mid;
        }
    }

    return left;
}

5.排列硬币

在这里插入图片描述

  • 在两个数中间去二分的查找一个合适的mid,找到mid后
  • 求出mid所对应的等差数列和,(mid * (mid + 1))/ 2;
  • 如果和 小于 硬币,就证明下一行可能继续放,所以left = mid
  • 如果和 大于 硬币,就证明当前这一行,肯定使不行了,所以使right = mid
  • 如果等于的话,返回mid就好了
  • 最后就使介于两个数中间,left + 1 == right.
  • 返回那个小的即可
int arrangeCoins(int n)
{
    long long left = 1,right = n;
    while(left + 1 != right)
    {
        long long mid = left + (right - left) / 2;
        long long sum = (mid*(mid+1)) / 2;    //等差数列
        if(sum > n)
        {
            //放的多了
            right = mid;
        }
        else if (sum < n)
        {
            left = mid;
        }
        else 
        {
            return mid;
        }
    }
    return left;
}

6. 寻找比目标字母大的最小字母

在这里插入图片描述

  • 这个字符数组是一个升序的数组,如果target 大于等于 最后一个,就说明里面肯定没有比target更大的了,返回数组第一个即可
  • 如果mid 大于 target 那么肯定使 right 变小
  • 如果mid 小于 target 那么肯定使 left 变大
  • 当他俩错过时候,返回left所在的位置即可

结合下面两张图可以理解
在这里插入图片描述
在这里插入图片描述

char nextGreatestLetter(char* letters, int lettersSize, char target)
{
    if(target >= letters[lettersSize-1])
    {
        return letters[0];
    }

    int left = 0, right = lettersSize - 1;
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(letters[mid] > target)
        {
            right = mid - 1;
        }
        else
        {
            left = mid + 1;
        }
    }

    return letters[left];
}

7. 二分查找

在这里插入图片描述

  • 嗯。。。经典,其实应该放在第一道题去刷的。绝对得是二分查找
int search(int* nums, int numsSize, int target)
{
    int left = 0,right = numsSize - 1;
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(nums[mid] < target)
        {
            left = mid + 1;
        }
        else if (nums[mid] > target)
        {
            right = mid - 1;
        }
        else
        {
            return mid;
        }
    }    


    return -1;
}

8.检查整数以及其两倍数是否存在

在这里插入图片描述

  • 暴力,和哈希,也能做出来。
  • 这里就先用二分吧
  • 首先将数组进行排序,然后往前遍历,要注意,如果当前数的二倍是负数,需要往前找。

代码逻辑:

bool checkIfExist(int* arr, int arrSize)
{

    //升序排序
    qsort(arr,arrSize,sizeof(arr[0]),cmp_int);
    int i;
    for (i = 0; i < arrSize - 1; i++)
    {
        int target = arr[i] * 2;
        //printf("%d target = %d\n",arr[i],target);
        if( target >= 0 && BinSearch(arr + i + 1,arrSize - i - 1,target))
        {
            //正数往后找
            return true;
        }
        else if(BinSearch(arr,i,target))
        {
            //负数往前找
            return true;
        }
    }

    return false;

}

函数:

int cmp_int(const void* x, const void* y)
{
    return *(int*)x - *(int*)y;
}

bool BinSearch(int* nums,int numsSize,int target)
{
    int left = 0,right = numsSize - 1;
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(nums[mid] > target)
        {
            right = mid - 1;
        } 
        else if(nums[mid] < target)
        {
            left = mid + 1;
        }
        else 
        {
            return true;
        }
    }

    return false;
}

9. 两个数组间的距离值

在这里插入图片描述

  • 这道题,暴力枚举会更简单,更直接的想到,我第一遍就是暴力过的
  • 这里写一下二分的方法吧
  • 首先二分的前提下一定是得有序的。
  • 所以,先给arr2进行排序。
  • 然后去arr2里面去找那个离arr1[i]中距离最近的那个数。
  • 如果离arr1[i]最近的数与arr1[i]的差距都 > d
  • 那样不就证明其余的都满足了吗?

但是出现了一个问题,就是我二分的时候,确定不了那个数,我只能确定两个,然后再单独判断哪一个更近,就比如下图中,不管查找1 和 4,都会再 -3 和 6 的时候停下来。
但是和1相邻的是-3 和 4 相邻的则是 6。
在这里插入图片描述

int cmp_int(const void* x, const void* y)
{
    return *(int*)x - *(int*)y;
}

int BinSearch(int* nums, int numsSize, int target)
{
    int left = 0, right = numsSize -1;
    int leftgarp = 0,rightgarp = 0;
    while(left + 1 != right && numsSize != 1)
    {
        int mid = (left + right) / 2;
        if(nums[mid] > target)
        {
            right = mid;
        }
        else if(nums[mid] < target)
        {
            left = mid;
        }
        else
        {
            return nums[mid];
        }
    }
    leftgarp = abs(nums[left] - target);
    rightgarp = abs(nums[right] - target);
    return leftgarp < rightgarp ? nums[left] : nums[right];
}

int findTheDistanceValue(int* arr1, int arr1Size, int* arr2, int arr2Size, int d)
{
    int i,j,ans = 0;
    qsort(arr2,arr2Size,sizeof(int),cmp_int);
    for (i = 0; i < arr1Size; i++)
    {
    	//找出距离arr1[i]最近的那个数
        int near = BinSearch(arr2,arr2Size,arr1[i]);
        if(abs(arr1[i] - near) > d)
        {
            ans++;
        }
    }
    return ans;
}

10.特殊的数组的特征值

在这里插入图片描述

  • 暴力肯定也是行得通的,暴力的目的,就是在nums中找出大于等于 x 的元素个数.
int specialArray(int* nums, int numsSize)
{
    int i,j;
    for (i = 0; i <= numsSize; i++)
    {
        int count = 0;  //记录有多少数据个大于i
        for (j = 0; j < numsSize; j++)
        {
            if(nums[j] >= i)
            {
                count++;
            }
        }
        if(count == i)
        {
            return i;
        }
    }

    return -1;
}

上面的代码中是用线性枚举的方式去找,而当然也可以用二分枚举的方式去找。

int specialArray(int* nums, int numsSize)
{
    int i,j;
    qsort(nums,numsSize,sizeof(int),cmp_int);
    for (i = 0; i <= numsSize; i++)
    {
        int count = BinSearch (nums,numsSize,i);  //记录有多少数据个大于i
        if(count == i)
        {
            return i;
        }
    }

    return -1;
}

代码逻辑是一样的,主要还得是看看这个二分到底是如何实现的,就是普通二分法还不行,因为它里面会有重复的元素。下面这张图中,不可以直接返回nums - mid。因为左边还有一个6会被忽视掉,所以在代码中得变一变条件
在这里插入图片描述

  • 但凡用到二分法,数组一定得是有序的,同样的计算出mid
  • nums[mid] >= right, right = mid
  • nums[mid] < right, left = mid + 1;
  • 当left 等于right 时候,还得去判断一下 当前的这个数,是否大于等于所给的target,
  • 如果大于返回numSize - left(right)
  • 如果小于的话,肯定返回0.
//找出里面有多少个 >= target 的元素个数
int BinSearch(int* nums,int numsSize,int target)
{
    int left = 0, right = numsSize - 1;
    while(left < right)
    {
        int mid = (left + right) / 2;
        if(nums[mid] >= target)
        {
            right = mid;
        }
        else if(nums[mid] < target)
        {   
            left = mid + 1;
        }
    }

    return nums[left] >= target ? numsSize - left : 0;
}

11.找出数组排序后的目标下标

在这里插入图片描述

  • 首先将数组进行排序,然后对其进行二分。
  • 找到所对应的target后,对其左边右边进行遍历,如果发现有重复的,将其也记录下来
  • 最后再给ans数组进行排序就好了
int cmp_int(const void* x, const void* y)
{
    return *(int*)x - *(int*)y;
}

int* targetIndices(int* nums, int numsSize, int target, int* returnSize)
{
    int* ans = (int*)malloc(sizeof(int) * numsSize);
    int size = 0,i;
    qsort(nums,numsSize,sizeof(int),cmp_int);
    int left = 0, right = numsSize - 1;
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(nums[mid] > target)
        {
            right = mid - 1;
        }
        else if(nums[mid] < target)
        {
            left = mid + 1;
        }
        else
        {
            int l = mid - 1,r = mid + 1;
            //当前的
            ans[size++] = mid;
            //左边
            while(l >= 0 && nums[l] == target)
            {
                ans[size++] = l--;
            }
            //右边
            while(r < numsSize && nums[r] == target)
            {
                ans[size++] = r++;
            }
            break;
        }
    }

    qsort(ans,size,sizeof(int),cmp_int);
    *returnSize = size;
    return ans;
}

12.和有限的最长子序列

在这里插入图片描述

  • queries(查询)就是去nums当中,找出 子序列 和 小于等于 queries[i]的长度。
  • 题目中最后一句话的意思就是说,只要是数组内的数据就行,所以我们可以先将数组排序。
  • 然后利用排好序的数组计算出每个所对应的前缀和
    在这里插入图片描述
  • 有了前缀和的数组,就可以去二分查找所对应的 queries[i]。
  • 有下面两种情况

这种情况下,题目中要求 <= 所以当前的也能取上,而left是下标,所以得 + 1在这里插入图片描述
而下面这种,12 大于10 就意味着,肯定不能取到12,所以返回前面的三个即可
在这里插入图片描述

int cmp_int(const void* x, const void* y)
{
    return *(int*)x - *(int*)y;
}

int BinarySearch(int* nums, int numsSize,int target)
{
    int left = 0,right = numsSize - 1;
    while(left < right)
    {
        int mid = (left + right) / 2;
        if(nums[mid] < target)
        {
            left = mid + 1;
        }
        else
        {
            right = mid;
        }
    }

    return nums[left] > target ? left : left + 1;
}


int* answerQueries(int* nums, int numsSize, int* queries, int queriesSize, int* returnSize)
{
    int* ans = (int*)malloc(sizeof(int) * queriesSize);
    int size = 0,i,j;
    //排序
    qsort(nums,numsSize,sizeof(int),cmp_int);
    //前缀和数组
    int* prefixsum = (int*)malloc(sizeof(int) * numsSize);
    int sum = 0;
    for (i = 0; i < numsSize; i++)
    {   
        sum += nums[i];
        prefixsum[i] = sum;
    }
    //找每一个对应的长度
    for (i = 0; i < queriesSize; i++)
    {
        ans[size++] = BinarySearch(prefixsum,numsSize,queries[i]);
    }
    *returnSize = size;
    return ans;
}


13.正整数和负数的最大计数

在这里插入图片描述

  • 这道题,和上面11题一样,我这道题是如果找到0的话,就分开两头去找最近的一个不是0的数。

要注意的是,下面的代码逻辑,不管如何,
都是right 代表最后一个负数出现位置,
left 代表一个正数出现的位置
看下图,不管是 1 或者 -1 都是不会改变上面的逻辑的
在这里插入图片描述

int maximumCount(int* nums, int numsSize)
{
    int pos = 0, neg = 0;
    int left = 0, right = numsSize - 1;
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(nums[mid] > 0)
        {
            right = mid - 1;
        }
        else if (nums[mid] < 0)
        {
            left = mid + 1;
        }
        else
        {
        	// 发现是0,分开去找,
            right = mid - 1;
            left = mid + 1;
            //左边
            while(right >= 0 && nums[right] == 0)
            {
                right--;
            }
            //右边
            while(left < numsSize && nums[left] == 0)
            {
                left++;
            }
            break;
        }
    }
    pos = numsSize - left;
    neg = right + 1;

    return pos > neg ? pos : neg;
}


14.最小公共值

在这里插入图片描述

  • 这道题,感觉双指针更快一点。
  • 下面是二分的解法:
  • 把nums1想象成一个target数组,去nums2中去找是否有nums[1]中的值即可
int BinarySearch(int* nums, int numsSize, int target)
{
    int left = 0, right = numsSize - 1;
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(nums[mid] > target)
        {
            right = mid - 1;
        }
        else if(nums[mid] < target)
        {
            left = mid + 1;
        }
        else 
        {
            return true;
        }
    }

    return false;
}


int getCommon(int* nums1, int nums1Size, int* nums2, int nums2Size)
{
    int i;
    for (i = 0; i < nums1Size; i++)
    {
        if(BinarySearch(nums2,nums2Size,nums1[i]))
        {
            return nums1[i];
        }
    }
    return -1;
}

15.统计和小于目标的下标对数目

在这里插入图片描述

  • 下面是暴力的做法:记住下面的nums[i] + nums[j] < target。后面会用到
int countPairs(int* nums, int numsSize, int target)
{
    int i,j,count = 0;
    for (i = 0; i < numsSize; i++)
    {
        for (j = i + 1; j < numsSize; j++)
        {
            if(nums[i] + nums[j] < target)
            {
                count++;
            }
        }
    }

    return count;
}

由题可知,我们只需要扎找到两个不同的下标 i 和 j 就好了。但是不能重复,比如说下图中
在这里插入图片描述
这个是题目中示例一排序后的数组,就是 nums[0] + nums[1] < 2. 但是不能nums[1] + nums[0] < 2.
这个就叫重复,也可以像题目中那样说 i < j,所以排序不会影响最后的结果。
而不重复的就是,你在传数组的时候,不用将自身传过来,从下一个的位置开始传

上面的暴力中会发现,我们呢是需要找 j,但是这个是线性枚举的找j方式,我们只需要换成二分枚举即可。
看下面的推到公式,简单的不等式对吧。 所以不等式右边的,就是在二分中与mid进行比较的值
在这里插入图片描述

  • 有了上面的两个思想,二分就简单了,不可重复mid与谁作比较.
  • 下面是代码。
int cmp_int(const void* x, const void* y)
{
    return *(int*)x - *(int*)y;
}

int BinartSearch(int* nums, int numsSize, int target)
{
    int left = 0, right = numsSize - 1;
    while(left < right)
    {
        int mid = (left + right) / 2;
        if(nums[mid] < target)
        {
            left = mid + 1;
        }
        else
        {
            right = mid;
        }
    }

    return nums[left] < target ? left + 1 : left;
}


int countPairs(int* nums, int numsSize, int target)
{
    int i,j,count = 0;
    qsort(nums,numsSize,sizeof(int),cmp_int);
    for (i = 0; i < numsSize - 1; i++)
    {
        count += BinartSearch(nums + i + 1,numsSize - i - 1,target - nums[i]);
    }

    return count;
}

16.LCP.早餐组合

在这里插入图片描述

  • 和上题一样,这个上题更简单,因为有两个数组,不用那么麻烦的考虑,在一个数组里传来传去。
  • 你饮料不可以超过 x - staple[i]。
int cmp_int(const void* x,const void* y)
{
    return *(int*)x - *(int*)y;
}

long long BinarySearch(int* nums, int numsSize, int target)
{
    long long left = 0, right = numsSize - 1;
    while(left < right)
    {
        long long mid =  (left + right) / 2; 
        if(nums[mid] > target)
        {
            right = mid - 1;
        }
        else
        {
            left = mid + 1;
        }
    }

    return nums[left] <= target ? left + 1 : left;
}


int breakfastNumber(int* staple, int stapleSize, int* drinks, int drinksSize, int x)
{
    int i,j;
    long long count = 0;
    qsort(drinks,drinksSize,sizeof(int),cmp_int);
    for (i = 0; i < stapleSize; i++)
    {
        count += BinarySearch(drinks,drinksSize,x - staple[i]);
    }

    return (int)(count % 1000000007);
}

// mid <= x - staple[i]

16.LCP.采购方案

在这里插入图片描述

  • 这道题和上一题是类似的。
int cmp_int(const void* x, const void* y)
{
    return *(int*)x - *(int*)y;
}


long long BinarySearch(int* nums, int numsSize, int target)
{
    long long left = 0, right = numsSize - 1;
    while(left < right)
    {
        long long mid = (left + right) / 2;
        if(nums[mid] > target)
        {
            right = mid - 1;
        }
        else
        {
            left = mid + 1;
        }
    }

    return nums[left] <= target ? left + 1 : left;
}


int purchasePlans(int* nums, int numsSize, int target)
{
    int i;
    qsort(nums,numsSize,sizeof(int),cmp_int);
    long long count = 0;
    for (i = 0; i < numsSize - 1; i++)
    {
        if(nums[i] > target)
        {
            break;
        }
        count += BinarySearch(nums + i + 1, numsSize - i - 1,target - nums[i]);
    }

    return (int)(count % 1000000007);
}

17.LCR.两数之和 || - 输入有序数组

在这里插入图片描述

  • 二分时候需要找的target 是 target - numbers[i]。
  • 还有要注意的是从i+1 的位置开始找,i以前数肯定都不行,也不允许自己加自己


/**
 * Note: The returned array must be malloced, assume caller calls free().
 */
int BinarySearch(int* nums,int left, int right, int target)
{
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(nums[mid] > target)
        {
            right = mid -1;
        }
        else if (nums[mid] < target)
        {
            left = mid + 1;
        }
        else 
        {
            return mid;
        }
    }
    return -1;
}
int* twoSum(int* numbers, int numbersSize, int target, int* returnSize)
{
    int* ans = (int*)malloc(sizeof(int) * 2);
    int i;
    for (i = 0; i < numbersSize; i++)
    {
        int index = BinarySearch(numbers,i + 1,numbersSize - 1,target - numbers[i]);
        if(index != -1)
        {
            ans[0] = i;
            ans[1] = index;
        }
    }
    *returnSize = 2;
    return ans;
}

这道题其实用双指针的效果也很好。



/**
 * Note: The returned array must be malloced, assume caller calls free().
 */
int* twoSum(int* numbers, int numbersSize, int target, int* returnSize)
{
    int left = 0, right = numbersSize - 1;
    int* ans = (int*)malloc(sizeof(int) * 2);
    *returnSize = 2;
    while(left < right)
    {
        int sum = numbers[left] + numbers[right];
        if(sum > target)
        {
            right--;
        }
        else  if(sum < target)
        {
            left++;
        }
        else
        {
            ans[0] = left;
            ans[1] = right;
            break;
        }
    }

    return ans;
}

18. LCR 搜索插入位置

在这里插入图片描述

  • ok啦。
int searchInsert(int* nums, int numsSize, int target)
{
    int left = 0, right = numsSize - 1;
    while(left < right)
    {
        int mid = (left + right) / 2;
        if(nums[mid] < target)
        {
            left = mid + 1;
        }
        else 
        {
            right = mid;
        }
    }

    return nums[left] < target ? left + 1 : left;
}

19.LCR 山脉素组的峰顶索引

在这里插入图片描述

  • 如果发现nums[mid] > nums[mid + 1]。它可能是峰顶,因为并没有与它的前一个去教,无法判断
    • 所以此时使 right = mid - 1
      在这里插入图片描述
  • 如果nums[mid] < nums[mid + 1],它不可能是峰顶,
  • 使left = mid +1;
int peakIndexInMountainArray(int* arr, int arrSize)
{
    int left = 0, right = arrSize - 2,ans = 0;
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(arr[mid] > arr[mid + 1])
        {
            //记录当前的这个峰值
            ans = mid;
            right = mid - 1;
        }
        else
        {
            left = mid + 1;
        }

    }

    return ans;
}

19.寻找旋转排序数组中的最小值

在这里插入图片描述

  • 额。。听我说,我在这里套娃呢,先使打开普通,又是打开困难,最后打开中等。
  • 只看简单题,没理解其中的意思,简单题目中描述,并没有说数组使有序旋转过的,我还以为,原来使有序的现在变成了无序。想了半天,硬使没想通,为啥这个无序的能用二分?
  • 这三道题的测试用例,也使有不同的。
  • 举两个例子吧。
    这种情况是我们可以直接发现,直接想到的。
    在这里插入图片描述
    下面这主要是知道,为什么相同的时候,得right–。
    在这里插入图片描述
int findMin(int* nums, int numsSize)
{   
    int left = 0,right = numsSize - 1;
    while(left < right)
    {
        int mid = (left + right) / 2;
        if(nums[mid] > nums[right])
        {
            left = mid + 1;
        }
        else if(nums[mid] < nums[right])
        {
            right = mid;
        }
        else
        {
            //相同
            right--;
        }
    }

    return nums[left];
}

20.LCR.统计目标成绩出现的次数

在这里插入图片描述

没得说,没得说,我刚开始想的哈希表,直接做出来,然后再写二分的办法,结果出来了个这。
我笑了哈哈哈,-1分,可以可以。我抗不了.
在这里插入图片描述

这道题,可以用哈希表,暴力,双指针都可以,我这边是使用的二分法

  • 这道题的二分法,还不太一样,需要分别求出左边的起始位置右边的结束位置
  • 先学会分别求左边和右边。

下图最后返回的下标是 3
在这里插入图片描述
右边,要注意右边返回的不是下标,而是所对应的前一个下标,所以最后还得减去1
在这里插入图片描述

  • 当会找左边和右边的位置时候,它还是有序的,他俩相减就是长度,也就是次数

可以看出,二分查找,才是里面最关键的一环

int countTarget(int* scores, int scoresSize, int target)
{
    int leftIndex = BinarySearch(scores,scoresSize,target,true);
    int rightIndex = BinarySearch(scores,scoresSize,target,false) - 1;
    if(leftIndex == rightIndex)
    {   //同一个位置,就是1次呗
        return 1;
    }

    return rightIndex - leftIndex + 1;
}

我用flag标识找左边还是找右边。


//flag == true 表示找最右边的target
//flag == false 表示找最左边的 target
int BinarySearch(int* nums,int numsSize,int target, bool flag)
{
    int left = 0, right = numsSize - 1;
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if((flag && nums[mid] >= target) || (!flag) && nums[mid] > target)
        {
            right = mid - 1;
        }
        else
        {
           left = mid + 1;
        }
    }

    return left;
}

21.LCR.点名

在这里插入图片描述

  • 如果records[mid] == mid, 说明至此之前的顺序肯定是对的,并没有缺失
  • 如果records[mid] != mid ,说明至此之肯定有顺序是错的,已经缺失或者正好就是自己
int takeAttendance(int* records, int recordsSize)
{
    int left = 0, right = recordsSize - 1;
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(records[mid] == mid)
        {
            left = mid + 1;
        } 
        else
        {
            right = mid - 1;
        }
    }

    return left;
}

22.LCR.查找总价格为目标值的两个商品

在这里插入图片描述

  • 这道题与前面的题类似,甚至更简单。
int BinarySearch(int* nums, int left, int right,int target)
{
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(nums[mid] > target)
        {
            right = mid - 1;
        }
        else if(nums[mid] < target)
        {
            left = mid + 1;
        }
        else
        {
            return target;
        }
    }

    return -1;
}

int* twoSum(int* price, int priceSize, int target, int* returnSize)
{
    int* ans = (int*)malloc(sizeof(int) * 2);
    *returnSize = 2;
    int i;
    for (i = 0; i < priceSize; i++)
    {
        int val = BinarySearch(price,i + 1,priceSize - 1,target - price[i]);
        if(val != -1)
        {
            ans[0] = price[i];
            ans[1] = val;
            return ans;
        }
    }

    return ans;
}

23.魔术索引

在这里插入图片描述

  • 这道题,在一个循环里的一般的二分法,肯定是不行的,看完下面图片,
  • 全部都是nums[mid] > mid。
  • 但是答案确实两个极端
    在这里插入图片描述
  • 所以一般的while循环方式肯定是不行的,接下来就得采取递归的方式,进行二分。
  • 运用递归的方式,在前面的数组转二叉搜索树中做过,很类似的递归方式,只是停止的条件不一致。
    在这里插入图片描述
  • 知道左边是[left,mid-1] 右边是[mid+1,right]ok了
  • 然后拿一个index用来记录nums[mid]是否等于 mid 如果找到了的话,就使index = mid就好了
  • 要注意,判断nums[mid] 是否等于mid 需要先将左边遍历到底后,再去进行,比如下张图,当第一次计算即可算出3合适,但是2也合适,它比3小,所以返回3肯定是错的在这里插入图片描述
void BinarySearch(int* nums, int left, int right,int* index)
{
    if(left > right)
    {
        return;
    }
    int mid = (right + left) / 2;
    BinarySearch(nums,left, mid - 1,index);
    if(*index != -1)
    {
        return;
    }
    if(nums[mid] == mid)
    {
        *index = mid;
        return;
    }
    BinarySearch(nums,mid + 1, right,index);

}


int findMagicIndex(int* nums, int numsSize)
{
    int index = -1;
    BinarySearch(nums,0,numsSize - 1,&index);
    return index;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1387717.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

windows使用redis-安装和配置

windows使用redis 安装和配置 下载安装方式一-使用压缩包安装解压到指定的文件Redis安装为Windows服务安装成功 方式二-MSI安装包安装完成 Redis配置远程访问1.修改配置文件redis.windows.conf2.修改完redis配置文件&#xff0c;必须重启redis 下载 先下载Redis for windows 的…

[GXYCTF2019]Ping Ping Ping

1.访问页面&#xff0c;提示传参为ip。2.?ip明显存在命令执行注入&#xff0c;使用 ; 或者 | 闭合上一条命令。 经过fuzz测试&#xff0c;过滤了空格、bash字符、flag字符、以及一些特殊符号。?ip;ls时&#xff0c;发现flag.php就在当前目录下。 3.构造POC&#xff0c;获取…

2024全新开发API接口调用管理系统网站源码 附教程

2024全新开发API接口调用管理系统网站源码 附教程 用layui框架写的 个人感觉很简洁 方便使用和二次开发

京东ES支持ZSTD压缩算法上线了:高性能,低成本 | 京东云技术团队

1 前言 在《ElasticSearch降本增效常见的方法》一文中曾提到过zstd压缩算法[1]&#xff0c;一步一个脚印我们终于在京东ES上线支持了zstd&#xff1b;我觉得促使目标完成主要以下几点原因&#xff1a; Elastic官方原因&#xff1a;zstd压缩算法没有在Elastic官方的开发计划中&…

正则表达式和爬虫

目录 一、正则表达式&#xff1a; 作用&#xff1a; 字符类&#xff08;只匹配一个字符&#xff09; 细节 预定义字符字符&#xff08;只匹配一个字符&#xff09; 细节 数量词 二、爬虫 Pattern Matcher 要点说明 一、正则表达式&#xff1a; 作用&#xff1a; 1、校验字符…

部署MinIO

一、安装部署MINIO 1.1 下载 wget https://dl.min.io/server/minio/release/linux-arm64/minio chmod x minio mv minio /usr/local/bin/ # 控制台启动可参考如下命令, 守护进程启动请看下一个代码块 # ./minio server /data /data --console-address ":9001"1.2 配…

Hive基础知识(十六):Hive-SQL分区表使用与优化

1. 分区表 分区表实际上就是对应一个 HDFS 文件系统上的独立的文件夹&#xff0c;该文件夹下是该分区所有的数据文件。Hive 中的分区就是分目录&#xff0c;把一个大的数据集根据业务需要分割成小的数据集。在查询时通过 WHERE 子句中的表达式选择查询所需要的指定的分区&…

通过myBatis将sql语句返回的值自动包装成一个java对象(2)

1.之前我们是如何执行一个sql语句自动包装成一个java对象呢&#xff1f; 1.创建一个mapper.xml&#xff0c;定义 执行的语句名字 和 包装成什么类 2.在总的配置文件里申明这个mapper 3.在java里通过sqlSession执行mapper里定义好的内容 我们还可以使用另一种方法实现第三步。现…

小程序中使用微信同声传译插件实现语音识别、语音合成、文本翻译功能----文本翻译(三)

官方文档链接&#xff1a;https://mp.weixin.qq.com/wxopen/plugindevdoc?appidwx069ba97219f66d99&token370941954&langzh_CN#- 要使用插件需要先在小程序管理后台的设置->第三方设置->插件管理中添加插件&#xff0c;目前该插件仅认证后的小程序。 文本翻译…

Vue项目 css下载字体并引入使用

1.下载字体 下载字体&#xff1a;字体下载,字体大全,免费字体下载,在线字体|字客网字客网是全球知名的字体下载与分享网站,齐全的中文,日文,韩文,英文,图标,美术设计,毛笔,钢笔,手写,书法字体大全,提供找字体,字体识别,字体下载,在线字体预览,字体转换,字体设计等服务。…

华为设备端口镜像设置

核心代码&#xff1a; observe-port int 编号 int 编号 mirror to observe-port both | inbound | outbound #both:将镜像端口的入和出流量同时复制到观察者端口 #inbound:将镜像端口的入流量复制到观察者端口 #outbound:将镜像端口的出流量复制到观察者端口配置后可使出入端口…

新手必看:腾讯云服务器购买详细图文教程

腾讯云服务器购买流程很简单&#xff0c;有两种购买方式&#xff0c;直接在官方活动上购买比较划算&#xff0c;在云服务器CVM或轻量应用服务器页面自定义购买价格比较贵&#xff0c;但是自定义购买云服务器CPU内存带宽配置选择范围广&#xff0c;活动上购买只能选择固定的活动…

基于SSM的项目监管系统设计与实现

末尾获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端框架&#xff1a;SSM 前端&#xff1a;采用JSP技术开发 数据库&#xff1a;MySQL5.7和Navicat管理工具结合 服务器&#xff1a;Tomcat8.5 开发软件&#xff1a;IDEA / Eclipse 是否Maven项目&#x…

基于SSM的驾校信息管理系统设计与实现

末尾获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端框架&#xff1a;SSM 前端&#xff1a;Vue、HTML 数据库&#xff1a;MySQL5.7和Navicat管理工具结合 服务器&#xff1a;Tomcat8.5 开发软件&#xff1a;IDEA / Eclipse 是否Maven项目&#xff1a;是…

自动化测试成本高效果差,意义在哪?

自动化测试的成本高效果差&#xff1f;首先这个结论就太过武断了一些。 任何技术都需要放到适合的地方去使用&#xff0c;否则一定是达不到理想的效果的。举例大炮打蚊子&#xff0c;同样是成本高效果差&#xff0c;难道大炮就没有存在的意义了吗&#xff1f; 当然不是&#…

3.0.0 网络安全技术

一、端口安全 1、端口隔离 1.1 简介 以太交换网络中为了实现报文之间的二层隔离&#xff0c;用户通常将*不同的端口*加入*不同的VLAN*&#xff0c;实现二层广播域的隔离。只通过VLAN实现报文二层隔离&#xff0c;会浪费有限的VLAN资源&#xff0c;同时也只能实现基础的隔离操…

金和OA jc6 Upload 任意文件上传漏洞复现

0x01 产品简介 金和OA协同办公管理系统软件(简称金和OA),本着简单、适用、高效的原则,贴合企事业单位的实际需求,实行通用化、标准化、智能化、人性化的产品设计,充分体现企事业单位规范管理、提高办公效率的核心思想,为用户提供一整套标准的办公自动化解决方案,以帮助…

移动端开发进阶之蓝牙通讯(二)

移动端开发进阶之蓝牙通讯&#xff08;二&#xff09; 蓝牙广播是一种无线通讯技术&#xff0c;通过无线电波传输数据&#xff1b; 在蓝牙低功耗&#xff08;BLE&#xff09;协议中&#xff0c;广播通信是其重要组成部分&#xff0c;主要有两类使用场景&#xff1a; 单一方向的…

基本BGP配置试验 :配置 IBGP 和 EBGP

一、预习&#xff1a; BGP&#xff1a;Border Gateway Protocol 没有精妙的算法&#xff0c;但能承载大量的路由&#xff0c;它不生产路由&#xff0c;它是路由的搬运工 使用TCP做为传输层协议&#xff0c;端口号179&#xff0c;使用触发式路由更新 1. BGP路由…

喜讯!矩阵起源子公司通过“国家高新技术企业”认定,引领数据库行业科技创新!

近日&#xff0c;全国高新技术企业认定管理工作领导小组办公室&#xff0c;公布了《上海市认定机构2023年认定报备的第二批高新技术企业备案公示名单》&#xff0c;矩阵起源子公司矩智原力&#xff08;上海&#xff09;信息科技有限公司&#xff08;以下简称“矩智原力”&#…