洛谷普及B3628 机器猫斗恶龙

题目:机器猫斗恶龙

题号:B3628 

难度:普及一

题目分析

思路

基本准则:保证每步生命值大于零

起初血量既要满足刚开始的减血,还要满足后面未知的扣血。

所以这题应该从后向前来判断。每一步都能决定前一步至少留有多少血量。

举个例子

血量变化                    -100,-200,500,200,400,-100,200,400,-200,-300

从后向前判断

-300 - > -200  这两步需要至少 501的血量

前面补给400,则400之前至少有101的血量,这样来凑够501

前面又要补给200,而200>101 则无论200之前有多少血量,只要能活着到200就足够

所以该步要清零,也就是说只要有一滴血能到达200补给就行,

再往前推,-100,则需要101,前面又遇到400>101,则再次清零变成一滴血,

再往前推遇到的遇到的500和200均为加血,所以也不用考虑,只要能或者走过-100,和-200就算通关后面,所以该数列看似很长,只需要按照该规律,就简化成了-100,-200,end,

所以该值只需要301滴血便足够。

构建代码

创建数组存储道路上的情况,加血/减血

scanf("%lld",&n);
    int a[n+1];
    
    for(i=1;i<=n;i++)
    scanf("%d",&a[i]);

初始一个为0的存储值的变量,然后从后向前相加,如果遇到补给>所需 的情况则清零

for(i=n;i>=1;i--)
    {    num+=a[i];
    if(num>0)
    num=0;    }

这样一直遍历到开头,便可以再num上取得所需至少的血量。

代码总和

#include <stdio.h>

int main() {
    long long num=0,i,n,p;
    
    scanf("%lld",&n);
    int a[n+1];
    
    for(i=1;i<=n;i++)
    scanf("%d",&a[i]);
    
    for(i=n;i>=1;i--)
    {    num+=a[i];
    if(num>0)
    num=0;    }
    
    p=1-num;
    printf("%lld",p);
    return 0;
}
    

这个办法相比于二分法解题简便的太多,不需要占用太多的时间和空间。

关于二分法,因为这个题,假设所需k滴血量,那么1 ~ k-1 滴血肯定不够,那么只要暴力模拟计算出k的最小值即可,但是这样过于浪费时间空间下列是二分法解决该题的代码

二分法解决该题

#include <stdio.h>

// 检查初始血量mid是否满足所有关卡血量始终>0
int check(int mid, int* a, int n) {
    int current = mid;
    for (int i = 0; i < n; i++) {
        current += a[i];
        if (current <= 0) return 0; // 不满足条件
    }
    return 1; // 满足条件
}

int main() {
    int n;
    scanf("%d", &n);
    int a[n];
    for (int i = 0; i < n; i++) {
        scanf("%d", &a[i]);
    }

    int low = 1, high = 1e9, ans = 1;
    while (low <= high) {
        int mid = (low + high) / 2;
        if (check(mid, a, n)) { // 满足条件,尝试找更小值
            ans = mid;
            high = mid - 1;
        } else { // 不满足,增大初始血量
            low = mid + 1;
        }
    }
    printf("%d\n", ans);
    return 0;
}

代码解释:

  1. 输入处理:读取关卡数量 n 和每个关卡的数值存入数组 a
  2. 二分查找
    • low 初始为 1(血量为正整数),high 设为较大值(确保覆盖所有可能)。
    • 每次计算中间值 mid,通过 check 函数模拟初始血量为 mid 时的关卡过程。
    • 若 check(mid, a, n) 返回 1(满足条件),则尝试更小的初始血量;若返回 0,则增大初始血量。
  3. 结果输出:二分结束后,ans 即为满足条件的最小初始血量。

还存在一种方法,逐步遍历计算最小前缀和

先获取关卡数量 n,再通过循环读取每个关卡的数值并存入数组 a

int n;
scanf("%d", &n);  // 读取关卡数量n
int a[n];
for (int i = 0; i < n; i++) {
    scanf("%d", &a[i]);  // 读取每个关卡的数值(正数为营地补血,负数为战斗扣血)
}

prefix 记录当前前缀和(类似 “当前血量”),min_sum 记录遍历过程中出现的最小前缀和。每次循环累加当前关卡数值,若新的前缀和更小,就更新 min_sum

int prefix = 0, min_sum = 0;
for (int i = 0; i < n; i++) {
    prefix += a[i];  // 计算前缀和(模拟每一步后的血量变化)
    if (prefix < min_sum) {
        min_sum = prefix;  // 更新过程中出现的最小前缀和
    }
}
  • 若 min_sum ≥0,说明即使没有初始血量,过程中血量也不会≤0,因此初始血量设为 1(题目要求血量为正整数)。
  • 若 min_sum <0,初始血量需弥补 min_sum 的负值,公式 −min_sum +1 确保任意时刻血量 >0。例如,min_sum = -600,初始血量为 600 +1 = 601
  • if (min_sum >= 0) {
        printf("1\n");  // 若最小前缀和 ≥0,初始血量至少为1(保证血量始终>0)
    } else {
        printf("%d\n", -min_sum + 1);  // 若最小前缀和 <0,初始血量需抵消最小前缀和的负数影响,+1确保血量始终>0
    }

    代码总和

  • #include <stdio.h>
    
    int main() {
        int n;
        scanf("%d", &n);
        int a[n];
        for (int i = 0; i < n; i++) {
            scanf("%d", &a[i]);
        }
        
        int prefix = 0, min_sum = 0;
        for (int i = 0; i < n; i++) {
            prefix += a[i];
            if (prefix < min_sum) {
                min_sum = prefix;
            }
        }
        
        if (min_sum >= 0) {
            printf("1\n");
        } else {
            printf("%d\n", -min_sum + 1);
        }
        return 0;
    }

    总之,方法不唯一,我认为第一种方法是最简便而且容易理解的,同时效率也是最高的,需要一定的逆向思维。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

LAOLONG-C

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值