蓝桥杯_left and throw

思考了许久没有结果,最后,还是一位擅长搜索资源的学长帮我找到了一个不错的代码,这个代码极其精妙,再一次印证了一句话,没有做不到的,只有想不到的,当然这个代码我拿到手的时候是个没有注释的代码,我费尽周折才从本质解读了这段代码的算法(众所周知,越是精妙的算法,可读性越差,当然有没有注释也会有很大的差距)。
接下来,就该先分享一下代码了:
#include <stdio.h>
#include <string.h>
#define TRUE 1
#define FALSE 0
#define max(a, b) a > b ? a : b
//定义数组大小为4,从一开始,空出下标为0,方便计算
int x[4];                  //三个人的位置
int l[4];                  //三个人的机动性(可移动距离)
int t[4];                  //三个人的抛的距离
int ans = 0;               //经过操作后的最远距离,初始化为0
int w[4];                  //初始化为0,0表示可以进行操作,非零表示不可以
int p[4];                  //初始化为0,表示a[i]所举起的人
int a[4] = {3, 3, 3, 3};   //初始化为3,表人的状态,这里a对应的二进制为0011,后三位分别是三个动作:抛出,举起,移动。0(无意义)0(不可抛出)1(未进行举起)1(未进行移动)。这道题中,a只有六个可能值:0(0000)、1(0001)、2(0010)、3(0011)、4(0100)、5(0101),表示人的六种状态
//bool类型
int near(int s)
{
    int i = 1;
    for (; i <= 3; i++)
    {
        if (s == x[i] + 1 || s == x[i] - 1)
        {
            return TRUE;
        }
    }
    return FALSE;
}
//dfs深度遍历
void dfs(int d)
{
    int i = 1, j = 1, e = 0;
    //每次都取最远(大)的位置
    for (; i <= 3; i++)
    {
        ans = max(ans, x[i]);
    }
    for (i = 1; i <= 3; i++)
    {
        //是否可以进行操作
        if (w[i])
        {
            continue;
        }
        //a[i] == 1 || a[i] == 3(未进行移动且不可抛出)
        if ((a[i] & 1) && !(a[i] & 4))
        {
            for (j = 1; j <= l[i]; j++)                         //移动
            {
                x[i] += j;                                      //a[i]向前移动j
                a[i] ^= 1;                                      //已移动
                if (near(x[i]) || j == l[i])                    //如果a[i]移动后的位置旁边有人或者移动距离达到上限
                {
                    dfs(d + 1);
                }
                x[i] -= j;                                      //归位
                x[i] -= j;                                      //a[i]向后移动j
                if (near(x[i]) || j == l[i])                    //如果a[i]移动后的位置旁边有人或者移动距离达到上限
                {
                    dfs(d + 1);
                }
                x[i] += j;                                      //归位
                a[i] ^= 1;                                      //还原为未移动
            }
        }
        //a[i] == 2 || a[i] == 3 || a[i] == 5(未进行举起)
        if (a[i] & 2)
        {
            for (j = 1; j <= 3; j++)                            //举起
            {
                if (i != j && !w[j] && t[i] > 0)                //是否可以进行操作
                {
                    if (x[i] == x[j] + 1 || x[j] == x[i] + 1)   //a[i]附近是否有人
                    {
                        w[j] = 1;                               //即将举起(抛出)j,抛出前将j是否可操作标记变更为否
                        a[i] ^= 2;                              //已举起
                        a[i] ^= 4;                              //可抛出
                        p[i] = j;                               //记录a[i]举起(抛出)了j
                        e = x[j];                               //记录a[j]的举起前位置
                        x[j] = -j;                              //a[j](被举起)的位置定为负数,只作用于下一层递归时的取最远位置的循环
                        dfs(d + 1);
                        x[j] = e;                               //归位
                        w[j] = 0;                               //还原为可以进行操作
                        a[i] ^= 2;                              //还原为未举起
                        a[i] ^= 4;                              //还原为不可抛出
                    }
                }
            }
        }
        //a[i] == 4 || a[i] == 5(可抛出)
        if (a[i] & 4)
        {
            for (j = 1; j <= t[i]; j++)                         //抛出
            {
                w[p[i]] = 0;                                    //变更a[j]为可操作(以下a[j]指a[i]所举起的人)
                a[i] ^= 4;                                      //不可抛出
                e = x[p[i]];                                    //记录a[j]被举起前位置
                x[p[i]] = x[i] + j;                             //抛出a[j],并更新a[j]位置
                if (near(x[p[i]]) || j == t[i])                 //如果a[j]被抛出后的位置旁边有人或者抛出距离达到上限
                {
                    dfs(d + 1);
                }
                x[p[i]] -= j;                                   //归位
                x[p[i]] -= j;                                   //a[j]向后抛出j
                if (near(x[p[i]]) || j == t[i])                 //如果a[j]被抛出后的位置旁边有人或者抛出距离达到上限
                {
                    dfs(d + 1);
                }
                x[p[i]] = e;                                    //还原a[j]为未举起前的位置
                a[i] ^= 4;                                      //还原a[j]为可抛出
                w[p[i]] = 1;                                    //还原a[j]为不可操作
            }
        }
    }
    return ;
}

int main()
{
    int i = 1;
    //键入每个人的信息
    for (; i <= 3; i++)
    {
        scanf("%d %d %d", &x[i], &l[i], &t[i]);
    }
    //深度优先遍历
    dfs(1);
    //输出最远距离
    printf("%d\n", ans);
    return 0;
}

  

