【蓝桥杯每日一题】砍竹子

砍竹子

2024-12-7 蓝桥杯每日一题 砍竹子 STL 贪心

题目大意

这天, 小明在砍竹子, 他面前有 nn 棵竹子排成一排, 一开始第 ii 棵竹子的 高度为 h i h_i hi. 他觉得一棵一棵砍太慢了, 决定使用魔法来砍竹子。魔法可以对连续的一 段相同高度的竹子使用, 假设这一段竹子的高度为 HH, 那么用一次魔法可以 把这一段竹子的高度都变为 [ ⌊ H 2 ⌋ + 1 ] \begin{bmatrix}\sqrt{\lfloor\frac{H}{2}\rfloor+1}\end{bmatrix} [⌊2H⌋+1 ], 其中 [ x ] \begin{bmatrix}x\end{bmatrix} [x] 表示对 x 向下取整。小明想 知道他最少使用多少次魔法可让所有的竹子的高度都变为 1 。

解题思路

这个题使用到了 STL 的技巧,有些不好想,咱们一步一步分析。

  1. 当前值改变的次数是否要加,取决于与它相邻的元素是否相等,这时候我们只看前面相邻的元素。。

  2. 当相邻元素减小到同一个值的时候,只需要计算一次即可,这一次怎么算,就是这些相邻相同的所有元素的对头那个来算,其他的都不算。

  3. 可以通过模拟每一个数的一个变化历程,然后对比前后的数据变化发现那些数是可以变化的。

  4. 举例:

    cpp 复制代码
    7 6 2 4 1 2
    7 2 1     2
    6 2 1     1
      2 1     0
    4   1     1
        1     0
      2 1     1
  5. 根据例子可以观察到每一个数的更新它所贡献的值只有与前一个数的变化历程不同的数的数量,那么就可以通过两个set集合来进行比较。

Accepted
cpp 复制代码
#include <bits/stdc++.h>

using namespace std;
typedef long long ll;

int main()
{
    int n;
    cin>>n;
    ll res = 0;
    set<ll> pri;
    for(int i = 1;i <= n;i++) {
       ll t;
       cin>>t;
       set<ll> pre;
       while(t > 1) {
           pre.insert(t);
           if(!pri.count(t)) {
               res++;
           }
           t = sqrtl(t/2+1);
       }
       pri = pre;
    }
    
    cout<<res<<endl;
    return 0;
}
思考

这道题还是很难想出来的,必须通过模拟来观察规律求解!!!

备注

想要一起备赛的小伙伴可以私信加我的联系方式!

相关推荐
虾..1 天前
Linux 简单日志程序
linux·运维·算法
Trent19851 天前
影楼精修-眼镜祛反光算法详解
图像处理·人工智能·算法·计算机视觉·aigc
蓝色汪洋1 天前
经典修路问题
开发语言·c++·算法
csuzhucong1 天前
122魔方、123魔方
算法
Salt_07281 天前
DAY 40 早停策略和模型权重的保存
人工智能·python·算法·机器学习
卜锦元1 天前
Golang后端性能优化手册(第三章:代码层面性能优化)
开发语言·数据结构·后端·算法·性能优化·golang
Binky6781 天前
力扣--回溯篇(2)
算法·leetcode·职场和发展
DARLING Zero two♡1 天前
接入 AI Ping 限免接口,让 GLM-4.7 与 MiniMax-M2.1 成为你的免费 C++ 审计专家
开发语言·c++·人工智能
程序喵大人1 天前
constexpr
开发语言·c++·constexpr
Larry_Yanan1 天前
Qt多进程(五)QUdpSocket
开发语言·c++·qt·学习·ui