题目链接:点击这里
题目大意:
俄罗斯轮盘赌,有一把左轮手枪,弹夹容量为
n
n
n ,现在要你往其中装
k
k
k 发子弹,旋转转轮,使得转轮的
n
n
n 种位置都是等概率的,然后自己先手开枪,两人轮流向自己开枪,活着的人获胜,
.
.
. 表示没子弹,
X
X
X 表示有子弹,题目求让自己存活几率最大的装弹方式下字典序最小的那一种字符串(
q
q
q 次询问,每次询问字符串的某一位是什么字符)
题目分析:
打表找规律…(我一开始表都打歪了,菜鸡本质暴露无遗) ,发现间隔着放最容易存活
可以先特判奇数,奇数与其比其小
1
1
1 的序列的差异就是奇数的比偶数的最后一位多一个
X
X
X
根据子弹数是否超过弹夹的一半分类,然后在确定哪一个位置有子弹,手动模拟几组打表的数据就会发现如代码中的规律
具体细节见代码:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<vector>
#include<set>
#include<map>
#define ll long long
#define inf 0x3f3f3f3f
#define int ll
using namespace std;
int read()
{
int res = 0,flag = 1;
char ch = getchar();
while(ch<'0' || ch>'9')
{
if(ch == '-') flag = -1;
ch = getchar();
}
while(ch>='0' && ch<='9')
{
res = (res<<3)+(res<<1)+(ch^48);//res*10+ch-'0';
ch = getchar();
}
return res*flag;
}
const int maxn = 2e5+5;
const int mod = 1e9+7;
const double pi = acos(-1);
const double eps = 1e-8;
int n,k,q;
bool check(int pos,int n,int k)
{
if(n&1)
{
if(pos == n && k)
return true;
n --;k--;
}
if(k > n/2)//子弹数超过弹夹容量的一半
{
if(pos >= n/2 || pos%2==0) return true; //偶数和超过弹夹一半的都有子弹
k -= n/2;
if((n-pos+1)/2 <= k) return true; //确定前一半位置中哪些有子弹
else return false;
}
else {
if(pos&1) return false; //奇数都没有子弹
if((n-pos)/2+1 <= k) return true; // 确定哪些偶数位置有子弹
else return false;
}
}
signed main()
{
// freopen("data.in","r",stdin);
// freopen("2.out","w",stdout);
n = read(),k = read(),q = read();
while(q--)
{
int pos = read();
if(check(pos,n,k)) putchar('X');
else putchar('.');
}
return 0;
}