POJ - 1741 - Tree - 点分治 模板
题意:
对于带权的一棵树,求树中距离不超过k的点的对数。
思路:
点分治的裸题。 将这棵树分成很多小的树,分治求解。
#include <algorithm>
#include <iterator>
#include <iostream>
#include <cstring>
#include <cstdlib>
#include <iomanip>
#include <bitset>
#include <cctype>
#include <cstdio>
#include <string>
#include <vector>
#include <cmath>
#include <queue>
#include <list>
#include <map>
#include <set>
using namespace std;
//#pragma GCC optimize(3)
//#pragma comment(linker, "/STACK:102400000,102400000") //c++
#define lson (l , mid , rt << 1)
#define rson (mid + 1 , r , rt << 1 | 1)
#define debug(x) cerr << #x << " = " << x << "\n";
#define pb push_back
#define pq priority_queue typedef long long ll;
typedef unsigned long long ull; typedef pair<ll ,ll > pll;
typedef pair<int ,int > pii;
typedef pair<int,pii> p3; //priority_queue<int> q;//这是一个大根堆q
//priority_queue<int,vector<int>,greater<int> >q;//这是一个小根堆q
#define fi first
#define se second
//#define endl '\n' #define OKC ios::sync_with_stdio(false);cin.tie(0)
#define FT(A,B,C) for(int A=B;A <= C;++A) //用来压行
#define REP(i , j , k) for(int i = j ; i < k ; ++i)
//priority_queue<int ,vector<int>, greater<int> >que; const ll mos = 0x7FFFFFFFLL; //
const ll nmos = 0x80000000LL; //-2147483648
const int inf = 0x3f3f3f3f;
const ll inff = 0x3f3f3f3f3f3f3f3fLL; //
const int mod = ; const double PI=acos(-1.0); // #define _DEBUG; //*//
#ifdef _DEBUG
freopen("input", "r", stdin);
// freopen("output.txt", "w", stdout);
#endif
/*-----------------------showtime----------------------*/
const int maxn = 1e5+;
int root = ,S,mx;
int n,k;
int sz[maxn],f[maxn],dis[maxn],cnt;
bool used[maxn];
struct node
{
int to,w,nx;
}e[maxn];
int h[maxn],tot = ;
void add(int u,int v,int w){
e[tot].to = v;
e[tot].w = w;
e[tot].nx = h[u];
h[u] = tot++;
}
void getRoot(int u, int fa){
sz[u] = ,f[u] = ;
for(int i = h[u] ; ~i; i= e[i].nx){
int v = e[i].to;
if(used[v] || fa == v)continue;
getRoot(v,u);
sz[u] += sz[v];
f[u] = max(f[u] , sz[v]);
}
f[u] = max(f[u],S - sz[u]);
if(f[u] < mx){root = u;mx = f[u];}
} void getDis(int u,int fa,int D){
for(int i=h[u] ; ~i; i=e[i].nx){
int v = e[i].to;
if(used[v]||v == fa)continue;
dis[++cnt] = D + e[i].w;
getDis(v,u,dis[cnt]);
}
} int getAns(int x,int D){
dis[cnt = ] = D;
getDis(x,,D);
sort(dis+,dis++cnt);
int le = ,ri =cnt,ans = ;
while(le <= ri){
if(dis[le] + dis[ri] <= k)ans += ri - le,le++;
else ri--;
}
return ans;
} int Divide(int x){
used[x] = true;
ll ans = getAns(x,);
for(int i=h[x]; ~i; i= e[i].nx){
int v = e[i].to;
if(used[v])continue;
ans -= getAns(v,e[i].w);
mx = inf,S = sz[v];
getRoot(v,x);ans += Divide(root);
}
return ans;
}
int main(){ while(~scanf("%d%d", &n, &k) && n+k)
{
memset(h,-,sizeof(h));
memset(used,false,sizeof(used));
tot = ;
for(int i=; i<n; i++){
int u,v,c;
scanf("%d%d%d", &u, &v,&c);
add(u,v,c);
add(v,u,c);
}
S = n;mx = inf;
getRoot(,-);
printf("%d\n",Divide(root));
}
return ;
}
POJ-1741
自己今天又写了一遍。
//点分治
//#pragma GCC optimize(3)
//#pragma comment(linker, "/STACK:102400000,102400000") //c++
// #pragma GCC diagnostic error "-std=c++11"
// #pragma comment(linker, "/stack:200000000")
// #pragma GCC target("sse,sse2,sse3,ssse3,sse4,popcnt,abm,mmx,avx,tune=native")
// #pragma GCC optimize("-fdelete-null-pointer-checks,inline-functions-called-once,-funsafe-loop-optimizations,-fexpensive-optimizations,-foptimize-sibling-calls,-ftree-switch-conversion,-finline-small-functions,inline-small-functions,-frerun-cse-after-loop,-fhoist-adjacent-loads,-findirect-inlining,-freorder-functions,no-stack-protector,-fpartial-inlining,-fsched-interblock,-fcse-follow-jumps,-fcse-skip-blocks,-falign-functions,-fstrict-overflow,-fstrict-aliasing,-fschedule-insns2,-ftree-tail-merge,inline-functions,-fschedule-insns,-freorder-blocks,-fwhole-program,-funroll-loops,-fthread-jumps,-fcrossjumping,-fcaller-saves,-fdevirtualize,-falign-labels,-falign-loops,-falign-jumps,unroll-loops,-fsched-spec,-ffast-math,Ofast,inline,-fgcse,-fgcse-lm,-fipa-sra,-ftree-pre,-ftree-vrp,-fpeephole2",3) #include <algorithm>
#include <iterator>
#include <iostream>
#include <cstring>
#include <cstdlib>
#include <iomanip>
#include <bitset>
#include <cctype>
#include <cstdio>
#include <string>
#include <vector>
#include <stack>
#include <cmath>
#include <queue>
#include <list>
#include <map>
#include <set>
#include <cassert> using namespace std;
#define lson (l , mid , rt << 1)
#define rson (mid + 1 , r , rt << 1 | 1)
#define debug(x) cerr << #x << " = " << x << "\n";
#define pb push_back
#define pq priority_queue typedef long long ll;
typedef unsigned long long ull;
//typedef __int128 bll;
typedef pair<ll ,ll > pll;
typedef pair<int ,int > pii;
typedef pair<int,pii> p3; //priority_queue<int> q;//这是一个大根堆q
//priority_queue<int,vector<int>,greater<int> >q;//这是一个小根堆q
#define fi first
#define se second
//#define endl '\n' #define OKC ios::sync_with_stdio(false);cin.tie(0)
#define FT(A,B,C) for(int A=B;A <= C;++A) //用来压行
#define REP(i , j , k) for(int i = j ; i < k ; ++i)
#define max3(a,b,c) max(max(a,b), c);
#define min3(a,b,c) min(min(a,b), c);
//priority_queue<int ,vector<int>, greater<int> >que; const ll mos = 0x7FFFFFFF; //
const ll nmos = 0x80000000; //-2147483648
const int inf = 0x3f3f3f3f;
const ll inff = 0x3f3f3f3f3f3f3f3f; //
const int mod = 1e9+;
const double esp = 1e-;
const double PI=acos(-1.0);
const double PHI=0.61803399; //黄金分割点
const double tPHI=0.38196601; template<typename T>
inline T read(T&x){
x=;int f=;char ch=getchar();
while (ch<''||ch>'') f|=(ch=='-'),ch=getchar();
while (ch>=''&&ch<='') x=x*+ch-'',ch=getchar();
return x=f?-x:x;
} /*-----------------------showtime----------------------*/
/*
using namespace std;
#define pb push_back
#define debug(x) cerr<<#x<<" = " << x<<endl;
#define fi first
#define se second
typedef pair<int,int> pii;
const int inf = 0x3f3f3f3f; */
int n,k,ans;
const int maxn = ;
vector<pii>mp[maxn];
int dp[maxn],cen[maxn];
void getsize(int u,int fa){
dp[u] = ;
for(int i=; i<mp[u].size(); i++){
int v = mp[u][i].fi;
if(fa == v || cen[v]) continue;
getsize(v, u);
dp[u] += dp[v];
}
} pii getbig(int u,int fa,int t){
pii res = pii(inf, u);
int mx = ;
for(int i=; i<mp[u].size(); i++){
int v = mp[u][i].fi;
if(v == fa || cen[v])continue;
res = min(res, getbig(v, u, t));
mx = max(mx, dp[v]);
} res = min(res, pii(max(mx, t - dp[u]), u));
return res;
} void dfs(int u,int fa,int c, vector<int> & b){
b.pb(c);
for(int i=; i<mp[u].size(); i++){
int v = mp[u][i].fi,d = c + mp[u][i].se;
if(v == fa || cen[v])continue;
dfs(v,u,d,b);
}
}
int cal(vector<int>&b){ sort(b.begin(), b.end()); int res = ,r = b.size();
for(int i=; i<b.size(); i++){
while(r && b[i] + b[r-] > k) r--;
if(r > i) res += r - ;
else res += r;
}
return res/;
} void solve(int u){
getsize(u, -);
int s = getbig(u, -,dp[u]).se;
cen[s] = ; for(int i=; i<mp[s].size(); i++){
int v = mp[s][i].fi;
if(cen[v])continue;
solve(v);
} vector<int>a;
a.pb();
for(int i=; i<mp[s].size(); i++)
{
int v = mp[s][i].fi;
if(cen[v])continue; vector<int>b;
dfs(v, s, mp[s][i].se, b);
ans -= cal(b);
a.insert(a.end(),b.begin(),b.end());
} ans += cal(a);
cen[s] = ;
}
int main(){
while(~scanf("%d%d", &n, &k) && n + k){
for(int i=; i<=n; i++) mp[i].clear();
for(int i=; i<n; i++){
int u,v,w;
scanf("%d%d%d", &u, &v, &w);
mp[u].pb(pii(v,w));
mp[v].pb(pii(u,w));
}
ans = ;
solve();
printf("%d\n", ans);
}
return ;
}
new
POJ - 1741 - Tree - 点分治 模板的更多相关文章
- POJ 1741 Tree ——点分治
[题目分析] 这貌似是做过第三道以Tree命名的题目了. 听说树分治的代码都很长,一直吓得不敢写,有生之年终于切掉这题. 点分治模板题目.自己YY了好久才写出来. 然后1A了,开心o(* ̄▽ ̄*)ブ ...
- POJ 1741.Tree 树分治 树形dp 树上点对
Tree Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 24258 Accepted: 8062 Description ...
- POJ 1741 Tree 树分治
Tree Description Give a tree with n vertices,each edge has a length(positive integer less than 1 ...
- [bzoj 1468][poj 1741]Tree [点分治]
Description Give a tree with n vertices,each edge has a length(positive integer less than 1001). Def ...
- POJ 1741 Tree(点分治点对<=k)
Description Give a tree with n vertices,each edge has a length(positive integer less than 1001). Def ...
- [poj 1741]Tree 点分治
题意 求树上距离不超过k的点对数,边权<=1000 题解 点分治. 点分治的思想就是取一个树的重心,这种路径只有两种情况,就是经过和不经过这个重心,如果不经过重心就把树剖开递归处 ...
- POJ 1741.Tree and 洛谷 P4178 Tree-树分治(点分治,容斥版) +二分 模板题-区间点对最短距离<=K的点对数量
POJ 1741. Tree Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 34141 Accepted: 11420 ...
- poj 1741 Tree(树的点分治)
poj 1741 Tree(树的点分治) 给出一个n个结点的树和一个整数k,问有多少个距离不超过k的点对. 首先对于一个树中的点对,要么经过根结点,要么不经过.所以我们可以把经过根节点的符合点对统计出 ...
- POJ 1741 Tree 求树上路径小于k的点对个数)
POJ 174 ...
随机推荐
- 【iOS】打印方法名
为了便于追踪程序运行过程,可以在日志打印方法名,示例代码如下: NSLog(@"%@", NSStringFromSelector(_cmd)); 结果如图所示: 此外,在多个中, ...
- JavaSE(一)Java程序的三个基本规则-组织形式,编译运行,命名规则
一.Java程序的组织形式 Java程序是一种纯粹的面向对象的程序设计语言,因此Java程序必须以类(class)的形式存在,类(class)是Java程序的最小程序单位. J ...
- javaweb入门-----jsp概念
jsp是什么? JSP:Java Server Pages java服务器端页面 *可以理解为 一个特殊的页面,其中既可以直接定义html标签,又可以定义java代码 *用于简化书写 <% %& ...
- 【游记】NOIP2019前传
声明 我的游记是一个完整的体系,如果没有阅读过往届文章,阅读可能会受到障碍. ~~~上一篇游记的传送门~~~ 前言 比完赛后,我沉浸在胜利中长达半个月,而后才清醒过来,意识到自己需要为NOIP2019 ...
- java订单生成工具类
欢迎来到付宗乐个人博客网站.本个人博客网站提供最新的站长新闻,各种互联网资讯. 还提供个人博客模板,最新最全的java教程,java面试题.在此我将尽我最大所能将此个人博客网站做的最好! 谢谢大家,愿 ...
- Kubernetes Pod 驱逐详解
原文链接:Kubernetes Pod 驱逐详解 在 Kubernetes 中,Pod 使用的资源最重要的是 CPU.内存和磁盘 IO,这些资源可以被分为可压缩资源(CPU)和不可压缩资源(内存,磁盘 ...
- [原创实践]redhat linux 5.3搭建Nexus
1:下载安装JDK,配置好环境变量(JAVA_HOME等) 下载linux下64位的jdk-7u45-linux-x64.tar.gz(百度网盘下载,官网的jdk-7u51-linux-x64.tar ...
- android ——滑动菜单
一.DrawerLayout是一个拥有两个子控件的布局,第一个子控件是主屏幕中显示的内容,第二个子控件是滑动菜单中显示的内容: <android.support.v4.widget.Drawer ...
- 通俗易懂--循环神经网络(RNN)的网络结构!(TensorFlow实现)
1. 什么是RNN 循环神经网络(Recurrent Neural Network, RNN)是一类以序列(sequence)数据为输入,在序列的演进方向进行递归(recursion)且所有节点(循环 ...
- EFCore + MySql codeFirst 迁移 Migration出现的问题
第二次使用Migration update-database的时候出现以下错误: System.NotImplementedException: The method or operation is ...