NOIP 2014 试做

rps生活大爆炸版石头剪刀布
题目描述 Description
石头剪刀布是常见的猜拳游戏:石头胜剪刀,剪刀胜布,布胜石头。如果两个人出拳一样,则不分胜负。在《生活大爆炸》第二季第8集中出现了一种石头剪刀布的升级版游戏。升级版游戏在传统的石头剪刀布游戏的基础上,增加了两个新手势:
斯波克:《星际迷航》角之一。 蜥蜴人:《星际迷航》中的反面角色。
这五种手势的胜关系如表一所示,表中列出的是甲对乙的游戏结果。
现在,小A和小B尝试玩这种升级版的猜拳游戏。已知他们的出拳都是有周期性规律的,但周期长度不一定相等。例如:如果小A以“石头-布-石头-剪刀-蜥蜴人-斯波克”长度为6的周期出拳,那么他的出拳序列就是“石头-布-石头-剪刀-蜥蜴人-斯波克-石头-布-石头-剪刀-蜥蜴人-斯波克-„„”,而如果小B以“剪刀-石头-布-斯波克-蜥蜴人”长度为5的周期出拳,那么他出拳的序列就是“剪刀-石头-布-斯波克-蜥蜴人-剪刀-石头-布-斯波克-蜥蜴人-„„”
已知小A和小B一共进行N次猜拳。每一次赢的人得1分,输的得0分;平局两人都得0分。现请你统计N次猜拳结束之后两人的得分。

输入描述 Input Description
输入文件名为rps.in。

第一行包含三个整数:N,NA,NB,分 别 表 示 共 进 行N次猜拳、小A出拳的周期长度,小B出拳的周期长度。数与数之间以一个空格分隔。

第二行包含NA个整数,表示小A出拳的规律,第三行包含NB个整数,表示小B出拳的规律。其中,0表示“剪刀”,1表示“石头”,2表示“布”,3表示“蜥蜴人”, 4表示“斯波克”。数与数之间以一个空格分隔。

输出描述 Output Description
输出文件名为rps.out。

输出一行, 包含两个整数,以一个空格分隔,分别表示小A、小B的得分。

数据范围很小…
简单的模拟(人懒勿喷)

#include<cstdio>
using namespace std;
const int maxn=500+500;
int aa[maxn];
int bb[maxn];
int ans;
int bns;
int main()
{

    int n,na,nb;
    scanf("%d%d%d",&n,&na,&nb);
    for(int i=1;i<=na;i++)
    {
        scanf("%d",&aa[i]);
    }
    for(int i=1;i<=nb;i++)
        scanf("%d",&bb[i]);
    for(int i=1;i<=n;i++)
    {
        aa[i+na]=aa[i];
        bb[i+nb]=bb[i];
    }
    for(int i=1;i<=n;i++)
    {
        if(aa[i]==bb[i])
            continue;
        if(aa[i]==0)
        {
            if(bb[i]==1)
                bns++;
            else if(bb[i]==2)
                ans++;
            else if(bb[i]==3)
                ans++;
            else if(bb[i]==4)
                bns++;

        }//
        else if(aa[i]==1)
        {
            if(bb[i]==0)
                ans++;
            else if(bb[i]==2)
                bns++;
            else if(bb[i]==3)
                ans++;
            else if(bb[i]==4)
                bns++;
        }//
        else if(aa[i]==2)
        {
            if(bb[i]==0)
                bns++;
            else if(bb[i]==1)
                ans++;
            else if(bb[i]==3)
                bns++;
            else if(bb[i]==4)
                ans++;
        }
        else if(aa[i]==3)
        {
            if(bb[i]==0)
                bns++;
            else if(bb[i]==1)
                bns++;
            else if(bb[i]==2)
                ans++;
            else if(bb[i]==4)
                ans++;
        }
        else if(aa[i]==4)
        {
            if(bb[i]==0)
                ans++;
            else if(bb[i]==1)
                ans++;
            else if(bb[i]==2)
                bns++;
            else if(bb[i]==3)
                bns++;
        }
    }
    printf("%d %d",ans,bns);
    return 0;
}

link联合权值
见本博客

