首页 > 其他分享 >Luogu P10838 『FLA - I』庭中有奇树 题解 [ 绿 ] [ 二分 ] [ 双指针 ] [ 树 ]

Luogu P10838 『FLA - I』庭中有奇树 题解 [ 绿 ] [ 二分 ] [ 双指针 ] [ 树 ]

时间:2024-12-19 22:54:01浏览次数:3  
标签:二分 node 传送 奇树 庭中 题解 路径 int 指针

庭中有奇树:很多算法揉在一起的好题。

转化题意

因为要封锁 \(m\) 条路径,根据贪心思想,他一定会封锁最短的 \(m\) 条路径。所以我们能走的最短传送路径就是最短的第 \(m+1\) 条路径。

这应该是本题最关键的一步转化了,几个月前降智了根本没想到这个。

做法

求第 \(m+1\) 短的路径,这个很显然通过二分求解,求出最后一个比它小的数的个数小于 \(m+1\) 个的数。

二分 check 函数的实现也是容易的,我们从起点 \(s\) 与终点 \(t\) 各自做一遍 dfs 求出两个点与其他点之间的距离,然后按路径长度从小到大排序。注意这时候要记录下标以进行相邻点传送与同点转送的特判。

接下来我们对于每一个传送起点,通过双指针维护最后一个小于 \(dis\) 的指针 \(p\) 计算出该点的贡献,最后遍历与该点相邻的点,判断他们是否被统计进答案,如果被统计进了就删掉这个贡献即可。注意自己传送到自己的情况也要特判。

答案在传送路径、直接走的路径和直接从 \(s\) 传送到 \(t\) 的 \(10^9\) 代价里取最小值即可。

时间复杂度 \(O(n \log v)\)。

代码

细节挺多的,尤其是 check 函数。

#include <bits/stdc++.h>
#define fi first
#define se second
#define lc (p<<1)
#define rc ((p<<1)|1)
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pi;
int n,m,k,s,t;
struct edge{
    int to,w;
};
struct node{
    ll w,id;
}d[2][100005];
ll td[100005];
vector<edge>g[100005];
bool cmp(node a,node b)
{
    return a.w<b.w;
}
void dfs(int u,int fa,int mode)
{
    for(auto eg:g[u])
    {
        int v=eg.to,w=eg.w;
        if(fa==v)continue;
        d[mode][v].w=d[mode][u].w+w;
        dfs(v,u,mode);
    }
}
ll check(ll dis)
{
    ll res=0,p=n;
    for(int i=1;i<=n;i++)
    {
        while(p>0&&d[0][i].w+d[1][p].w>=dis)p--;
        res+=p;
        int u=d[0][i].id;
        for(auto eg:g[u])
        {
            int v=eg.to;
            if(d[0][i].w+td[v]<dis)res--;
        }
        if(d[0][i].w+td[u]<dis)res--;
    }
    return res;
}
int main()
{
    //freopen("sample.in","r",stdin);
    //freopen("sample.out","w",stdout);
    ios::sync_with_stdio(0);
    cin.tie(0);
    cout.tie(0);
    cin>>n>>m>>k>>s>>t;
    for(int i=1;i<n;i++)
    {
        int u,v,w;
        cin>>u>>v>>w;
        g[u].push_back({v,w});
        g[v].push_back({u,w});
    }
    dfs(s,0,0);
    dfs(t,0,1);
    for(int i=1;i<=n;i++)
    {
        d[0][i].id=i;
        d[1][i].id=i;
        td[i]=d[1][i].w;
    }
    sort(d[0]+1,d[0]+n+1,cmp);
    sort(d[1]+1,d[1]+n+1,cmp);
    ll l=0,r=1e18,mid;
    while(l<r)
    {
        mid=(l+r+1)>>1;
        if(check(mid)<m+1)l=mid;
        else r=mid-1;
    }
    ll ans=min(ll(1e9),min(l+k,td[s]));
    cout<<ans;
    return 0;
}

标签:二分,node,传送,奇树,庭中,题解,路径,int,指针
From: https://www.cnblogs.com/zhr0102/p/18618072

