[AGC025E]Walking on a Tree
题意:有一棵树,你要按顺序在树上走$m$次,每次从$u_i$到$v_i$或从$v_i$到$u_i$,走完后,如果一条边被单向经过,那么它贡献$1$的价值,如果一条边被双向经过,那么它贡献$2$的价值,给出所有的$(u_i,v_i)$,你要安排每次走路的方向以最大化价值
设边$i$被经过的次数为$c_i$,容易看出答案的上界是$\sum\min(c_i,2)$,下面我们构造性地说明这个上界总可以被达到
如果$n=1$,树中没有边,接下来考虑$n\gt1$
任选一个叶子$u$,设$e$为连接它的唯一一边,$v$为$e$的另一端点
如果$c_e=0$,直接删掉$u$,这对答案没有影响
如果$c_e=1$,我们也可以删掉$u$,只不过经过$u$的那条路径的端点要缩至$v$,此时$e$贡献的价值为$1$
如果$c_e\geq2$,不停地选取两条路径$(u,a)$和$(u,b)$,那么我们可以把$(u,a)$和$(u,b)$换为$(a,b)$,如果$(a,b)$最终被决定为$a\rightarrow b$,那么还原过来就是$a\rightarrow u,u\rightarrow b$,否则是$b\rightarrow u,u\rightarrow a$,设$(u,c)$为$(u,a)$和$(u,b)$的交,那么$(u,c)$上的每一条边都一定能被双向经过,所以$e$贡献的价值为$2$,重复这个过程直至$c_e\lt2$,这样就构造性地说明了我们可以达到上界
我写得比较丑,毕竟是STL不用钱系列...
#include<stdio.h> #include<string.h> #include<vector> using namespace std; int h[2010],nex[4010],to[4010],dep[2010],fa[2010],fav[2010],M; void add(int a,int b){ M++; to[M]=b; nex[M]=h[a]; h[a]=M; } void dfs(int x){ dep[x]=dep[fa[x]]+1; for(int i=h[x];i;i=nex[i]){ if(to[i]!=fa[x]){ fa[to[i]]=x; dfs(to[i]); } } } void col(int x,int y){ if(dep[x]<dep[y])swap(x,y); while(dep[x]>dep[y]){ fav[x]++; x=fa[x]; } while(x!=y){ fav[x]++; fav[y]++; x=fa[x]; y=fa[y]; } } struct rt{ int x,y; rt(int a=0,int b=0){x=a;y=b;} }; bool operator==(rt a,rt b){return a.x==b.x&&a.y==b.y;} int d[2010],n; vector<rt>solve(vector<rt>v){ if(v.size()==0)return v; int x,u,i; for(x=1;x<=n;x++){ if(d[x]==1)break; } if(x>n)return v; for(i=h[x];i;i=nex[i]){ if(d[to[i]]){ u=to[i]; break; } } d[x]--; d[u]--; vector<rt>lx,nv; vector<int>id; int pos[2010]; memset(pos,-1,sizeof(pos)); for(i=0;i<(int)v.size();i++){ if(v[i].x==x||v[i].y==x){ lx.push_back(v[i]); id.push_back(i); }else{ nv.push_back(v[i]); pos[i]=nv.size()-1; } } if(lx.size()==0)return solve(v); for(i=0;i+1<(int)lx.size();i+=2){ if(lx[i].y==x)swap(lx[i].x,lx[i].y); if(lx[i+1].y==x)swap(lx[i+1].x,lx[i+1].y); if(lx[i]==lx[i+1]){ if(v[id[i]]==v[id[i+1]])swap(v[id[i]].x,v[id[i]].y); continue; } nv.push_back(rt(lx[i].y,lx[i+1].y)); pos[id[i]]=pos[id[i+1]]=nv.size()-1; } if(lx.size()&1){ if(lx[lx.size()-1].y==x)swap(lx[lx.size()-1].x,lx[lx.size()-1].y); if(lx[lx.size()-1].y!=u){ nv.push_back(rt(u,lx[lx.size()-1].y)); pos[id[lx.size()-1]]=nv.size()-1; } } nv=solve(nv); for(i=0;i<(int)v.size();i++){ if(pos[i]!=-1&&!(v[i].x==nv[pos[i]].x||v[i].y==nv[pos[i]].y))swap(v[i].x,v[i].y); } return v; } vector<rt>v; int main(){ int m,i,x,y,ans; scanf("%d%d",&n,&m); for(i=1;i<n;i++){ scanf("%d%d",&x,&y); add(x,y); add(y,x); d[x]++; d[y]++; } dfs(1); for(i=1;i<=m;i++){ scanf("%d%d",&x,&y); v.push_back(rt(x,y)); col(x,y); } v=solve(v); ans=0; for(i=2;i<=n;i++)ans+=min(2,fav[i]); printf("%d\n",ans); for(rt t:v)printf("%d %d\n",t.x,t.y); }
[AGC025E]Walking on a Tree的更多相关文章
- K-D Tree
这篇随笔是对Wikipedia上k-d tree词条的摘录, 我认为解释得相当生动详细, 是一篇不可多得的好文. Overview A \(k\)-d tree (short for \(k\)-di ...
- 使控件具有 Tilt 效果
步骤1:添加类: /* Copyright (c) 2010 Microsoft Corporation. All rights reserved. Use of this sample source ...
- WinPhone学习笔记(四)——磁贴
对每个Windows Phone的使用者来说,给他们的第一印象就是大大小小的磁贴——Metro,本篇介绍的是Windows Phone的磁贴,提到的有开始菜单的磁贴,也有在App里面的磁贴. 开始菜单 ...
- java基础(1)-比较jdk5,jdk6,jdk7的新特性
jdk8已经出来好长时间了,这里自己学习时简单总结的jdk5,jdk6和jdk7的新特性:本文提纲: 一.jdk5的新特性 二.jdk6的新特性 三.jdk7的新特性 一.jdk5的新特性 首先简单介 ...
- 立即执行函数(IIFE)的理解与运用
作为JavaScript的常用语法,立即执行函数IIFE(Immediately-Invoked Function Expression)是值得我们认真去学习探究的. 一.创建函数的两种方式 我们先从 ...
- An NIO.2 primer--reference
Part 1: The asynchronous channel APIs The More New I/O APIs for the Java™ Platform (NIO.2) is one of ...
- (转载)XML Tutorial for iOS: How To Choose The Best XML Parser for Your iPhone Project
There are a lot of options when it comes to parsing XML on the iPhone. The iPhone SDK comes with two ...
- Windows Phone 8 ControlTiltEffect
/* Copyright (c) 2010 Microsoft Corporation. All rights reserved. Use of this sample source code is ...
- jQuery的使用及关于框架造型(转)
Introduction 正如jQuery所宣称的一样,Write Less, Do More.很多时候我们喜欢用它来解决问题.但增加一个库必然意味着更大的网络负担,意味着更高的页面初始载入时间.并且 ...
随机推荐
- Js跑马灯效果 && 在Vue中使用
DEMO: <!DOCTYPE html><html> <head> <title>滚动播报</title> <meta charse ...
- java Collections.sort()实现List排序的默认方法和自定义方法【转】
1.java提供的默认list排序方法 主要代码: List<String> list = new ArrayList();list.add("刘媛媛"); list. ...
- codeforces902B. Coloring a Tree
B. Coloring a Tree 题目链接: https://codeforces.com/contest/902/problem/B 题意:给你一颗树,原先是没有颜色的,需要你给树填色成指定的样 ...
- CommonJs/ES6/AMD模块的用法以及区别
github地址: 一直以来对CommonJs/AMD/CMD/ES6的文件模块加载一直懵懵懂懂.甚至有时会将CommonJs的exports和ES6的export.default搞混.趁着学习web ...
- 【Mysql优化】索引优化策略
1:索引类型 1.1 B-tree索引 注: 名叫btree索引,大的方面看,都用的平衡树,但具体的实现上, 各引擎稍有不同, 比如,严格的说,NDB引擎,使用的是T-tree Myisam,in ...
- junit单元测试+junit与Spring结合
配置:右键要加入单元测试的工程,选择properties,然后选择java build path,选择add library,选择junit即可. 编写:右键要测试的class,new一个junit ...
- 【 总结 】crontab 使用脚本及直接获取HTTP状态码
一.在crontab里面计划执行的脚本,所有的命令都要写出绝对路径.因为crontab的独立的进程,可能无法直接加载环境变量. 二.在判断网站能否正常访问一般的思路: 1. 判断网站是否能够正常打开. ...
- ApplicationCommands 应用程序常见命令
ApplicationCommands用于表示应用程序程序员经常遇到的常见命令,类似于ctrl+c 在WPF中,许多控件都自动集成了固有的命令集.比如文本框TextBox就提供了复制(Copy),粘贴 ...
- 串口通讯超时的设置与含义(COMMTIMEOUTS)
COMMTIMEOUTS:COMMTIMEOUTS主要用于串口超时参数设置.COMMTIMEOUTS结构如下: typedef struct _COMMTIMEOUTS { DWORD ReadInt ...
- oracle 11g在大表中添加字段及默认值--加速
今天遇到这个问题了.简单的增加语句,默认SQLPLUS执行,却会超时. 要增加客户端的TIMEOUT时间才可以解决.(感觉超过两三分钟,默认超时30秒) 另外, 也可以用两步操作(1,增加字段,2,修 ...