bird 飞扬的kotori
【问题᧿述】
Flappy Bird 是一款风靡一时的休闲手机游戏。玩家
需要不断控制点击手机屏幕的频率来调节小鸟的飞行高度,让
小鸟顺利通过画面右方的管道缝隙。如果小鸟一不小心撞到了
水管或者掉在地上的话,便宣告失败。
为了简化问题,我们对游戏规则进行了简化和改编:
1. 游戏界面是一个长为n,高 为m的二维平面,其中有
k个管道(忽略管道的宽度)。
2. 小鸟始终在游戏界面内移动。小鸟从游戏界面最左边
任意整数高度位置出发,到达游戏界面最右边时,游
戏完成。
3. 小鸟每个单位时间沿横坐标方向右移的距离为1,竖直移动的距离由玩家控制。如
果点击屏幕,小鸟就会上升一定高度X,每个单位时间可以点击多次,效果叠加;
如果不点击屏幕,小鸟就会下降一定高度Y。小鸟位于横坐标方向不同位置时,上
升的高度X和下降的高度Y可能互不相同。
4. 小鸟高度等于0或者小鸟碰到管道时,游 戏 失 败 。小 鸟 高 度 为m时,无法再上升。
现在,请你判断是否可以完成游戏。如果可以,输出最少点击屏幕数;否则,输出小鸟
最多可以通过多少个管道缝隙。
【输入】
输入文件名为 bird.in。
第1行有3个整数n,m,k,分别表示游戏界面的长度,高度和水管的数量,每两个
整数之间用一个空格隔开;
接下来的n行,每行2个用一个空格隔开的整数X和Y,依次表示在横坐标位置0~n-1
上玩家点击屏幕后,小鸟在下一位置上升的高度X,以及在这个位置上玩家不点击屏幕时,
小鸟在下一位置下降的高度Y。
接下来k行,每行3个整数P,L,H,每两个整数之间用一个空格隔开。每行表示一
个管道,其中P表示管道的横坐标,L表示此管道缝隙的下边沿高度为L,H表示管道缝隙
上边沿的高度(输入数据保证P各不相同,但不保证按照大小顺序给出)。
【输出】
输出文件名为bird.out。
共两行。
第一行,包含一个整数,如果可以成功完成游戏,则输出1,否则输出0。
第二行,包含一个整数,如果第一行为1,则输出成功完成游戏需要最少点击屏幕数,
否则,输出小鸟最多可以通过多少个管道缝隙。
【数据范围】
对于30%的数据:5≤n≤10,5≤m≤10,k=0,保证存在一组最优解使得同一单位时间最多
点击屏幕3次;
对于50%的数据:5≤n≤20,5≤m≤10,保证存在一组最优解使得同一单位时间最多点击屏
幕3次;
对于70%的数据:5≤n≤1000,5≤m≤100;
对于100%的数据:5≤n≤10000,5≤m≤1000,0≤k< n,0< X< m,0< Y< m,0< P< n,0≤L< H ≤ m,
L+1< H。

只打了35分爆搜….

#include<algorithm>
#include<cstring>
#include<cstdio>
#define mem(a) memset(a,0,sizeof(a))
#define fk puts("jhaa");
using namespace std;
const int maxn=10000+5;
int n,m,k;
struct meico
{
    int up;
    int fl;
}sz[maxn];
int mapp[maxn][1005];
bool flag;
int ans=0x7fffffff;
int farr=0;
void dfs1(int num,int hi,int cnt)
{
    if(num==n)
    {
        flag=1;
        ans=min(ans,cnt);
        return;
    }
    farr=max(farr,num);
    int hah;
    if(!sz[num].up)
        hah=0;
    else
        hah=m/(sz[num].up)+1;
    for(int i=0;i<=hah;i++)     
    {
        meico mm=sz[num];
        if(!i)
        {
            if(hi-mm.fl>0)
                dfs1(num+1,hi-mm.fl,cnt);
        }
        else
        {
            int tmp=hi+i*mm.up;
            if(tmp>m)
                tmp=m;
            dfs1(num+1,tmp,cnt+i);
        }
    }
}
void solve1()
{
    for(int i=0;i<n;i++)
        scanf("%d%d",&sz[i].up,&sz[i].fl);

    dfs1(0,m,0);
    if(flag)
        printf("1\n%d",ans);
    else
        printf("0\n",farr);
}
int guanzi[maxn];
int qzh[maxn];
void dfs2(int num,int hi,int cnt)
{
    if(cnt>=ans)
        return;
    if(num==n)
    {
        flag=1;
        ans=min(ans,cnt);
        return;
    }
    farr=max(farr,qzh[num]);
    int hah;
    if(!sz[num].up)
        hah=0;
    else
        hah=m/(sz[num].up)+1;
    for(int i=0;i<=hah;i++)     
    {
        meico mm=sz[num];

        if(!i)
        {
            if(hi-mm.fl>0)
            if(!mapp[num+1][hi-mm.fl])
                dfs2(num+1,hi-mm.fl,cnt);
        }
        else
        {
            int tmp=hi+i*mm.up;
            if(tmp>m)
                tmp=m;
            if(!mapp[num+1][tmp])
            {
                dfs2(num+1,tmp,cnt+i);
            }
        }
    }
}

void solve2()
{
    mem(mapp);
    for(int i=0;i<n;i++)
        scanf("%d%d",&sz[i].up,&sz[i].fl);
    while(k--)
    {
        int a,b,c;
        scanf("%d%d%d",&a,&b,&c);
        for(int i=m;i>=c;i--)
            mapp[a][i]=1;
        for(int i=0;i<=b;i++)
            mapp[a][i]=1;
        guanzi[a]++;
    }
    for(int i=1;i<=n;i++)
    {
        if(!guanzi[i])
            qzh[i]+=qzh[i-1];
        else
            qzh[i]+=qzh[i-1],qzh[i+1]++;
    }
    for(int i=m;i>=0;i--)
        if(!mapp[1][i-sz[0].fl])
            dfs2(0,i,0); 

    if(flag)
        printf("1\n%d",ans);
    else
        printf("0\n%d",farr);
}
int main()
{
    scanf("%d%d%d",&n,&m,&k);
    if(k==0)
        solve1();
    else if(n<=20&&m<=10)
        solve2();
    return 0;
}