相关文章

  • 组合数学+ybt题解
    加法原理乘法原理排列数从\(n\)个数中任取\(m\)个元素的排列的方案数,表示为\(A^m_n=\frac{n!}{(n-m)!}\)\(0!=1\)全排列\(A^n_n\)组合数从\(n\)个元素中取出\(m\)个元素的组合的个数,表示为\(\dbinom{n}{m}=\frac{A^m_n}{m!}=\frac{n!}{m!(n-m)!}\)如何理解呢?......
  • P3316 [SDOI2014] 里面还是外面 题解
    实现有些傻瓜,喜提时空双最劣解。首先要判断一个点是否在多边形内,一个比较好的方法是从这个点向上引一条射线,若和奇数条边相交就在多边形内,否则在多边形外。二维信息,考虑用树套树维护。把多边形的每一条边都扔到它\(x\)坐标范围的线段树节点里,即线段树节点\((l,r)\)里面维护......
  • 洛谷 P11337 「COI 2019」IZLET 题解
    如果每条边连接的两个点颜色都不相同,那么可以使用如下策略确定每个点的颜色:令\(c_{i,j}\)为\(i\)到\(j\)的路径上的颜色数。对于每个点\(u\),以其为根进行一次dfs,往下找直到找到一个和它颜色相同的或者一个叶子就回溯,如果遇到颜色相同的就将它们在并查集上合并。考虑如何......
  • [Yandex Cup 2024 Qual F] Zeroth Rome 题解
    前言Englishversionofthiseditorialisprovidedafterthesamplecode.题意简述本题为交互题,你需要猜\(t\)个\([0,2024]\)间的非负整数\(x_1,x_2,\ldots,x_t\),可以询问最多\(15\)次,每次询问形如“给定一个大小为\(N(1\leN\le2025)\)的集合\(S\)满足\(S\)......
  • [APC001H] Generalized Insertion Sort 题解
    将\(a_i\)视为放在结点\(i\)上面的球;称位置\(i\)对应的球为\(i\),区别于“位置\(i\)上面的球为\(a_i\)”。考虑树是一条链的时候怎么做(下称链插入方法):此时只需要将这条链上面的球按照编号从上到下排序。这是一个类似插入排序的过程,维护深度最大的的若干个球编号的相对顺......
  • [USACO24OPEN] Grass Segments G 题解
    考虑对于一个区间\([l_i,r_i]\),最少重叠长度为\(k_i\),怎样的区间\([l_j,r_j]\)可以与前者产生贡献;首先\(r_j-l_j\gek_i\),在满足这个条件的情况下需要有\(r_j\gel_i+k_i\landl_j\ler_i-k_i\),这里\(\land\)表示合取,即C++中的\(\mathrm{and}\)。正难则反,考虑用长度\(......
  • 题解:P10483 小猫爬山
    思路第一眼我以为是个背包,但由于是分组,所以有多个缆车,明显不能用背包。我做这题是因为老师要求,那是我们在学深搜减枝,所以我就开始写深搜。这一题实际上是先选一直最重的猫,然后搞个\(sum\)数组,每搞一个新缆车的就下一个下标继续放,如果能放就放,当然也要搞一个能放但不放的。减枝......
  • RHCE环境公共问题解决(9.0)
    关于如何使用远程软件进行连接环境问题查看此处网络适配器模式,如果是NAT请修改为仅主机模式Vmware有两张网卡,一张是Vmware1,一张是Vmware8(环境必须用仅主机,避免环境判分错误)默认情况下,仅主机模式修改Vmware1网卡 打开Windows的网络适配器可以看到两张网卡 环境IP为1......
  • AT_agc030_f [AGC030F] Permutation and Minimum 题解
    先去掉相邻两个都填完的位置,对于两个都没填的记个数为\(c\),最后只需要将答案乘上\(c!\)。接下来考虑从小到大枚举所有数进行dp,记\(f_{i,j,k}\)表示考虑完前\(1\simi\),有\(j\)个数需要跟一个位置确定的数匹配,有\(k\)个数需要跟后面一个自由的数匹配,考虑当前的数:如果......
  • AT_agc009_d [AGC009D] Uninity 题解
    一道妙妙题。一句话题意:求点分树的高度最小值。给所有点填上一个数表示它子树\(k\),考虑一种填法什么时候满足条件,发现当且仅当任意两对值相同的点之间的路径上存在一个权值更大的点。考虑随便找一个点作为根从叶子节点开始贪心填值,每次选择能填的最小值,发现这样填最终的值的最......