【NOI2011】【P1308】道路修建

时间:2022-01-26 09:46:52

这题也太水了吧,为什么不是我这届的NOI(╯‵□′)╯︵┻━┻

原题:

在 W 星球上有 n 个国家。为了各自国家的经济发展,他们决定在各个国家 之间建设双向道路使得国家之间连通。但是每个国家的国王都很吝啬,他们只愿 意修建恰好 n – 1 条双向道路。
每条道路的修建都要付出一定的费用,这个费用等于道路长度乘以道路两端 的国家个数之差的绝对值。例如,在下图中,虚线所示道路两端分别有 2 个、4 个国家,如果该道路长度为 1,则费用为 1×|2 – 4|=2。图中圆圈里的数字表示国 家的编号。

由于国家的数量十分庞大,道路的建造方案有很多种,同时每种方案的修建
费用难以用人工计算,国王们决定找人设计一个软件,对于给定的建造方案,计 算出所需要的费用。请你帮助国王们设计一个这样的软件

n<=1000000,ci<=1000000

刚开始看错题了,以为是给出任意两个国家的距离,然后求生成树……

但是树直接给出来了,直接求值即可不用维护最优值,就是个模拟么……

所以这道题应该算是树形递推而不是树归

先随便挑一个点搞树,稍加观察即可发现每个点和它爹之间连的边的权值就是|n-size[x]-size[x]|*value[x](value[x]表示x和爹之间连边的长度)

然后就是求size

数据很大所以dfs会炸掉(Linux或者手动开栈不会炸?),所以使用bfs

先dfs一遍,把树建起来,然后倒着把队列遍历一遍,因为bfs是一层一层推得,所以反遍历队列就保证先到儿子,在到爹,然后就可以更新了

最后从2-n根据上面的公式↑求值累加即可(我把根节点设为1,根节点不用更新因为根节点没爹)

注意会炸int,如果乘法运算的结果会炸int的话乘的两个数必须至少有一个是longlong,这里把求绝对值函数的返回值设为longlong是一个不错的选择

代码:

 #include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
using namespace std;
int read(){int z=,mark=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-')mark=-; ch=getchar();}
while(ch>=''&&ch<=''){z=(z<<)+(z<<)+ch-''; ch=getchar();}
return z*mark;
}
inline long long jue(long long x,long long y){return (x>y)?x-y:y-x;}//注意绝对值也要longlong
struct ddd{int next,y,value;}e[];int LINK[],ltop=;
inline void insert(int x,int y,int z){e[++ltop].next=LINK[x];LINK[x]=ltop;e[ltop].y=y;e[ltop].value=z;}
struct dcd{int child,brother,father,value,size;}tree[];
inline void insert_tree(int x,int y,int z){tree[y].brother=tree[x].child;tree[x].child=y;tree[y].father=x;tree[y].value=z;}
int n;
int dui[],tou=;
void get_tree(){
dui[tou=]=; tree[].father=tree[].value=;
for(int k=;k<=tou;k++)
for(int i=LINK[dui[k]];i;i=e[i].next)if(e[i].y!=tree[dui[k]].father){
insert_tree(dui[k],e[i].y,e[i].value);
dui[++tou]=e[i].y;
}
for(int k=tou;k>=;k--){
tree[dui[k]].size=;
for(int i=LINK[dui[k]];i;i=e[i].next)if(e[i].y!=tree[dui[k]].father)
tree[dui[k]].size+=tree[e[i].y].size;
}
}
int main(){//freopen("ddd.in","r",stdin);
cin>>n;
int _left,_right,_value;
for(int i=;i<n;i++){
_left=read(); _right=read(); _value=read();
insert(_left,_right,_value); insert(_right,_left,_value);
}
get_tree();
long long ans=;
for(int i=;i<=n;i++) ans+=jue(n-tree[i].size,tree[i].size)*tree[i].value;
cout<<ans<<endl;
return ;
}

