[CSP-S 2023] 种树 —— 二分+前缀和

news/2024/7/15 18:08:03 标签: 算法, 图论

This way

题意:

    一开始以为是水题,敲了一个二分+贪心检查的代码,20分。发现从根往某个节点x走的时候,一路走来的子树上的节点到已栽树的节点的距离会变短,那么并不能按照初始情况贪心。
    于是就想着检查时候用线段树,存的是(每个节点最晚开始时间-它距离最近栽树的点的距离)往后就将这个称为ddl。每一步都往当前最小值的位置走,每走一步,将当前这一步的子树区间+1,如此往复。当走到一个点发现已经走的步数>这个点最晚开始时间时候就是not。但是代码过于繁杂,最终放弃了这样思路,而且常数可能会比较大,最终如果TLE了血亏。
    首先这道题的答案满足二分的性质,考虑使用二分。二分出来结束时间的时候,我们可以求出每个点的最晚到达时间,首先分c>=0和c<0两种情况。对于c<0的时候又要分三种情况。其实就是等差数列求和公式,但是注意会爆longlong,所以转乘为除。我这里使用二分去找答案,当然直接算好像也行?
    发现其实每个点的ddl就是它子树的ddl最小值,也就是每个点的ddl可视为子树中最小ddl-当前点到ddl最小的节点的距离,例如:
在这里插入图片描述
假设点1的最晚开始时间是第10天,点2是第3天,点3是第50天,点4是第90天,点5是第4天。那么转换过来,其实它们真实的ddl如下:
在这里插入图片描述
    这个时候我们只需要将所有真·ddl存到桶里面,再做一个前缀和,记为num[i]。若i<num[i],则表示你走了i步,但是有超过i个点的ddl在i步之内(我们在上图处理完之后,所有链上的ddl必然是递增的也就是如果点x需要走10步,那father[x]最大为9,father[father[x]]最大为8,也就是为x做铺垫),那么表示无法在i步内满足num[i]个点的ddl。

#include<bits/stdc++.h>
using namespace std;
#define ll long long
const int N=1e5+5;
ll a[N],b[N],c[N],en[N],e,shou,mo;
int n,x,y,dep[N],u,tim,num[N],t[N];
vector<int>vec[N];
bool vis[N];
#define pii pair<int,int>
vector<pii>day;
int dfs(int x,int fa){
    for(int ne:vec[x]){
        if(ne==fa)continue;
        t[x]=min(t[x],dfs(ne,x)-1);
    }
    num[t[x]]++;
    return t[x];
}
bool check(ll d){
    day.clear();
    memset(num,0,sizeof num);
    for(int i=1;i<=n;i++){
        ll l=1,r=min(1ll*n,d);t[i]=-1;
        while(l<=r){
            ll x=l+r>>1;
            if(c[i]>=0){
                if((a[i]*2ll+d-x)/(d-x+1)<=2*b[i]+(x+d)*c[i])t[i]=x,l=x+1;
                else r=x-1;
            }
            else{
                c[i]=-c[i];
                if(en[i]<=x){
                    if(a[i]<=d-x+1)t[i]=x,l=x+1;
                    else r=x-1;
                }
                else if(en[i]<=d){
                    e=en[i]-1;
                    shou=b[i]-x*c[i],mo=b[i]-e*c[i];
                    if((2*a[i]-2*(d-e)+e-x)/(e-x+1)<=(shou+mo))t[i]=x,l=x+1;
                    else r=x-1;
                }
                else{
                    ll shou=b[i]-x*c[i],mo=b[i]-d*c[i];
                    if((2*a[i]+d-x)/(d-x+1)<=(shou+mo))t[i]=x,l=x+1;
                    else r=x-1;
                }
                c[i]=-c[i];
            }
        }
        if(t[i]-dep[i]<=0)return 0;
    }
    dfs(1,0);
    for(int i=1;i<=n;i++){
        num[i]+=num[i-1];
        if(num[i]>i)
            return 0;
    }
    return 1;
}
int main()
{
    ll l=n,r=0,ans=-1;
    scanf("%d",&n);
    for(int i=1;i<=n;i++){
        scanf("%lld%lld%lld",&a[i],&b[i],&c[i]);
        r=max(r,a[i]);
        if(c[i]<0)
            en[i]=(b[i]-c[i]-1)/(-c[i]);
    }
    r=min(r,1000000000ll);
    for(int i=1;i<n;i++){
        scanf("%d%d",&x,&y);
        vec[x].push_back(y),vec[y].push_back(x);
    }
    while(l<=r){
        ll mid=l+r>>1;
        if(check(mid))r=mid-1,ans=mid;
        else l=mid+1;
    }
    printf("%lld\n",ans);
    return 0;
}


