zoukankan      html  css  js  c++  java
  • bzoj4311向量(线段树分治+斜率优化)

    第二道线段树分治。

    首先设当前向量是(x,y),剩余有两个不同的向量(u1,v1)(u2,v2),假设u1>u2,则移项可得,若(u1,v1)优于(u2,v2),则-x/y>(v1-v2)/(u1-u2),然后维护上凸壳后进行三分即可,复杂度O(nlog2n),如果将询问排序扫一遍,可以优化到O(nlogn),当然我没写。

    #include<bits/stdc++.h>
    #define lson l,mid,rt<<1
    #define rson mid+1,r,rt<<1|1
    using namespace std;
    typedef long long ll;
    const int N=2e5+7;
    struct node{int x,y,l,r;}p[N],q[N];
    int n,tim,tot,top;
    ll ans[N];
    vector<node>a[N<<2];
    node st[N];
    bool cmp(node a,node b){return a.x==b.x?a.y<b.y:a.x<b.x;}
    ll dot(node a,node b){return 1ll*a.x*b.x+1ll*a.y*b.y;}
    ll cross(node a,node b,node c){return 1ll*(a.x-c.x)*(b.y-c.y)-1ll*(a.y-c.y)*(b.x-c.x);}
    void update(int L,int R,int id,int l,int r,int rt)
    {
        if(L<=l&&r<=R){a[rt].push_back(p[id]);return;}
        int mid=l+r>>1;
        if(L<=mid)update(L,R,id,lson);
        if(R>mid)update(L,R,id,rson);
    }
    ll query(int id)
    {
        int l=1,r=top,m1,m2;
        ll ret=0;
        while(r-l>=3)
        {
            m1=l+(r-l)/3,m2=r-(r-l)/3;
            if(dot(q[id],st[m1])<=dot(q[id],st[m2]))l=m1;else r=m2;
        }
        for(int i=l;i<=r;i++)ret=max(ret,dot(q[id],st[i]));
        return ret;
    }
    void work(int x,int l,int r)
    {
        if(!a[x].size())return;
        top=0;
        sort(a[x].begin(),a[x].end(),cmp);
        for(int i=0;i<a[x].size();i++)
        {
            while(top>1&&cross(st[top-1],st[top],a[x][i])>=0)top--;
            st[++top]=a[x][i];
        }
        for(int i=l;i<=r;i++)ans[i]=max(ans[i],query(i));
    }
    void divide(int l,int r,int rt)
    {
        work(rt,l,r);
        if(l==r)return;
        int mid=l+r>>1;divide(lson),divide(rson);
    }
    int main()
    {
        scanf("%d",&n);
        for(int i=1,op,x,y;i<=n;i++)
        {
            scanf("%d%d",&op,&x);
            if(op==1)scanf("%d",&y),p[++tot]=(node){x,y,tim+1,-1};
            else if(op==2)p[x].r=tim;
            else scanf("%d",&y),q[++tim]=(node){x,y,tim,tim};
        }
        for(int i=1;i<=tot;i++)if(p[i].r==-1)p[i].r=tim;
        for(int i=1;i<=tot;i++)if(p[i].l<=p[i].r)update(p[i].l,p[i].r,i,1,tim,1);
        divide(1,tim,1);
        for(int i=1;i<=tim;i++)printf("%lld
    ",ans[i]);
    }
    View Code
  • 相关阅读:
    JS产生随机数的几个用法!
    title与alt的区别
    jquery select取值,赋值操作
    DIV+CSS中标签dl dt dd常用的用法
    vi/vim键盘图
    win7系统注册表的权限修改
    win7 链接打印机时提示未知的用户名或错误密码
    关于无法把程序(Adobe Fireworks CS5)添加到打开方式的解决办法
    把网页发送到桌面代码
    iframe多层嵌套时获取元素总结
  • 原文地址:https://www.cnblogs.com/hfctf0210/p/11044327.html
Copyright © 2011-2022 走看看