【NOI2011】【P1308】道路修建的更多相关文章

  1. 【NOI2011】道路修建 BFS

    [NOI2011]道路修建 Description 在 W 星球上有 n 个国家.为了各自国家的经济发展,他们决定在各个国家之间建设双向道路使得国家之间连通.但是每个国家的国王都很吝啬,他们只愿意修建 ...

  2. 【BZOJ】【2435】【NOI2011】道路修建

    DFS/DP 本来以为是一道傻逼题,然而跪了好久……一直RE…… 直接dfs就好了……x->y val=c  :  ans+=abs(n-size[y]-size[y])*c; 然而为啥会一直R ...

  3. 【LOJ】&num;2445&period; 「NOI2011」道路修建

    题解 看完题目我的第一个反应是--要求最小花费的方案?!怎么求??? 然后我把题读完了.好吧. 记录一下size就行,比NOIP普及组还要不如的题= = 代码 #include <iostrea ...

  4. 冲刺NOIP2015提高组复赛模拟试题(五)2&period;道路修建

    2.道路修建 描述 Description liouzhou_101最悲痛的回忆就是NOI2011的道路修建,当时开了系统堆栈,结果无限RE… 出于某种报复心理,就把那题神奇了一下: 在 Z星球上有N ...

  5. bzoj 2435&colon; &lbrack;Noi2011&rsqb;道路修建 树上 dp

    2435: [Noi2011]道路修建 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...

  6. NOI2011道路修建

    2435: [Noi2011]道路修建 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1974  Solved: 550[Submit][Status ...

  7. BZOJ 2435&colon; &lbrack;Noi2011&rsqb;道路修建&lpar; dfs &rpar;

    NOI的水题...直接一遍DFS即可 ------------------------------------------------------------------------- #includ ...

  8. 2435&colon; &lbrack;Noi2011&rsqb;道路修建

    2435: [Noi2011]道路修建 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2188  Solved: 639[Submit][Status ...

  9. BZOJ&lowbar;2435&lowbar;&lbrack;Noi2011&rsqb;道路修建&lowbar;dfs

    BZOJ_2435_[Noi2011]道路修建_dfs 题意: http://www.lydsy.com/JudgeOnline/problem.php?id=2435 分析: dfs搞定. 我怕爆栈 ...

  10. BZOJ 2435&colon; &lbrack;Noi2011&rsqb;道路修建 dfs搜图

    2435: [Noi2011]道路修建 Description 在 W 星球上有 n 个国家.为了各自国家的经济发展,他们决定在各个国家之间建设双向道路使得国家之间连通.但是每个国家的国王都很吝啬,他 ...

随机推荐

  1. oracle分层查询中的start with和connect by&lpar;树结构查询&rpar;

    来源:  http://blog.csdn.net/itmyhome1990/article/details/16338637   ORACLE是一个关系数据库管理系统,它用表的形式组织数据,在某些表 ...

  2. XPATH基础入门资料

    http://www.w3school.com.cn/xpath/xpath_syntax.asp 不错的网址,入门学习资料

  3. &lbrack;资料&rsqb;常用Windows CMD指令

    1. 查找80端口开放情况 netstat -aon|findstr “80″ 2. 用netstat查询端口占用程序的PID,显示列表的最后一列就是占用程序的PID,然后再利用tasklist找到这 ...

  4. libbspatch&period;so

    http://www.zhihu.com/question/21154099 http://blog.csdn.net/hmg25/article/details/8100896 91助手和Googl ...

  5. 【转】Ruby入门教程(一)

    1. Ruby环境搭建 在Windows下,搭建Ruby环境,比较简单的方法是在“RubyInstaller”上下载一个合适的版本(D瓜哥使用的是最新版),直接安装就可以了. 另外,吐槽两句,网上有人 ...

  6. UVaLive 3708

    题意:周长为10000的圆上等距分布n个雕塑,求再加入m个雕塑后,为使所有雕塑等距分布所需移动原来n个雕塑的最小总距离. 分析:计算相对距离. #include<cstdio> #incl ...

  7. delphi获取mdb密码

    function GetPasswordFromAccess(AFileName: string): string;var  myms: TMemoryStream;  b: array of Byt ...

  8. Eclipse Code Template 设置自动加注释(转)

    Eclipse Code Template 设置自动加注释 设置注释模板的入口: Window->Preference->Java->Code Style->Code Temp ...

  9. 分页:T-SQL存储过程和EF存储过程的使用

    首先准备好分页的T-SQL语句: create proc usp_activityFenYe @pageIndex int, @pageSize int, @pageCount int output ...

  10. 力扣算法题—079单词搜索【DFS】

    给定一个二维网格和一个单词,找出该单词是否存在于网格中. 单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水平相邻或垂直相邻的单元格.同一个单元格内的字母不允许被重复使用. ...