LeetCode二分专题
二分
二分模板
两个模板:1.最大值最小模板一,2.最小值最大用模板二
单调性、两段性的性质

版本1:二分绿色端点是答案,最大值最小

int bsearch_1(int l, int r){
while (l < r){
int mid = l + r >> 1; //下取整
if (check(mid)) r = mid;
else l = mid + 1;
}
return l;
}
版本2:二分红色端点是答案,最小值最大

int bsearch_2(int l, int r){
while (l < r){
int mid = l + r + 1 >> 1; //上取整
if (check(mid)) l = mid;
else r = mid - 1;
}
return l;
}
模板2要+1后再下取整

二分最后l和r相等
模板小结:


69. x 的平方根
check性质:t的平方<=x
找最小值最大,模板2
注意精度:long long mid = (long long )(l + (long long ) r + 1) >> 1;

class Solution {
public:
bool check(long long mid,int x){
return mid*mid <= x;
}
int mySqrt(int x) {
int l = 0,r = x;
while (l < r){
long long mid = (long long )(l + (long long ) r + 1) >> 1;
if(check(mid,x)) l = mid;
else r = mid - 1;
}
return l;
}
};
35. 搜索插入位置
check性质:t>=x,即第一个比x大或等的位置
最大值最小,模板1

class Solution {
public:
bool check(int mid,int x){
return mid >= x;
}
int searchInsert(vector<int>& nums, int target) {
if(nums.size() == 0 || nums.back() < target) return nums.size();
int l = 0,r = nums.size()-1;
while(l<r){
long long mid = (long long )(l + r )>>1;
if(check(nums[mid],target)) r = mid;
else l = mid + 1;
}
return l;
}
};
34. 在排序数组中查找元素的第一个和最后一个位置
两个性质
1.check1:t>=x,找大于等于8的第一个位置,最大值最小,模板1

2.check2:t<=x,找小于等于8的最后1个位置,最小值最大,模板2

class Solution {
public:
vector<int> searchRange(vector<int>& nums, int target) {
if(nums.size() == 0) return {-1,-1};
int l = 0,r = nums.size() - 1;
while(l < r){
int mid = l + r >> 1;
if(nums[mid] >= target) r = mid;
else l = mid + 1;
}
if(nums[l] != target) return {-1,-1};
int start = l;
l = 0,r = nums.size() - 1;
while(l < r){
int mid = l + r + 1 >> 1;
if(nums[mid] <= target) l = mid;
else r = mid - 1;
}
return {start, l};
}
};
74. 搜索二维矩阵
最大值最小
或者最小值最大,找到目标即可
class Solution {
public:
bool searchMatrix(vector<vector<int>>& matrix, int target) {
if(matrix.size() == 0 || matrix[0].size() == 0) return false;
int n = matrix.size();
int m = matrix[0].size();
int l = 0 ,r = n * m - 1;
while(l < r){
int mid = l + r >> 1;
if(matrix[mid/m][mid%m] >= target) r = mid;
else l = mid + 1;
}
if(matrix[r/m][r%m] == target) return true;
return false;
}
};
153. 寻找旋转排序数组中的最小值
check性质,nums[mid] <= nums.back() ,且找最大(右半段)满足的最小下标

或者 nums[mid] < nums[0],且找最小值最大
class Solution {
public:
int findMin(vector<int>& nums) {
if(nums.size() == 0) return 0;
int l = 0,r = nums.size()-1;
while(l < r){
int mid = l + r >> 1;
if(nums[mid] <= nums.back()) r = mid;
else l = mid + 1;
}
return nums[r];
}
};
33. 搜索旋转排序数组
思路,按153思路,先找最小值,分成两段;然后按74题思路(最大值最小,且>=target)
两次二分,注意r--后的边界
class Solution {
public:
int search(vector<int>& nums, int target) {
if(nums.size() == 0) return -1;
int l = 0,r = nums.size()-1;
while(l < r){
int mid = l + r >> 1;
if(nums[mid] <= nums.back()) r = mid;
else l = mid + 1;
}
if(target <= nums.back()) r = nums.size()-1;
else l = 0,r--;
while(l < r){
int mid = l + r >> 1;
if(nums[mid] >= target) r = mid;
else l = mid + 1;
}
if(target == nums[l]) return l; //这里要用l 否则之前r-- r可能是-1
return -1;
}
};
278. 第一个错误的版本
这道题给了check函数,两段,第一个出错的版本只需要找最大值最小
注意溢出 long long mid = (long long )(l + r >> 1);