时间: 2024-08-03 22:16:02

蓝桥杯_left and throw的相关文章

2015年蓝桥杯省赛B组C/C++(试题+答案)

首先说,这次我是第二次参加蓝桥杯(大学里最后一次),可这次去连个三等都没拿到,有些心灰意冷,比上一次还差, 当时看到成绩出来的时候有些失落,但是跌倒了,再站起来继续跑就可以了.可能是状态不好吧,纯属自我安慰. 接下来我把今年的题目又重新做了一遍,写下了这篇博客,如果也有需要探讨答案的,希望可以有帮助. 第一题: 第1题:统计不含4的数字 题目大意 统计10000至99999中,不包含4的数值个数. 解题分析: 第一种解法: 数学方法,这种是在网上看到的一种解法: 最高位除了0.4不能使用,其余8

蓝桥杯——算法训练之乘积最大

问题描述 今年是国际数学联盟确定的"2000--世界数学年",又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一个好朋友XZ也有幸得以参加.活动中,主持人给所有参加活动的选手出了这样一道题目: 设有一个长度为N的数字串,要求选手使用K个乘号将它分成K+1个部分,找出一种分法,使得这K+1个部分的乘积能够为最大. 同时,为了帮助选手能够正确理解题意,主持人还举了如下的一个例子: 有一个数字串:312, 当N=3,K=1时

2016 第七届蓝桥杯 c/c++ B组省赛真题及解题报告

2016 第七届蓝桥杯 c/c++ B组省赛真题及解题报告 勘误1:第6题第4个 if最后一个条件粗心写错了,答案应为1580. 条件应为abs(a[3]-a[7])!=1,宝宝心理苦啊.!感谢zzh童鞋的提醒. 勘误2:第7题在推断连通的时候条件写错了,后两个if条件中是应该是<=12 落了一个等于号.正确答案应为116. 1.煤球数目 有一堆煤球.堆成三角棱锥形.详细: 第一层放1个, 第二层3个(排列成三角形), 第三层6个(排列成三角形), 第四层10个(排列成三角形). -. 假设一共

蓝桥杯——判定字符的位置。

蓝桥杯——判断字符的位置 题目: 输入一个字符串,编写程序输出该字符串中元音字母的首次出现位置,如果没有元音字母输出0.英语元音字母只有‘a’.‘e’.‘i’.‘o’.‘u’五个. 样例输入: hello样例输出:2 样例输入: apple样例输出:1 样例输入: pmp样例输出:0 java code: import java.util.*;public class Yuanyingzifu {    public static int fun(String str)    {        

蓝桥杯 地宫取宝(12&#39;)

X 国王有一个地宫宝库.是n x m 个格子的矩阵.每个格子放一件宝贝.每个宝贝贴着价值标签. 地宫的入口在左上角,出口在右下角. 小明被带到地宫的入口,国王要求他只能向右或向下行走. 走过某个格子时,如果那个格子中的宝贝价值比小明手中任意宝贝价值都大, 小明就可以拿起它(当然,也可以不拿). 当小明走到出口时,如果他手中的宝贝恰好是k件,则这些宝贝就可以送给小明. 请你帮小明算一算,在给定的局面下,他有多少种不同的行动方案能获得这k件宝贝. [数据格式] 输入一行 3个整数,用空格分开:n m

蓝桥杯-开心的金明

//蓝桥杯-算法训练 开心的金明 //评测结果 AC //动态规划 01背包 #include <cstdio> #include <algorithm> #include <cstring> using namespace std; const int maxn = 30005; int main() { int N, m; scanf( "%d%d", &N, &m ); int f[maxn]; int v, w; memset

第六届蓝桥杯2015-省赛-C语言大学B组 个人题解

题目连接:http://course.baidu.com/view/2d86a6c1960590c69fc37622.html 1,奖券数目52488 #include <iostream> using namespace std; bool isOK(int a) { while(a) { if(a%10==4) return false; a /= 10; } return true; } int main() { int ans = 0; for(int i=10000; i<=9

第五届蓝桥杯全国软件设计大赛--2013年校内选拔赛Java题目

第五届蓝桥杯全国软件设计大赛 2013年校内选拔赛Java题目 一.考生注意: (1)[结果填空题]要求参赛选手根据题目描述直接填写结果.求解方式不限.不要求源代码. 把答案存入[考生文件夹]下对应题号的文件中即可. (2)[代码填空题]要求参赛选手在弄清给定代码工作原理的基础上填写缺失的部分,使得程序逻辑正确.完整.所填写的代码不超过一条语句(即中间不能出现分号). 把填空的答案(仅填空处的答案,不包括题面已存在的代码)存入[考生文件夹]下对应题号的文件中中即可. (3)[编程题]要求选手设计

蓝桥杯——真题训练之蚂蚁感冒

标题:蚂蚁感冒 长100厘米的细长直杆子上有n只蚂蚁.它们的头有的朝左,有的朝右. 每只蚂蚁都只能沿着杆子向前爬,速度是1厘米/秒. 当两只蚂蚁碰面时,它们会同时掉头往相反的方向爬行. 这些蚂蚁中,有1只蚂蚁感冒了.并且在和其它蚂蚁碰面时,会把感冒传染给碰到的蚂蚁. 请你计算,当所有蚂蚁都爬离杆子时,有多少只蚂蚁患上了感冒. [数据格式] 第一行输入一个整数n (1 < n < 50), 表示蚂蚁的总数. 接着的一行是n个用空格分开的整数 Xi (-100 < Xi < 100),