Live2D

Solution -「CF 1586F」Defender of Childhood Dreams

Description

  Link.

  定义有向图 G=(V,E)|V|=nu,vEu<v。求一个对 E 的染色 f,使得 v1,v2,,vk+1,|{f(vi,vi+1)i[1,k]}|=1,同时最小化 f 的值域大小。

  2k<n103

Solution

  设 f 的值域大小为 c,断言:clogkn

  证明 该结论的等价表述是,若 f 值域大小为 c,则 nkc。当 c=0 时显然成立。接下来对 c 进行归纳:

  任取一个 f 值域大小为 c 的,被合法染色的图 G,并任取某种颜色 x,据此将点集 V 划分为 V1,V2,,Vt,使得 Vi 的导出子图中不存在颜色为 x 的边。这些点集之间的连边颜色全部为 x,所以 tk。而仅考虑某个 Vi 的导出子图,它至多用 c1 中颜色染色,由归纳假设,|Vi|kc1,继而 |V|=|Vi|kkc1=kc。 

  模仿归纳方法得到构造方法:划分点集,将点集之间的边染色,而后递归处理。复杂度上限为 O(n2)

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 )

const int MAXN = 1e3;
int n, k, mxc, ans[MAXN + 5][MAXN + 5];

inline void solve( const int l, const int r, const int clr ) {
    if ( l == r ) return ;
    if ( clr > mxc ) mxc = clr;
    int s = ( r - l + k ) / k;
    for ( int i = l; i <= r; i += s ) {
        solve( i, std::min( i + s - 1, r ), clr + 1 );
        rep ( u, l, i - 1 ) rep ( v, i, std::min( i + s - 1, r ) ) {
            ans[u][v] = clr;
        }
    }
}

int main() {
    scanf( "%d %d", &n, &k );
    
    solve( 1, n, 1 );
    printf( "%d\n", mxc );
    rep ( i, 1, n ) rep ( j, i + 1, n ) printf( "%d ", ans[i][j] );
    putchar( '\n' );
    return 0;
}

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