// The API isBadVersion is defined for you.
// bool isBadVersion(int version);
class Solution {
public:
int firstBadVersion(int n) {
long long l = 1 ,r = n;
while(l < r){
long long mid = (long long )(l + r >> 1);
if(isBadVersion(mid) == true) r = mid;
else l = mid + 1;
}
return r;
}
};
162. 寻找峰值
难度中等190 返回任何一个峰值所在位置即可。
方法1:线性扫描一遍
方法2:二分,比较mid 和 mid+1 如果mid+1的值比mid值大与等于那么mid+1后肯定有峰值,如果mid比mid+1小说明左边或者mid一定存在峰值
边界不需要判,分析:while循环里mid不会等于最后1个点,因为当mid=最后1个点,说明l=r都等于最后1个点,那么就退出循环了


class Solution {
public:
int findPeakElement(vector<int>& nums) {
int l = 0 , r = nums.size()-1;
while( l < r){
int mid = l + r >> 1;
if(nums[mid] < nums[mid+1]) l = mid + 1;
else r = mid;
}
return l;
}
};
287. 寻找重复数
只能使用额外的 O(1) 的空间。
时间复杂度小于 O(n^2) 。
二分答案,二分这个数值,O(N)遍历一遍数组统计在Lmid区间内的数的个数,如果个数比区间大小大那么说明有一个重复,重复的数就在这个Lmid闭区间中,让r=mid,否则在左区间(不包含mid)让L=mid+1
class Solution {
public:
int findDuplicate(vector<int>& nums) {
int n = nums.size() - 1;
int l = 1, r = n;
while( l < r){ //二分答案 二分这个数是多少
int mid = l + r >> 1;
int cnt = 0;
for(int i=0;i < nums.size();i++){
if(nums[i] >= l && nums[i] <= mid) cnt++;
}
if(cnt > mid - l + 1) r = mid;
else l = mid + 1;
}
return r;
}
};
275. H指数 II

满足单调性,二分

最小值最大;l最小为0,r最大为n = size个
h满足,h-1也一定满足,可以把区间分成两部分,前一部分红色一定满足,绿色不满足,所以就是找最小值最大

