Live2D

Solution -「ARC 125F」Tree Degree Subset Sum

Description

  Link.

  给定含有 n 个结点的树,求非负整数对 (x,y) 的数量,满足存在 SV, |S|=xuSdu=y,其中 du 表示点 u 的度数。

  n2×105

Solution

  方便期间,以下所有 du 表示 u 的度数 1

  出题人莫名其妙告诉你一棵树,无非是强调 d=n2,自然想到根号分治。不过朴素 DP 的状态数量就已经难以接受,我们需要更多的结论。

  比如这个结论:

结论:若 x1x2(x1,y),(x2,y) 均合法,那么 x3[x1,x2], (x3,y) 也合法。

证明   对于某个 y,取出最小 xl 和最大的 xr,使得 (xl,y),(xr,y) 合法。设 {dn} 中有 z 个值为 0,则我们只需证明 xrxl2z,这是由于 (xl,y) 的选取中必然不含 0,那么 (xl+1,y),(xl+2,y),,(xl+z,y) 都合法,(xrk,y) 同理。

  考虑任意一个 SV,令 dS=uSdu,那么

  • dS|S|z,显然;
  • dS|S|z2dSd=n2,取等时 |S|nz,得证。

  即 zdS|S|z2,考虑将 (xl,y)(xr,y) 代入,有 zyxryxlz2,可以推出 xrxl2z2,故已有原命题成立。 

  所以,我们只需要对于每个 y,DP 求出 xlxr 就能得到答案。DP 时根号分治,内部用单调队列优化即可。复杂度 O(nn)

Code

/*~Rainybunny~*/

#include <bits/stdc++.h>

#define rep( i, l, r ) for ( int i = l, rep##i = r; i <= rep##i; ++i )
#define per( i, r, l ) for ( int i = r, per##i = l; i >= per##i; --i )

typedef long long LL;

const int MAXN = 2e5;
int n, d[MAXN + 5], f[MAXN + 5], g[MAXN + 5];

inline void trans( const int v, const int c, const int r,
  int* h, const auto& cmp ) { // cmp(a,b) is true <=> a is the better value.
    static int que[MAXN + 5], th[MAXN + 5];
    int hd = 1, tl = 0;
    for ( int i = r; i <= n - 2; i += v ) {
        th[i] = h[i];
        while ( hd <= tl && que[hd] + c * v < i ) ++hd;
        while ( hd <= tl
          && cmp( th[i] - i / v, th[que[tl]] - que[tl] / v ) ) --tl;
        que[++tl] = i;
        h[i] = th[que[hd]] + ( i - que[hd] ) / v;
    }
}

int main() {
    scanf( "%d", &n );
    rep ( i, 1, n ) d[i] = -1;
    rep ( i, 2, n ) {
        int u, v; scanf( "%d %d", &u, &v );
        ++d[u], ++d[v];
    }

    std::sort( d + 1, d + n + 1 );
    memset( f, 0x3f, sizeof f ), memset( g, 0xc0, sizeof g );
    f[0] = g[0] = 0;
    for ( int l = 1, r; l <= n; l = r + 1 ) {
        for ( r = l; r < n && d[r + 1] == d[l]; ++r );
        int v = d[l], c = r - l + 1;
        if ( !v ) { g[0] = c; continue; }
        rep ( r, 0, v - 1 ) {
            trans( v, c, r, f,
              []( const int u, const int v ) { return u < v; } );
            trans( v, c, r, g,
              []( const int u, const int v ) { return u > v; } );
        }
    }

    LL ans = 0;
    rep ( i, 0, n - 2 ) {
        // printf( "%d: [%d,%d]\n", i, f[i], g[i] );
        if ( f[i] <= g[i] ) ans += g[i] - f[i] + 1;
    }
    printf( "%lld\n", ans );
    return 0;
}

posted @   Rainybunny  阅读(58)  评论(0编辑  收藏  举报
编辑推荐:
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现
历史上的今天:
2020-08-23 Solution -「LOCAL」Burning Flowers
点击右上角即可分享
微信分享提示