策略游戏

P8818 [CSP-S 2022] 策略游戏

以下的分析,定义正数 0,负数 0

我们发现,如果第一个人取了正数,第二个人如果有负数,那么就取绝对值最大的负数,即最小的数;如果没有,就取最小的正数,也是最小的数。

如果第一个人取了负数,第二个人如果有正数,那么就取最大的正数,即最大的数;如果没有,就取绝对值最小的负数,也是最大的数。

于是,第二个人的最优策略只有最大值或者最小值。

对于第一个人,我们发现他只可能取最小的正数、最大的负数、最小值、最大值四者其一。

所以我们只要枚举第一个人的四选一,第二人的二选一,然后判断一下第一个人是否存在最小的正数、最大的负数即可。

第一个人要 4 个最值,第二个人 2 个,于是开了 6ST 表维护。

时间复杂度 O(nlogn+mlogm+q)

#include<cstdio>
#include<algorithm>
using namespace std;
#define Ed for(int i=h[x];~i;i=ne[i])
#define Ls(i,l,r) for(int i=l;i<r;++i)
#define Rs(i,l,r) for(int i=l;i>r;--i)
#define Le(i,l,r) for(int i=l;i<=r;++i)
#define Re(i,l,r) for(int i=l;i>=r;--i)
#define L(i,l) for(int i=0;i<l;++i)
#define E(i,l) for(int i=1;i<=l;++i)
#define W(t) while(t--)
#define Wh while
typedef long long ll;

const int N=100010,M=17,INF=1e9+1;
int n,m,q,a[N],b[N],LOG[N],amx[M][N],amn[M][N],bmx[M][N],bmn[M][N],fmx[M][N],zmn[M][N];
void init(int mn[][N],int mx[][N],int n,int a[]){
    E(i, n)mn[0][i]=mx[0][i]=a[i];
    E(i, LOG[n])
        for(int j=1;j+(1<<i)-1<=n;++j){
            mn[i][j]=min(mn[i-1][j],mn[i-1][j+(1<<i-1)]);
            mx[i][j]=max(mx[i-1][j],mx[i-1][j+(1<<i-1)]);
        }
}
void Init(){
    E(i, n){
        fmx[0][i]=a[i]<=0?a[i]:-INF;
        zmn[0][i]=a[i]>=0?a[i]:INF;
    }
    E(i, LOG[n])
        for(int j=1;j+(1<<i)-1<=n;++j){
            zmn[i][j]=min(zmn[i-1][j],zmn[i-1][j+(1<<i-1)]);
            fmx[i][j]=max(fmx[i-1][j],fmx[i-1][j+(1<<i-1)]);
        }
}
int qmax(int f[][N],int l,int r){
    int k=LOG[r-l+1];
    return max(f[k][l],f[k][r-(1<<k)+1]);
}
int qf(int l,int r){
    int k=LOG[r-l+1];
    return max(fmx[k][l],fmx[k][r-(1<<k)+1]);
}
int qz(int l,int r){
    int k=LOG[r-l+1];
    return min(zmn[k][l],zmn[k][r-(1<<k)+1]);
}
int qmin(int f[][N],int l,int r){
    int k=LOG[r-l+1];
    return min(f[k][l],f[k][r-(1<<k)+1]);
}
int main(){
    #ifndef ONLINE_JUDGE
    freopen("1.in","r",stdin);
    #endif
    scanf("%d%d%d",&n,&m,&q);
    int lim=max(n,m);
    Le(i, 2, lim)LOG[i]=LOG[i>>1]+1;
    E(i, n)scanf("%d",a+i);
    E(i, m)scanf("%d",b+i);
    init(amn,amx,n,a);
    init(bmn,bmx,m,b);
    Init();
    W(q){
        int l1,r1,l2,r2;
        scanf("%d%d%d%d",&l1,&r1,&l2,&r2);
        ll aa[]={qz(l1,r1),qf(l1,r1),qmax(amx,l1,r1),qmin(amn,l1,r1)};
        ll bb[]={qmax(bmx,l2,r2),qmin(bmn,l2,r2)};
        ll ans=-2e18;
        L(i, 4)
            if(i>1||aa[i]!=INF&&aa[i]!=-INF)
                ans=max(ans,min(aa[i]*bb[0],aa[i]*bb[1]));
        printf("%lld\n",ans);
    }
    return 0;
}

本文作者:wscqwq

本文链接:https://www.cnblogs.com/wscqwq/p/17729845.html

版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。

posted @   wscqwq  阅读(16)  评论(0编辑  收藏  举报
点击右上角即可分享
微信分享提示
评论
收藏
关注
推荐
深色
回顶
收起