[CSP-S模拟测试]:卡常题/b(基环树+DP)
题目描述
$ρ$有一个二分连通无向图,$X$方点、$Y$方点均为$n$个(编号为$1\sim n$)。
这个二分图比较特殊,每一个$Y$方点的度为$2$,一条黑色边,一条白色边。
所有黑色边权值均为$a$,所有白色边权值均为$b$。
选择一个$X$方点,代价为连接的所有边的权值之和。
激活一个$Y$方点,需要选择至少一个与之相邻的$X$方点。
现在,$ρ$想激活每个$Y$方点,他想知道最小的总代价。
不过$ρ$很善良,他给你开了$O2$优化。
这样你就不会被卡常了。
当然,除非你真的连读入优化都不想写,或者常数真的丑死。
输入格式
第一行:三个正整数$n$、$a$、$b$。
接下来$n$行:每行两个正整数,第$i$行表示第$i-1$个$Y$方点的黑色边连接的$X$方点,白色边连接的$X$方点。
输出格式
第一行:一个整数,代表最小的总代价。
样例
样例输入:
4 2 3
1 2
3 1
1 4
2 3
样例输出:
12
数据范围与提示
$20\%$的数据:$n\leqslant 20$。
$40\%$的数据:$n\leqslant {10}^3$。
另外$10\%$的数据:$a=b=1$。
另外$20\%$的数据:保证每个$X$方点连接的边颜色相同。
另外$10\%$的数据:保证原图是一个大小为$2\times n$的环。
$100\%$的数据:$n\leqslant {10}^6,a,b\leqslant 100$,保证无重边。
题解
发现这张图其实就是一个基环树,所以我们可以做基环树$DP$。
首先,找到这个环,然后删掉这条边,这样就变成了一棵树了,向两个方向进行$DP$,取较小的即可。
时间复杂度:$\Theta(n)$。
期望得分:$100$分。
实际得分:$100$分。
代码时刻
#include<bits/stdc++.h>
using namespace std;
struct rec{int nxt,to;}e[2000001];
struct node{int i,l,r;}root;
int head[1000001],cnt=1;
int n,a,b;
int sam[1000001];
int dp[2][1000001][2];
bool vis[1000001];
void add(int x,int y)
{
e[++cnt].nxt=head[x];
e[cnt].to=y;
head[x]=cnt;
}
void pre_dfs(int x,int fa)
{
vis[x]=1;
for(int i=head[x];i;i=e[i].nxt)
{
if(e[i].to==fa)continue;
if(vis[e[i].to]){root=(node){i,x,e[i].to};return;}
pre_dfs(e[i].to,x);
}
}
void DP0(int x,int fa)
{
int sum=0;
dp[0][x][0]=0;
dp[0][x][1]=sam[x];
for(int i=head[x];i;i=e[i].nxt)
{
if(i==root.i||(i^1)==root.i||e[i].to==fa)continue;
DP0(e[i].to,x);
sum+=min(dp[0][e[i].to][0],dp[0][e[i].to][1]);
dp[0][x][0]+=dp[0][e[i].to][1];
}
dp[0][x][1]+=sum;
}
void DP1(int x,int fa)
{
int sum=0;
dp[1][x][0]=0;
dp[1][x][1]=sam[x];
for(int i=head[x];i;i=e[i].nxt)
{
if(i==root.i||(i^1)==root.i||e[i].to==fa)continue;
DP1(e[i].to,x);
sum+=min(dp[1][e[i].to][0],dp[1][e[i].to][1]);
dp[1][x][0]+=dp[1][e[i].to][1];
}
dp[1][x][1]+=sum;
}
int main()
{
scanf("%d%d%d",&n,&a,&b);
for(int i=1;i<=n;i++)
{
int x,y;
scanf("%d%d",&x,&y);
sam[x]+=a;sam[y]+=b;
add(x,y);add(y,x);
}
pre_dfs(1,0);
DP0(root.l,0);
DP1(root.r,0);
cout<<min(dp[0][root.l][1],dp[1][root.r][1])<<endl;
return 0;
}
rp++
[CSP-S模拟测试]:卡常题/b(基环树+DP)的更多相关文章
- NOIP模拟测试39,思维禁锢专场「工业题·玄学题·卡常题」
工业题 题解 抱歉,题解没时间写了 代码 #include<bits/stdc++.h> using namespace std; #define ll long long #define ...
- Codeforces 986D - Perfect Encoding(FFT+爪巴卡常题)
题面传送门 题意:给出 \(n\),构造出序列 \(b_1,b_2,\dots,b_m\) 使得 \(\prod\limits_{i=1}^mb_i\geq n\),求 \(\sum\limits_{ ...
- [CSP-S模拟测试]:小L的数(数位DP+模拟)
题目传送门(内部题132) 输入格式 第一行一个整数$t$. 接下来$t$行每行一个整数$n$. 输出格式 $t$行,每行一个整数表示答案. 样例 样例输入: 41818231232691052109 ...
- 联赛模拟测试5 涂色游戏 矩阵优化DP
题目描述 分析 定义出\(dp[i][j]\)为第\(i\)列涂\(j\)种颜色的方案数 然后我们要解决几个问题 首先是求出某一列涂恰好\(i\)种颜色的方案数\(d[i]\) 如果没有限制必须涂\( ...
- NOIP 模拟 $13\; \text{卡常题}$
题解 一道环套树的最小点覆盖题目,所谓环套树就是有在 \(n\) 个点 \(n\) 条边的无向联通图中存在一个环 我们可以发现其去掉一条环上的边后就是一棵树 那么对于此题,我们把所有 \(x\) 方点 ...
- NOIP模拟13「工业题·卡常题·玄学题」
T1:工业题 基本思路 这题有一个重要的小转化: 我们将原来的函数看作一个矩阵,\(f(i,j-1)*a\)相当于从\(j-1\)向右走一步并贡献a,\(f(i-1,j)*b\)相当于从\(i-1 ...
- [CSP-S模拟测试]:联(小清新线段树)
题目描述 由于出题人懒所以没有背景.一个无限长的$01$序列,初始全为$0$,每次选择一个区间$[l,r]$进行操作,有三种操作:$\bullet 1\ l\ r$将$[l,r]$中所有元素变成$1$ ...
- [CSP-S模拟测试]:Cover(单调栈++单调队列+DP)
题目传送门(内部题126) 输入格式 第一行两个个整数$n,m$表示区间的长度与彩灯的数量. 接下来$m$行,每行三个整数$l_i,r_i,a_i$表示一条彩灯能够覆盖的区间以及它的美观程度. 输出格 ...
- [CSP-S模拟测试]:购物(柯朵莉树)
题目描述 $visit_world$有一个商店,商店里卖$N$个商品,第$i$个的价格为$a[i]$我们称一个正整数$K$是美妙的,当且仅当我们可以在商店里选购若干个商品,使得价格之和落在区间$[K, ...
随机推荐
- vue将页面导出成pdf
npm i jspdf-html2canvas prinOut(){ // 导出pdf let page = document.querySelector('.app-main'); // page ...
- [CSP-S模拟测试]:z(模拟+map+小根堆)
题目背景 $\frac{1}{4}$遇到了一道水题,$eooooo$完全不会做,于是去请教小$D$.结果小$D$已经去了阿塞拜疆,于是,$\frac{1}{4}$只好来问你,这道题是这样的: 题目描述 ...
- GridManager 隐藏列
GridManager 表格管理组件, 对列的隐藏与显示的操作有两种方式. 初始化时指定列为隐藏或显示状态.方式如下: <table></table> var table = ...
- delphi vlc 安装bug 处理编译错误"0" is an invalid value for the "DebugInformation" parameter of the "DCC"
处理编译错误"0" is an invalid value for the "DebugInformation" parameter of the "DCC" [摘要:http://blog.csdn ...
- 返回闭包不能引用循环变量,请改写count()函数,让它正确返回能计算1x1、2x2、3x3的函数。
错误写法: 正确写法:
- ORCAL 数据库的约束以及SQL语言的四种类型
oracle数据库约束: 定义:要输入的这个值是一个什么样的值, 或者是哪个范围内的值 作用: 确保完整性, 确保精确性 1, 非空约束(not null) 记录一条信息的时候如果用户名和密码没有被记 ...
- laravel在路由中设置中间件
//单个 路由 Route::get( 'admin/admin/index' , [ 'middleware' => 'old', 'uses' => 'Admin\AdminContr ...
- python序列基本操作
这里讲一基本概念:容器---可以包含其他对象的对象:两种主要的容器是序列(列表和元祖)和映射(字典) 关于序列的通用基本操作:python中常用的序列主要有两种:列表和元祖 -------索引,切片 ...
- hibernate的拦截器和监听器
拦截器(Intercept):顾名思义,拦截操作,也就是在Hibernate做出动作之前会调用的方法.如果你有需要在Hibernate操作数据库之前想要做的操作,就需要用到这个东西了. 监听器(Lis ...
- 2018-10-8-如何安装-btsync
title author date CreateTime categories 如何安装 btsync lindexi 2018-10-8 9:15:6 +0800 2018-2-13 17:23:3 ...