class Solution {
public:
int hIndex(vector<int>& citations) {
int n = citations.size() - 1;
int l = 0, r = citations.size();
while(l < r){
int mid = (l + r + 1) >> 1;
if(citations[n - mid + 1] >= mid) l = mid;
else r = mid - 1;
}
return l;
}
};
LeetCode二分专题的更多相关文章
- LeetCode刷题 二分专题
二分专题 二分的题目类型 对于满足二段性的题目的两套模板 模板一 模板如下 模板二 模板如下 解决二分题目的一般流程 LeeCode实战 LC69.x的平方根 解法思路 LC35.搜索插入位置 解法思 ...
- LeetCode 字符串专题(一)
目录 LeetCode 字符串专题 <c++> \([5]\) Longest Palindromic Substring \([28]\) Implement strStr() [\(4 ...
- LeetCode树专题
LeetCode树专题 98. 验证二叉搜索树 二叉搜索树,每个结点的值都有一个范围 /** * Definition for a binary tree node. * struct TreeNod ...
- leetcode二分查找问题整理
自从做完leetcode上的三道关于二分查找的题后,我觉得它是比链表找环还恶心的题,首先能写出bugfree代码的人就不多,而且可以有各种变形,适合面试的时候不断挑战面试者,一个程序猿写代码解决问题的 ...
- leetcode 二分查找
https://oj.leetcode.com/problems/search-for-a-range/就是一个二分查找,没事练练手 public class Solution { public in ...
- 学习笔记--APIO 2018 二分专题 By wuvin
前言: 在APIO 2018 Day2下午听wuvin讲二分,听了一上午的神仙,现在终于有可以听懂了. 专题: 平均边权最大 题目链接:https://www.questoj.cn/problem/3 ...
- Leetcode: 二分搜索法
package com.LeetCode; /** * 算法:二分搜索法查找一个值,并返回索引值 * https://leetcode.com/problems/search-insert-posit ...
- 算法leetcode二分算法
二分算法通常用于有序序列中查找元素: 有序序列中是否存在满足某条件的元素: 有序序列中第一个满足某条件的元素的位置: 有序序列中最后一个满足某条件的元素的位置. 思路很简单,细节是魔鬼. 一.有序序列 ...
- [leetcode]二分查找总结
Search for a Range 1.最简单的想法,用最普通的二分查找,找到target,然后向左右扩张,大量的重复的target,就会出现O(n)效率. class Solution { pub ...
随机推荐
- B2 - TV Subscriptions (Hard Version)
题目连接:https://codeforces.com/contest/1247/problem/B2 题解:双指针,,一个头,一个尾,头部进入,尾部退出,一开始先记录1到k,并记录每个数字出现的次数 ...
- Springboot:IDEA重调安装依赖窗口(二)
Settings-Plugins 搜索Editstarters: 安装完插件 重启idea: 查看安装是否成功: 在pom.xml 右键: 选择热部署依赖 点击ok进行自动装配: 热部署依赖环境已经配 ...
- 模糊字符串匹配:FuzzyWuzzy
FuzzyWuzzy 模糊字符串匹配,它使用Levenshtein Distance来计算简单易用的包中序列之间的差异. 前置条件 Python 2.7 or higher difflib pytho ...
- Youtube推荐算法的前世今生
第一阶段,基于User-Video图游历算法,2008年[1]. 在这个阶段,YouTube认为应该给用户推荐曾经观看过视频的同类视频,或者说拥有同一标签的视频.然而此时,YouTube的视频已是数千 ...
- Java 多线程实现方式一:继承Thread类
java 通过继承Thread类实现多线程很多简单: 只需要重写run方法即可. 比如我们分三个线程去京东下载三张图片: 1.先写个下载类: 注意导入CommonsIO 包 public class ...
- Jmeter与LoadRunner的比较
一.与Loadrunner的比较相似点 1.Jmeter的架构跟loadrunner原理一样, 都是通过中间代理,监控&收集并发客户端发现的指令,把他们生成脚本,再发送到应用服务器,再监控服务 ...
- 0day学习笔记(3)Windows定位API引起的惨案(原理)
段选择器FS与TEB WinNT内核下内存采用保护模式,段寄存器的意义与实模式汇编下的意义不同.另外,FS存的是段选择子,而不是实模式下的高16位基地址. FS寄存器指向当前活动线程的TEB结构(线程 ...
- 数组的forEach和map和for方法的区别
一.定义 foreach(): 从头到尾遍历数组,为每个元素调用指定的函数. map(): 将调用的数组的每个元素传递给指定的函数,并返回一个数组,他包含该函数的返回值. 传递的函数是 forea ...
- 2019-2020-1 20199310《Linux内核原理与分析》第四周作业
1.问题描述 在前面的文章中,已经接触过一些Linux内核的知识,本文将进一步从Linux内核源代码的目录结构入手,在Oracle VM VirtualBox的Linux环境中构造一个简单的操作系统M ...
- NumPy学习指南(第2版)
第一章 NumPy快速入门 首先,我们将介绍如何在不同的操作系统中安装NumPy和相关软件,并给出使用NumPy的简单示例代码. 然后,我们将简单介绍IPython(一种交互式shell工具). 如前 ...