首页 > 其他分享 >整体二分学习笔记

整体二分学习笔记

时间:2023-09-07 15:55:08浏览次数:43  
标签:二分 int 询问 mid 笔记 学习 MAXN 答案

有一些题目需要用到二分,但多次询问直接二分,会导致 TLE,那么就需要用到一个离线算法,将多个询问放在一起二分,这就是整体二分。

条件

能够用整体二分解决的题目需要满足以下性质:
1.题目具有可二分性(即单调性);
2.修改对判定答案的贡献相互独立,修改之间互不影响效果。
3.修改如果对判定答案有贡献,则贡献为一个确定的,与判定标准无关的值;
4.贡献满足交换律,结合律,具有可加性;
5.题目允许使用离线算法。

方法

记 \([l,r]\) 为答案的值域,\([L,R]\) 为答案的定义域。

  • 首先将所有操作按时间顺序存入数组, 开始分治;
  • 在每一层分治中,利用数据结构(一般是树状数组)统计当前查询的答案和 \(mid\) 之间的关系;
  • 根据查询出来的答案和 \(mid\) 间的关系(小于等于 \(mid\) 和大于 \(mid\))将当前处理的操作序列分为 \(q1\) 和 \(q2\) 两部分,分别递归处理;
  • 当 \(l=r\) 时,找到答案,记录答案并返回即可。
    需要注意的是,在整体二分的过程中,若当前处理的值域为 \([l,r]\),则此时最终答案范围不在 \([l,r]\) 的询问会在其他时候处理。

求全局第 \(k\) 小(多次询问)

可以对于每个询问进行一次二分;但是,也可以把所有的询问放在一起二分。

先考虑二分的本质:假设要猜一个 \([l,r]\) 之间的数,猜测答案是 \(m = \lfloor\frac{l + r}{2}\rfloor\),然后去验证 \(m\) 的正确性,再调整边界。这样做每次询问的复杂度为 \(\Theta(\log n)\),若询问次数为 \(q\),则时间复杂度为 \(\Theta(q\log n)\)。

回过头来,对于当前的所有询问,可以去猜测所有询问的答案都是 \(mid\),然后去依次验证每个询问的答案应该是小于等于 \(mid\) 的还是大于 \(mid\) 的,并将询问分为两个部分,对于每个部分继续二分。

注意:如果一个询问的答案是大于 \(mid\) 的,则在将其划至右侧前需更新它的 \(k\),即,如果当前数列中小于等于 \(mid\) 的数有 \(t\) 个,则将询问划分后实际是在右区间询问第 \(k - t\) 小数。如果一个部分的 \(l = r\) 了,则结束这个部分的二分。利用线段树的相关知识,我们每次将整个答案可能在的区间 \([1,maxans]\) 划分成了若干个部分,这样的划分共进行了 \(\Theta(\log maxans)\) 次,一次划分会将整个操作序列操作一次。若对整个序列进行操作,并支持对应的查询的时间复杂度为 \(\Theta(T)\),则整体二分的时间复杂度为 \(\Theta(T\log n)\)。

求区间第 \(k\) 小(多次询问)

涉及到给定区间的查询,再按之前的方法进行二分就会导致 check 函数的时间复杂度爆炸。仍然考虑询问与值域中点 \(m\) 的关系:若询问区间内小于等于 \(m\) 的数有 \(t\) 个,询问的是区间内的 \(k\) 小数,则当 \(k \leqslant t\) 时,答案应小于等于 \(m\);否则,答案应大于 \(m\)。

此处需记录一个区间小于等于指定数的数的数量,即单点加,求区间和,可用树状数组快速处理。为提高效率,只对数列中值在值域区间 \([l,r]\) 的数进行统计,即,在进一步递归之前,不仅将询问划分,将当前处理的数按值域范围划为两半。

例题 P3834【模板】可持久化线段树 2

点击查看代码
#include<bits/stdc++.h>
using namespace std;
const int MAXN=4e5+5;
const int INF=0x7f7f7f7f;

struct Query
{
    int l,r,k,id,type;
}q[MAXN],q1[MAXN],q2[MAXN];

int t[MAXN],ans[MAXN];
int n,m,cnt;

inline int lowbit(int x) {return x&-x;}

inline void add(int x,int k)
{
    for(int i=x;i<=MAXN;i+=lowbit(i)) t[i]+=k;
    return;
}

inline int ask(int x)
{
    int res=0;
    for(int i=x;i>=1;i-=lowbit(i)) res+=t[i];
    return res;
}

