随笔 - 55  文章 - 0  评论 - 12  阅读 - 62114

[QZOI2019]Game 题解

QZOI2019 CSP-S模拟赛 T1

错误的贪心导致考场上只有10pts... 看来以后贪心还是需要先证明啊

题目描述

小A和小B在玩一个游戏,他们两个人每人有 n 张牌,每张牌有一个点数,并且在接下来的 n 个回合中每回合他们两人会分别打出手中的一张牌,点数严格更高的一方得一分,然而现在小A通过某种神秘的方法得到了小B的出牌顺序,现在他希望规划自己的出牌顺序使得自己在得分尽可能高的前提下出牌的字典序尽可能大。

题意分析

给出一个数列b,给出一个相同长度的数列a,要求重新排列a,使得最多的ai>bi,输出字典序最大的一种方案。

思路分析

很自然地想到贪心。

考场上:字典序最大,那倒着最小不就得了(草我是什么sb),10pts

为什么是错的?实际上,如果在线输入在线构造数列a,有很多种情况会导致错误。

可以发现,得分与数列的顺序是没有关系的。因此我们完全可以先构造出最大得分,再来构造最大字典序。

具体实现

怎么构造最大得分?可以将ab都排序,然后降序遍历b,对于当前的b,如果当前的ai能得分,就得一分;否则继续遍历。

然后我们构造出一个得分方案。设想,构造出一个得分方案后,我们为了保证得分不变,不能再动已经定下来的这些牌;但我们可以用更大的牌来代替其中的一些牌,这样得分不变。因此我们先构造出一个最小的方案,这样剩下的大牌可以放到不得分的地方,也可以来替代得分的地方,可能性才多。当然,对于构造出的这个方案,我们也要使它在不改变牌组成的情况下,字典序最大。我们暂且将这些没有得分的大牌称为活动牌。

然后我们就开始从前往后遍历b序列,若当前元素处于得分方案内,看看有没有活动牌可以替代它,如果有,就用这张牌替代它,并让原来的方案牌成为活动牌;若当前元素不处于得分方案内,我们也可以从方案牌中拿一张最大的牌,当然,如果要拿牌,这张牌一定要可以得分,否则会导致得分减少,同时,因为这里会从方案牌中拿牌,处理得分方案内的牌时,要注意判断当前分配到的方案牌有没有提前被取走。

以上的算法可以通过multiset实现,部分步骤也可以用大根堆替代以降低复杂度,但比较复杂。

下面给出本文代码中用到的一些multiset操作的解释:(可以参考蓝书或网上资料)

multiset<int> s:定义一个有序可重集s
multiset<int>::iterator it:定义一个迭代器it,可以用“*”取消引用
s.insert(x):向s中插入数x
s.erase(it):从s中删除迭代器it指向的元素
s.find(x):返回s中指向x的迭代器
s.begin():返回s中指向第一个元素的迭代器
s.end():返回s中指向最后一个元素后一个地址的迭代器
s.rbegin():返回s中指向最后一个元素的反向迭代器,反向的意思是,若执行++操作,则会向s的前一位移动

考场上忘记函数怎么写怎么办?打开Dev-Cpp目录,搜索头文件set,找到对应的函数即可

复制代码
//贪心
//先保证得分最大,再保证字典序最大。可以先处理出最大得分的一种方案,然后再调整字典序到最大
//其它有的贪心算法可能会导致字典序或者得分顾此失彼,是错误的 
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<set>
#define id(i) w[i].id
#define v(i) w[i].v
using namespace std;
const int N=1e5+100;
struct B
{
    int id,v;
}w[N];
int n,cnt;
int b[N],ans[N];
multiset<int> s1,s2;
multiset<int>:: iterator it;
bool cmp(B x,B y)
{
    return x.v<y.v;
} 
int main()
{
    freopen("game.in","r",stdin);
    freopen("game.out","w",stdout);
    scanf("%d",&n);
    for(int i=1;i<=n;i++)
        scanf("%d",&b[i]),v(i)=b[i],id(i)=i;
    for(int i=1,x;i<=n;i++)
        scanf("%d",&x),s1.insert(x),s2.insert(x);
    sort(w+1,w+n+1,cmp);
    for(int i=n;i;i--)
        if(*s2.rbegin()>v(i))
            cnt++,s2.erase(s2.find(*s2.rbegin()));//计算最大得分 
    //先构造出一种出牌最小的方案,这样剩下的活动的牌就大,构造大字典序的空间就大 
    //本步过后,s1存储的就是方案外的活动牌
    //活动牌的定义是,不可以得分的,可以随意移动的牌 
    for(int i=1;i<=cnt;i++)
    {
        it=s1.upper_bound(v(i));
        ans[id(i)]=*it;s1.erase(s1.find(*it));
    }
    s2.clear();
    for(int i=1;i<=n;i++)
        if(ans[i])
            s2.insert(ans[i]);//s2存储已构造出的一种方案 
    for(int i=n;i;i--)
        if(ans[i])
        {
            it=s2.upper_bound(b[i]);
            ans[i]=*it,s2.erase(it);
        }//将s2的字典序弄到最大
    for(int i=1;i<=n;i++)
        if(ans[i])
            s2.insert(ans[i]);//再存回去 
    //不改变原方案的牌,为了使字典序最大,可以在活动牌和本来分配给它的牌中找最大的 
    for(int i=1;i<=n;i++)
        if(ans[i] && s2.find(ans[i])!=s2.end())//原方案已分配的部分 
            s1.insert(ans[i]),s2.erase(s2.find(ans[i])),printf("%d ",*s1.rbegin()),s1.erase(s1.find(*s1.rbegin()));
        else//没被分配到的部分 
        {
            if(s2.size() &&(*s1.rbegin()>b[i] || *s2.rbegin()>b[i]))//可以在s2中取个最大的拉过去得分 
                s1.insert(*s2.rbegin()),s2.erase(s2.find(*s2.rbegin()));
            printf("%d ",*s1.rbegin()),s1.erase(s1.find(*s1.rbegin()));
        }
    return 0;
}
复制代码
posted on   TEoS  阅读(489)  评论(0编辑  收藏  举报
编辑推荐:
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 现代计算机视觉入门之:什么是图片特征编码
· .NET 9 new features-C#13新的锁类型和语义
· Linux系统下SQL Server数据库镜像配置全流程详解
· 现代计算机视觉入门之:什么是视频
阅读排行:
· Sdcb Chats 技术博客:数据库 ID 选型的曲折之路 - 从 Guid 到自增 ID,再到
· .NET Core GC压缩(compact_phase)底层原理浅谈
· Winform-耗时操作导致界面渲染滞后
· Phi小模型开发教程:C#使用本地模型Phi视觉模型分析图像,实现图片分类、搜索等功能
· 语音处理 开源项目 EchoSharp
< 2025年1月 >
29 30 31 1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31 1
2 3 4 5 6 7 8

点击右上角即可分享
微信分享提示