蓝桥杯思维训练营(四)

文章目录

小红打怪

小红打怪


思路分析:可以看到ai的范围较大,如果我们直接一个个进行暴力遍历的话,会超时。当我们的攻击的次数越大的时候,怪物的血量就会越少,这里就有一个单调的规律在里面,我们可以考虑以 攻击的轮次作为二分的变量,进行二分,对于每一个二分的mid,我们设计好对应的check函数,判断在mid轮攻击下,是否可以消灭所有的怪兽

python 复制代码
n = int(input())
num = list(map(int,input().split()))
n = len(num)

def check(mid):
    # 判断在mid次数之内是否将怪物全部消灭
    # 先处理群体攻击
    newnum = [max(0,u-mid) for u in num ]
    # 再处理相邻的攻击,遇到相邻才处理,多余的次数转变为单次攻击
    count = mid
    for i in range(n-1):
        if newnum[i]!=0 and count!=0:
            # 这个sub的值别漏了对count的判断
            sub = min(newnum[i],newnum[i+1],count)
            newnum[i]-=sub
            newnum[i+1]-=sub
            count-=sub
        if count==0:
            break
    # 处理单次操作,如果count还有剩余,那么总的操作次数是 mid+count
    a = sum(newnum)
    if a <= mid + count:
        return True
    else:
        return False

# 二分操作
left,right = 0,max(num)
while left<=right:
    mid = (left+right)//2
    if check(mid):
        right=mid-1
    else:
        left=mid+1
print(left)

算法实现的细节:首先check函数首先处理的是全局攻击,然后再进行处理相邻的攻击(相邻攻击的时候,为了节省次数,我们只处理相邻的情况,每次减去的部分sub是min(newnum[i],newnum[i+1],count)),如果减完之后,count有剩余,则加到单个攻击里面,最后判断sum剩余的情况与我们的攻击次数是否够

python 复制代码
    count = mid
    for i in range(n-1):
        if newnum[i]!=0 and count!=0:
            # 这个sub的值别漏了对count的判断
            sub = min(newnum[i],newnum[i+1],count)
            newnum[i]-=sub
            newnum[i+1]-=sub
            count-=sub
        if count==0:
            break

494.目标和

494.目标和


思路分析:我们可以使用类似于0-1背包的问题,选择加的整数,
要加的整数为p ,整个数组的和为s,那么要减去的数目就是s-p ,也就是有 p - (s-p) = target,经过转换就有 p = (s+target)/2,也就是 这个 target + s必须是偶数,由于nums[i]没有负数,所以target<0的时候也无解

python 复制代码
class Solution:
    def findTargetSumWays(self, nums: List[int], target: int) -> int:
        target += sum(nums)
        if target < 0 or target % 2 :
            return 0
        target //= 2

        n = len(nums)

        @cache
        # dfs(i,c)表示,前i中数字中,满足target的组合数
        def dfs(i,c):
            # i < 0,表示选择完了
            if i < 0:
                return 1 if c == 0 else 0
            if c < nums[i]:
                return dfs(i-1,c)
            return dfs(i-1,c) + dfs(i-1,c-nums[i])
        
        return dfs(n-1,target)
相关推荐
我爱计算机视觉9 分钟前
ICCV 2025 (Highlight) Being-VL:师夷长技,用NLP的BPE算法统一视觉语言模型
人工智能·算法·语言模型·自然语言处理
virtual_k1smet6 小时前
#等价于e * d ≡ 1 mod φ(n) #模逆元详解
人工智能·算法·机器学习
可触的未来,发芽的智生6 小时前
新奇特:神经网络的集团作战思维,权重共享层的智慧
人工智能·python·神经网络·算法·架构
_屈臣_6 小时前
卡特兰数【模板】(四个公式模板)
c++·算法
坚持编程的菜鸟7 小时前
LeetCode每日一题——交替合并字符串
c语言·算法·leetcode
悦悦子a啊7 小时前
[Java]PTA: jmu-Java-02基本语法-08-ArrayList入门
java·开发语言·算法
xlq223228 小时前
12.排序(上)
数据结构·算法·排序算法
努力学习的小廉8 小时前
我爱学算法之—— 分治-快排
c++·算法
未知陨落9 小时前
LeetCode:77.买卖股票的最佳时机
算法·leetcode
DeeGLMath9 小时前
排序算法的并行加速实现
算法·排序算法·joblib·sortingx