http://www.niftyadmin.cn/n/5113902.html

相关文章

Java面试题-Java核心基础-第九天(泛型)

目录 一、泛型的理解 二、泛型的作用 三、泛型有哪些使用方式 四、上限、下限通配符的使用 五、泛型的原理 一、泛型的理解 泛型在jdk5中开始有的&#xff0c;泛型其实就是将类型进行参数话&#xff0c;使得类型在编译时就确定了&#xff0c;这种类型参数可以用在类、接口…

详解对于ReadView 机制如何判断当前事务能够看见

ReadView 机制就是用来判断当前事务能够看见哪些版本的&#xff0c;一个 ReadView 主要包含如下几个部分&#xff1a; m_ids&#xff1a;生成 ReadView 时有哪些事务在执行但是还没提交的&#xff08;称为 “活跃事务”&#xff09;&#xff0c;这些活跃事务的 id 就存在这个字…

[Python进阶] 获取计算机相关信息:WMI

6.13 获取计算机相关信息&#xff1a;WMI 6.13.1 前言 在计算机的日常使用或者编程开发时&#xff0c;有时候需要知道计算机的一些相关信息&#xff0c;比如&#xff1a;计算机硬件信息&#xff0c;包括&#xff1a;CPU、内存、键鼠、显示器、网络状况等。计算机操作系统信息…

ios 实现PDF,Word,Excel等文档类型的读取与预览

文章目录 一、前言二、iCould相关配置三、功能实现3.1 UIDocumentPickerViewController 选取控制器3.2 读取文件3.3 文档预览3.3.1 下载并保存3.3.2 QLPreviewController预览文档四、总结一、前言 最近正在研发的项目有一个需求: 允许用户将iCloud中的文档上传,实现文件的流…

【ML】cheatsheet

LR 原理与面试题目DT, Adaboost, GBDT, xgboost 原理 细节 与 例子 https://www.cnblogs.com/createMoMo/p/12635709.html xgboost挺详细的算法原理与例子 https://zhuanlan.zhihu.com/p/660468945 着重lightgbm就xgboost的改善方向 https://zhuanlan.zhihu.com/p/366952043机器…

因为做了这样的项目,成为了offer收割机!

作者&#xff1a;小傅哥 博客&#xff1a;https://bugstack.cn 沉淀、分享、成长&#xff0c;让自己和他人都能有所收获&#xff01;&#x1f604; 文章目录 一、项目视图二、学习路线1. 实习生2. 校招生3. 社招生 四、目标路径五、项目组合 注意&#xff1a;小傅哥的星球&…

【Rust】4 一文讲解重点 pattern matching | trait | 生命周期 | 闭包 | 迭代器 | 智能指针 | 并发与并行

文章目录 一、pattern matching二、trait2.1 常见 trait2.1.1 Copy 和 Clone2.1.2 PartialEq 和 Eq2.1.3 PartialOrd 和 Ord2.1.4 Hash2.1.5 From, Into, TryFrom, TryInto 2.2 概念2.2.1 关联类型2.2.2 关联常量2.3.3 泛型关联类型2.3.3.1 示例: 用泛型关联类型, 创建集合工厂…

代码随想录算法训练营第五十九天| LeetCode 647 回文子串、LeetCode 516 最长回文子序列、动态规划总结

1 LeetCode 647 回文子串 题目链接&#xff1a;LeetCode 647 回文子串 文章讲解&#xff1a;代码随想录(programmercarl.com) 视频讲解&#xff1a;动态规划&#xff0c;字符串性质决定了DP数组的定义 | LeetCode&#xff1a;647.回文子串 2 LeetCode 516 最长回文子序列 题目链…