51Nod 1530 稳定方块

本文介绍了一个关于博弈论的游戏问题,玩家需要轮流移除方块并确保剩余结构稳定,目标是最小化或最大化由方块编号组成的数值。文章提供了一种贪心算法的解决方案,并详细解释了实现细节。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

瓦西亚和皮台亚摆放了m个方块。方块被编号为0到m-1(每个号码出现恰好一次)。现在建立一个座标系OX表示地面,OY的方向是竖直向上的。每一方块的左下角有一个座标而且是整点座标。
摆放好的方块一定要是稳定的。稳定的含意是每一个不在地面上的方块在他的下面至少有一个方块与他相接触。可以是共边,也可以是共点的。也就是说如果方块座标为(x,y),要么y=0,或者存在一个方块的座标为(x-1,y-1)或者 (x,y-1) 或者 (x+1,y-1)。
现在瓦西亚和皮台亚要轮流把这些方块一个个拆下来。按照拆下来的顺序从左到右摆成一行,那么方块上面的编号就会组成一个m进制的数字。
拆的过程中,要始终保持剩下的方块稳定。瓦西亚想要最终的数字尽可能大,而皮台亚想要尽可能小,瓦西亚先开始拆。
请帮助计算一下最终形成的数字是多少,结果比较大,输出对 109+9 取余后的结果。

解题报告:
用时:1h10min,1WA1TLE
一开始认为就是开优先队列跑拓扑排序,后来发现度不为0也可以入队,所以只拿了60,然后我想到了正确贪心:
对于瓦西亚的从后往前枚举,直到出现第一个能消除的,皮台亚的同理.
然后打了这个贪心的暴力验证一下,发现是对的,考虑优化:
我们把所有可以消除的点丢入优先队列中,然后每次取出编号最小的,我们需要维护一个\(res[i]\),表示\(i\)最下面还有几个没有消除的点,然后我们检查一个点不合法我们就判断其上面的点是否\(res[i]<=1\),注意每消除一个点就要去更新上面点的\(res\)值,并且如果\(res[i]<=1\)时还要check他上方的点的下方的三个点是否会不合法,这样一个点最多入队三次,均摊复杂度\(O(nlogn)\)

#include <algorithm>
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <queue>
#include <cstdio>
#include <vector>
#include <cmath>
#define RG register
#define il inline
#define iter iterator
#define Max(a,b) ((a)>(b)?(a):(b))
#define Min(a,b) ((a)<(b)?(a):(b))
using namespace std;
typedef long long ll;
const int N=1e5+5,inf=1e9+5,mod=1e9+9;
int n;bool vis[N];
struct node{
    int x,y,id;
    bool operator <(const node &pp)const{
        if(y!=pp.y)return y<pp.y;
        return x<pp.x;
    }
}a[N];
struct comp{
    bool operator ()(int &i,int &j)const{
        return i>j;
    }
};
priority_queue<int>q;
priority_queue<int,vector<int>,comp>qm;
vector<int>s[N];
int b[N],m=0,num=0,head[N],to[N*3],nxt[N*3],du[N],re[N];
void link(int x,int y){nxt[++num]=head[x];to[num]=y;head[x]=num;}
bool check(int x){
    if(vis[x])return false;
    for(int i=head[x];i;i=nxt[i]){
        if(!vis[to[i]] && du[to[i]]<=1)return false;
    }
    return true;
}
bool ca[N];
void solve(){
    bool t=0;int x;
    for(int i=1;i<=n;i++){
        if(!t){
            while(!q.empty()){
                if(!ca[q.top()])q.pop();
                else break;
            }
            x=q.top();q.pop();
        }
        else{
            while(!qm.empty()){
                if(!ca[qm.top()])qm.pop();
                else break;
            }
            x=qm.top();qm.pop();
        }
        vis[x]=true;ca[x]=false;
        for(int k=0,sz=s[x].size(),u;k<sz;k++){
            u=s[x][k];
            if(check(u))ca[u]=true,qm.push(u);q.push(u);
        }
        for(int j=head[x];j;j=nxt[j]){
            du[to[j]]--;
            for(int k=0,sz=s[to[j]].size(),u;k<sz;k++){
                u=s[to[j]][k];
                if(!check(u))ca[u]=false;
                else{
                    ca[u]=true;qm.push(u);q.push(u);
                }
            }
        }
        re[i]=x-1;t^=1;
    }
    ll ans=0,mul=1;
    for(int i=n;i>=1;i--){
        ans+=mul*re[i];
        ans%=mod;
        mul*=n;mul%=mod;
    }
    printf("%lld\n",ans);
}
void work()
{
    scanf("%d",&n);
    for(int i=1;i<=n;i++){
        scanf("%d%d",&a[i].x,&a[i].y);
        a[i].id=i;
    }
    sort(a+1,a+n+1);
    for(int i=1;i<=n;i++)b[++m]=a[i].y;
    int sta;
    for(int i=1;i<=n;i++){
        sta=lower_bound(b+1,b+m+1,a[i].y-1)-b;
        for(int j=sta;j<i;j++){
            if(a[j].y!=a[i].y-1)break;
            if(abs(a[j].x-a[i].x)<=1){
                link(a[j].id,a[i].id);du[a[i].id]++;
                s[a[i].id].push_back(a[j].id);
            }
        }
    }
    for(int i=1;i<=n;i++){
        if(check(i))q.push(i),qm.push(i),ca[i]=true;
    }
    solve();
}

int main()
{
    work();
    return 0;
}

转载于:https://www.cnblogs.com/Yuzao/p/7524081.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值