wireless无线网络发射选址
题目描述 Description
随着智能手机的日益普及,人们对无线网的需求日益增大。某城市决定对城市内的公共场所覆盖无线网。

假设该城市的布局为由严格平行的129条东西向街道和129条南北向街道所形成的网格状,并且相邻的平行街道之间的距离都是恒定值1。东西向街道从北到南依次编号为0,1,2…128,南北向街道从西到东依次编号为0,1,2…128。

东西向街道和南北向街道相交形成路口,规定编号为x的南北向街道和编号为y的东西向街道形成的路口的坐标是(x, y)。 在 某 些 路 口 存 在 一 定 数 量 的 公 共 场 所 。

由于政府财政问题,只能安装一个大型无线网络发射器。该无线网络发射器的传播范围是一个以该点为中心,边长为2*d的正方形。传播范围包括正方形边界。

例如下图是一个d = 1的无线网络发射器的覆盖范围示意图。
现在政府有关部门准备安装一个传播参数为d的无线网络发射器,希望你帮助他们在城市内找出合适的安装地点,使得覆盖的公共场所最多。
输入描述 Input Description
输入文件名为wireless.in。

第一行包含一个整数d,表示无线网络发射器的传播距离。

第二行包含一个整数n,表示有公共场所的路口数目。

接下来n行,每行给出三个整数x, y, k, 中间用一个空格隔开,分别代表路口的坐标(x, y)以及该路口公共场所的数量。同一坐标只会给出一次。

输出描述 Output Description
输出文件名为wireless.out。
输出一行,包含两整数,用一个空格隔开,分别表示能覆盖最多公共场所的安装地点方案数,以及能覆盖的最多公共场所的数量。
数据范围及提示 Data Size & Hint
对于100%的数据,1 ≤ d ≤ 20,1 ≤ n ≤ 20, 0 ≤ x ≤ 128, 0 ≤ y ≤ 128, 0 < k ≤ 1,000,000。

简单的模拟

#include<cstdio>
using namespace std;
typedef long long ll;
ll d,n;
ll wifi[1230][1230];
ll deal(ll x,ll y)
{
    ll ans=0;
    for(ll i=x-d;i<=x+d;i++)
    {
        for(ll j=y-d;j<=y+d;j++)
        {
            ans+=wifi[i][j];
        }
    }
    return ans;
}
int main()
{
    ll tot=0,maxans=0;
    scanf("%lld%lld",&d,&n);
    for(ll i=1;i<=n;i++)
    {
        ll x,y,k;
        scanf("%lld%lld%lld",&x,&y,&k);
        wifi[x+30][y+30]=k;
    }
    for(ll i=0+30;i<=128+30;i++)
    {
        for(ll j=0+30;j<=128+30;j++)
        {
            ll ans=deal(i,j);
            if(ans>maxans)
            {
                tot=1,maxans=ans;
                continue;
            }
            if(ans==maxans)
                tot++;
        }
    }
    printf("%lld %lld",tot,maxans);
    return 0;
}

road寻找道路
见本博客

equation解方程
题目描述 Description
已知多项式方程:
a0+a1*x+a2*x^2+…+an*x^n=0
输入描述 Input Description
输入文件名为equation.in。
输入共n+2行。
第一行包含2个整数n、m,每两个整数之间用一个空格隔开。
接下来的n+1行每行包含一个整数,依次为a0,a1,a2,……,an。
输出描述 Output Description
输出文件名为equation.out。
第一行输出方程在[1, m]内的整数解的个数。
接下来每行一个整数,按照从小到大的顺序依次输出方程在[1, m]内的一个整数解。
数据范围及提示 Data Size & Hint
对于30%的数据,0< n≤ 2|ai|≤100an≠0m≤100
30分暴力…

#include<cstdio>
using namespace std;
const int maxn=10000;
int n,m;
int xs[maxn];
int jie[maxn];
int tot=0;
int ksm(int x,int y)
{
    if(!y)  return 1;
    if(y==1)    return x;
    int hah=ksm(x,y>>1);
    if(y&1)
        return hah*hah*x;
    return hah*hah;
}
void deal(int x)
{
    int ans=0;
    for(int i=0;i<=n;i++)
        ans+=(i[xs]*ksm(x,i));
    if(!ans)
        tot++,tot[jie]=x;
}
int main()
{
    scanf("%d%d",&n,&m);
    for(int i=0;i<=n;i++)
        scanf("%d",&i[xs]);
    for(int i=1;i<=m;i++)
        deal(i);

    printf("%d\n",tot);
    for(int i=1;i<=tot;i++)
        printf("%d\n",i[jie]);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值