inline void solve(int l,int r,int ql,int qr)
{
    if(ql>qr) return;
    if(l==r)
    {
        for(int i=ql;i<=qr;i++)
            if(q[i].type==2) ans[q[i].id]=l;
        return;
    }
    int mid=(l+r)>>1,cnt1=0,cnt2=0;
    for(int i=ql;i<=qr;i++)
    {
        if(q[i].type==1)
        {
            if(q[i].l<=mid)
            {
                add(q[i].id,1);
                q1[++cnt1]=q[i];
            }
            else q2[++cnt2]=q[i];
        }
        else
        {
            int res=ask(q[i].r)-ask(q[i].l-1);
            if(res>=q[i].k) q1[++cnt1]=q[i];
            else q[i].k-=res,q2[++cnt2]=q[i];
        }
    }
    for(int i=1;i<=cnt1;i++)
        if(q1[i].type==1) add(q1[i].id,-1);
    for(int i=1;i<=cnt1;i++) q[ql+i-1]=q1[i];
    for(int i=1;i<=cnt2;i++) q[ql+i+cnt1-1]=q2[i];
    solve(l,mid,ql,ql+cnt1-1),solve(mid+1,r,ql+cnt1,qr);
    return;
}


int main()
{
    ios_base::sync_with_stdio(false);
    cin.tie(0),cout.tie(0);
    cin>>n>>m;
    for(int i=1;i<=n;i++)
    {
        int x; cin>>x;
        q[++cnt]={x,x,0,i,1};
    }
    for(int i=1;i<=m;i++)
    {
        int l,r,k; cin>>l>>r>>k;
        q[++cnt]={l,r,k,i,2};
    }
    solve(-INF,INF,1,cnt);
    for(int i=1;i<=m;i++) printf("%d\n",ans[i]);
    return 0;
}

标签:二分,int,询问,mid,笔记,学习,MAXN,答案
From: https://www.cnblogs.com/code-ac/p/17685163.html

相关文章

  • 在小红书上学习专业摄影后期处理技巧
    嘿,小红书的朋友们!今天我要和大家分享一些在小红书上学习专业摄影后期处理技巧的秘诀。作为一个热爱摄影的人,我深知后期处理对于一张照片的重要性。它能够让平凡的照片变得生动有趣,甚至让你的朋友都惊叹不已!下面,就让我来带领大家一起进入这个色彩斑斓的后期处理世界吧!1.选择合......
  • 【爬虫笔记】Python爬虫简单运用爬取代理IP
    一、前言近些年来,网络上的爬虫越来越多,很多网站都针对爬虫进行了限制,封禁了一些不规则的请求。为了实现正常的网络爬虫任务,爬虫常用代理IP来隐藏自己的真实IP,避免被服务器封禁。本文将介绍如何使用Python爬虫来获取代理IP,以及如何在爬虫中使用代理IP。二、获取代理IP获取代理IP有两......
  • 趣味微项目:玩转Python编程,轻松学习快乐成长!
    ......
  • Acegi-security-samples-tutorial-1.0.7.zip 实例学习笔记
    eclipse环境下新建一个webproject项目AcegiTest,把Acegi-security-samples-tutorial-1.0.7.zip中的代码放入项目中的对应目录下:IE地址栏输入:http://localhost/AcegiTest/转到index.jsp 页面: 主页任何人可浏览此页面。安全页面超级安全页面 (1)点“安全页面”则进入登录页......
  • Heritrix架构学习笔记(三)
    3、Frontier链接制造工厂在heritrix-1.12.1/docs/articles/developer_manual/frontier.html下可找到Heritrix的官方文档的一个Frontier例子:/***AsimpleFrontierimplementationfortutorialpurposes*/publicclassMyFront......
  • Heritrix架构学习笔记(一)
    1、抓取起点CrawlOrder在heritrix-1.12.1/docs/apidocs目录下可以查看其API:org.archive.crawler.datamodelClassCrawlOrderjava.lang.Objectjavax.management.Attributeorg.archive.crawler.settings.Typeorg.archive.crawler.settings.Complex......
  • [个人笔记][C#]异步调用控制流的一些测试结论
    await调用逻辑总结如下:调用线程A执行到await时,在await处返回并继续执行调用点后面的代码,await处新开一个线程B执行task线程B执行完task后继续执行await后面的代码如果再次遇到await,线程B在await处返回,新开一个线程C执行task线程C执行完task后继续执行await后面的代码"新开......
  • sqlserver移植为Oracle笔记(更新,新增字段名;批量新增记录;日期查询;截取字串函数)
    下面是这两天在项目要sqlserver和oracle兼容的改造中测试出来的笔记:--sqlserver--更改主键字段名'ID'为'ID_'sp_rename  'tb_doc_cat_statistic.ID','ID_','column'--新增字段cat_codealtertabletb_doc_cat_statisticaddcat_codevarchar(100) --oracle--......
  • Informer模型学习记录
    Informer时间序列模型data1.WTH.csv水厂csv格式数据,总共13列,包含一列标签,12列特征,后面输入模型维度:12每隔一小时一条记录每个时间点对应多个特征,最后一个数据值作为数据标签2.ECL.csvcsv格式数据3.data_loadercols=list(df_raw.columns);cols.re......
  • 【刷题笔记】
    题目Givenacollectionofcandidatenumbers(candidates)andatargetnumber(target),findalluniquecombinationsin candidates wherethecandidatenumberssumsto target.Eachnumberin candidates mayonlybeused once inthecombination.Note:All......