18 条题解
-
4passer_1 LV 9 @ 2017-10-16 22:42:21
#include <cstdio> #include <cstring> #include <cstdlib> #include <cmath> #include <algorithm> using namespace std; const int root=1; const int MAXN=300033; struct Edge{ int nx,to,val; Edge(){} Edge(int nx,int to,int val):nx(nx),to(to),val(val){} }E[MAXN<<1]; int first[MAXN],ecnt=1; inline void addedge(int f,int t,int val) { E[++ecnt]=Edge(first[f],t,val);first[f]=ecnt; E[++ecnt]=Edge(first[t],f,val);first[t]=ecnt; } int n,m; int dep[MAXN],f[21][MAXN],dis[MAXN]; int sum[MAXN]; int l=0,r=0,mid,ans; int Length[MAXN]; void LCA_INIT(int e) { int r; for(int i=1;i<=20;++i) f[i][e]=f[i-1][f[i-1][e]]; for(int i=first[e];i;i=E[i].nx) { if((r=E[i].to)==f[0][e]) continue; dep[r]=dep[e]+1; f[0][r]=e; Length[r]=E[i].val; dis[r]=dis[e]+E[i].val; LCA_INIT(r); } } int LCA(int l,int r) { if(dep[r]<dep[l]) swap(l,r); for(int i=20;i>=0;--i) { if(dep[r]-(1<<i)<dep[l]) continue; r=f[i][r]; } if(l==r) return r; for(int i=20;i>=0;--i) { if(f[i][r]==f[i][l]) continue; r=f[i][r];l=f[i][l]; } return f[0][r]; } struct Stage{ int l,r; int LCA,len; }S[MAXN]; void Push(int e) { int r; for(int i=first[e];i;i=E[i].nx) { if((r=E[i].to)==f[0][e]) continue; Push(r); sum[e]+=sum[r]; } } inline bool check() { memset(sum,0,sizeof sum); int MX=0,cnt=0; for(int i=1;i<=m;++i) { if(S[i].len>mid) { ++sum[S[i].l]; ++sum[S[i].r]; sum[S[i].LCA]-=2; MX=max(MX,S[i].len-mid); ++cnt; } } Push(root); for(int i=1;i<=n;++i) if(sum[i]==cnt&&Length[i]>=MX) return true; return false; } int main() { int x,y,z; scanf("%d%d",&n,&m); for(int i=1;i<n;++i) { scanf("%d%d%d",&x,&y,&z); addedge(x,y,z); r+=z; } dep[root]=1; LCA_INIT(root); for(int i=1;i<=m;++i) { scanf("%d%d",&S[i].l,&S[i].r); S[i].LCA=LCA(S[i].l,S[i].r); S[i].len=dis[S[i].l]+dis[S[i].r]-(dis[S[i].LCA]<<1); } while(l<=r) { mid=(l+r)>>1; if(check()) ans=mid,r=mid-1; else l=mid+1; } printf("%d",ans); return 0; }
-
32017-10-25 21:12:21@
先用lca求出每种运输计划的长度,然后按长度从长到短对每种运输计划排序。二分答案,先找到有多少条运输计划长度比答案长,记录为last,然后对这些比答案长的运输计划进行树上差分,再dfs求树上前缀和。对所有经过次数为last的边找最大值maxn,再用最长的计划减去maxn看是否小于等于答案。
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
int n,m,a[300005],c,dis[300005],d[300005],f[300005][20],maxm,tree[300005],last,maxn;
struct node{
int next,to,w;
}b[600005];
struct edge{
int u,v,z,l;
}e[300005];
void add(int x,int y,int z){
b[++c].to=y;
b[c].next=a[x];
b[c].w=z;
a[x]=c;
b[++c].to=x;
b[c].next=a[y];
b[c].w=z;
a[y]=c;
}
void dfs(int x,int from){
for(int i=a[x];i;i=b[i].next){
int y=b[i].to;
if(y==from)continue;
d[y]=d[x]+1;
dis[y]=dis[x]+b[i].w;
f[y][0]=x;
dfs(y,x);
}
}
int lca(int x,int y){
if(d[x]>d[y])swap(x,y);
int l=d[y]-d[x];
for(int i=19;i>=0;i--){
if((1<<i)&l)y=f[y][i];
}
if(x!=y){
for(int i=19;i>=0;i--){
if(f[x][i]!=f[y][i]){
x=f[x][i];
y=f[y][i];
}
}
x=f[x][0];
}
return x;
}
bool cmp(edge a,edge b){
return a.l>b.l;
}
void dfs2(int x,int from){
for(int i=a[x];i;i=b[i].next){
int y=b[i].to;
if(y==from)continue;
dfs2(y,x);
tree[x]+=tree[y];
}
if(tree[x]==last){
if(dis[x]-dis[from]>maxn)maxn=dis[x]-dis[from];
}
}
bool judge(int x){
memset(tree,0,sizeof(tree));
last=m;
for(int i=1;i<=m;i++){
if(e[i].l>x){
tree[e[i].u]++;
tree[e[i].v]++;
tree[e[i].z]-=2;
}
else {
last=i-1;
break;
}
}
maxn=0;
dfs2(1,0);
if(e[1].l-maxn<=x)return true;
return false;
}
int main(){
cin>>n>>m;
for(int i=1;i<=n-1;i++){
int x,y,z;
scanf("%d%d%d",&x,&y,&z);
add(x,y,z);
}
dfs(1,0);
for(int j=1;j<=19;j++)
for(int i=1;i<=n;i++)f[i][j]=f[f[i][j-1]][j-1];
for(int i=1;i<=m;i++){
int x,y;
scanf("%d%d",&x,&y);
int z=lca(x,y);
int l=dis[x]+dis[y]-2*dis[z];
e[i].l=l;
e[i].u=x;
e[i].v=y;
e[i].z=z;
maxm=max(maxm,l);
}
sort(e+1,e+m+1,cmp);
int left=0,right=maxm;
while(left<right){
/*if(right-left==1){
if(judge(right))left=right;
break;
}*/
int mid=(left+right)/2;
if(judge(mid))right=mid;
else left=mid+1;
}
cout<<left<<endl;
} -
22016-11-14 19:56:20@
二份答案+LCA
迷之卡时,“被逼无奈”写了读入优化#include <iostream> #include <cstring> #include <cstdio> #include <cctype> #include <algorithm> using namespace std; #define N 700100 struct node { int to,next,w; }e[N]; template<typename T> inline void read(T &x) { char ch; while ( !isdigit((ch=getchar())) ); for (x=ch-'0';isdigit((ch=getchar()));x=(x<<3)+(x<<1)+ch-'0'); ungetc(ch,stdin); } int head[N],dep[N],n,m,tot = 0,cnt; int s[N][20],sum[N],dis[N],dist[N]; int A[N],B[N],LCA[N],fa[N],v[N]; void add(int u,int v,int w) { e[tot].to = v; e[tot].w = w; e[tot].next = head[u]; head[u] = tot++; e[tot].to = u; e[tot].w = w;e[tot].next = head[v]; head[v] = tot++; } void dfs(int x) { for (int i=head[x];~i;i=e[i].next) if ( e[i].to != fa[x] ) { dep[e[i].to] = dep[x] + 1; fa[e[i].to] = x; v[e[i].to] = e[i].w; dis[e[i].to] = dis[x] + e[i].w; dfs(e[i].to); } } int lca(int x,int y) { if ( dep[x] > dep[y] ) swap(x,y); for (int i=19;i>=0;i--) if ( dep[y] - dep[x] >= (1<<i) ) { y = s[y][i]; } if ( x == y ) return x; for (int i=19;i>=0;i--) if ( s[x][i] != s[y][i] ) { x = s[x][i]; y = s[y][i]; } return fa[x]; } void what(int x) { for (int i=head[x];~i;i=e[i].next) if ( e[i].to != fa[x] ) { what(e[i].to); sum[x] += sum[e[i].to]; } } bool check(int x) { int dec = 0; cnt = 0; for (int i=1;i<=n;i++) sum[i] = 0; for (int i=1;i<=m;i++) if ( dist[i] > x ) { cnt ++; dec = max(dec,dist[i]-x); sum[A[i]] ++; sum[B[i]] ++; sum[LCA[i]] -= 2; } what(1); for (int i=1;i<=n;i++) if ( (sum[i] == cnt ) && ( v[i] >= dec ) ) return 1; return 0; } int main() { int a,b,c; memset(head,-1,sizeof head); read(n); read(m); for (int i=1;i<n;i++) { read(a);read(b);read(c); add(a,b,c); } dfs(1); for (int i=1;i<=n;i++) s[i][0] = fa[i]; for (int h=1;h<20;h++) for (int i=1;i<=n;i++) s[i][h] = s[s[i][h-1]][h-1]; for (int i=1;i<=m;i++) { read(A[i]); read(B[i]); LCA[i] = lca(A[i],B[i]); //cout<<LCA[i]<<endl; dist[i] = dis[A[i]] + dis[B[i]] - 2*dis[LCA[i]]; } int L = 0,R = 0; for (int i=1;i<=m;i++) R = max(R,dist[i]); int mid; while ( L <= R ) { mid = ( L + R ) >> 1; if ( check(mid) ) R = mid - 1; else L = mid + 1; } cout<<L<<endl; return 0; }
-
12018-08-20 09:13:38@
#include<iostream> #include<cstring> #include<vector> #include<cstdio> #include<algorithm> #define MAXN 300010 using namespace std; int N,M; struct edge { int u,v; int w; } E[MAXN<<1]; int EC=0,Ef[MAXN<<1],En[MAXN<<1]; void inline ADD_EDGE(int u,int v,int w) { E[++EC].u=u; E[EC].v=v; E[EC].w=w; En[EC]=Ef[u]; Ef[u]=EC; } struct query { int u,v; int LCA; int l; bool operator > (const query& q) const { return l>q.l; } } Q[MAXN<<1]; int Qf[MAXN<<1],Qn[MAXN<<1]; void inline ADD_QUERY(int u,int v,int i) { Q[i].u=u; Q[i].v=v; Q[i].LCA=Q[i].l=0; Qn[(i<<1)]=Qf[u]; Qf[u]=(i<<1); Qn[(i<<1)|1]=Qf[v]; Qf[v]=(i<<1)|1; } int dist[MAXN],P[MAXN],visited[MAXN],nw[MAXN],fa[MAXN]; int FIND_SET(int u) { if(P[u]==u) return u; else return P[u]=FIND_SET(P[u]); } void DFS(int u) { P[u]=u; for(int i=Ef[u];i;i=En[i]) { edge& e=E[i]; if(e.v!=fa[u]) { dist[e.v]=dist[u]+e.w; nw[e.v]=e.w; fa[e.v]=u; DFS(e.v); P[e.v]=u; } } visited[u]=true; for(int i=Qf[u];i;i=Qn[i]) { query& q=Q[i>>1]; if(!q.LCA) { if(q.v==u) swap(q.u,q.v); if(visited[q.v]) { q.LCA=FIND_SET(q.v); q.l=dist[u]+dist[q.v]-2*dist[FIND_SET(q.v)]; } } } } int inline Q_FIND(int minl) { int l=1,u=M; while(l<u) { int m=(l+u+1)>>1; if(Q[m].l>minl) l=m; else u=m-1; } return l; } int tot,cnt[MAXN],rtn[MAXN],min_nw; bool CHECK_DFS(int u) { rtn[u]=cnt[u]; for(int i=Ef[u];i;i=En[i]) { edge& e=E[i]; if(e.v!=fa[u]) { //fa[]在预处理DFS时已建立完成 if(CHECK_DFS(e.v)) return true; rtn[u]+=rtn[e.v]; } } return (rtn[u]>=tot&&nw[u]>=min_nw); } bool inline CHECK(int t) { memset(cnt,0,sizeof(cnt)); tot=Q_FIND(t); for(int i=1;i<=tot;i++) { cnt[Q[i].u]++; cnt[Q[i].v]++; cnt[Q[i].LCA]-=2; } min_nw=Q[1].l-t; return CHECK_DFS(1); } int MAX_W=0; int main() { memset(Ef,0,sizeof(Ef)); memset(Qf,0,sizeof(Qf)); scanf("%d%d",&N,&M); int u,v,w,l,m; for(int i=1;i<N;i++) { scanf("%d%d%d",&u,&v,&w); MAX_W=max(MAX_W,w); ADD_EDGE(u,v,w); ADD_EDGE(v,u,w); } for(int i=1;i<=M;i++) { scanf("%d%d",&u,&v); ADD_QUERY(u,v,i); } memset(visited,false,sizeof(visited)); dist[1]=0; nw[1]=0; fa[1]=1; DFS(1); sort(Q+1,Q+1+M,greater<query>()); l=Q[1].l-MAX_W; u=Q[1].l-1; while(l<u) { m=(l+u)>>1; if(CHECK(m)) u=m; else l=m+1; } printf("%d\n",l); return 0; }
-
12017-07-28 19:37:15@
NOIP2015Day1T3
有些难
找到两个好东西:
1. 给力的题解-> https://blog.sengxian.com/solutions/noip-2015-day2
2. 给力的UOJ-> http://uoj.ac/problem/150 (Vjios本题的数据略有些水)然后贴两份代码,一个在UOJ超时,一个KA常数成功(Vijos测大约快2倍),这让我明白了为什么很多标算都是写bfs的了。
代码1:
UOJ:95分 AC*19 TLE*1
Vijos:100分 AC*20#pragma comment(linker, "/STACK:10240000,10240000") #include <cstring> #include <algorithm> #include <cstdio> #include <cstdlib> #include <cmath> #include <vector> using namespace std; const int N=300000+5,M=N*2,Inf=N*1000; void read(int &x){ x=0; char ch=getchar(); while (!('0'<=ch&&ch<='9')) ch=getchar(); while ('0'<=ch&&ch<='9'){ x=x*10+ch-48; ch=getchar(); } } struct Edge{ int cnt,y[M],z[M],nxt[M],fst[N]; void set(){ cnt=0; memset(y,0,sizeof y); memset(z,0,sizeof z); memset(nxt,0,sizeof nxt); memset(fst,0,sizeof fst); } void add(int a,int b,int c){ cnt++; y[cnt]=b,z[cnt]=c; nxt[cnt]=fst[a],fst[a]=cnt; } }e; int n,m; vector <int> Tree[N]; int father[N],son[N],deep[N],dis[N],fadis[N]; int Anst[N][20];//Ancestor struct Query{ int x,y,LCA,cost; }q[N]; int Nextsum[N]; void Build_Tree(int prev,int rt){ Tree[rt].clear(),deep[rt]=deep[prev]+1,son[rt]=0,Anst[rt][0]=father[rt]=prev; for (int i=1;(1<<i)<=deep[rt];i++) Anst[rt][i]=Anst[Anst[rt][i-1]][i-1]; for (int i=e.fst[rt];i;i=e.nxt[i]) if (e.y[i]!=prev){ son[rt]++,Tree[rt].push_back(e.y[i]), fadis[e.y[i]]=e.z[i],dis[e.y[i]]=dis[rt]+e.z[i]; Build_Tree(rt,e.y[i]); } } int LCA(int a,int b){ if (deep[a]>deep[b]) swap(a,b); for (int i=deep[b]-deep[a],j=0;i>0;i>>=1,j++) if (i&1) b=Anst[b][j]; if (a==b) return a; int k; for (k=0;(1<<k)<=deep[a];k++); for (;k>=0;k--) if ((1<<k)<=deep[a]&&Anst[a][k]!=Anst[b][k]) a=Anst[a][k],b=Anst[b][k]; return Anst[a][0]; } void Get_Nextsum(int rt){ for (int i=0;i<son[rt];i++){ Get_Nextsum(Tree[rt][i]); Nextsum[rt]+=Nextsum[Tree[rt][i]]; } } bool check(int t){ int total=0,Maxcost=0,Maxcut=0; memset(Nextsum,0,sizeof Nextsum); for (int i=1;i<=m;i++) if (q[i].cost>t){ Maxcost=max(Maxcost,q[i].cost-t); total++; Nextsum[q[i].x]++; Nextsum[q[i].y]++; Nextsum[q[i].LCA]-=2; } Get_Nextsum(1); for (int i=1;i<=n;i++) if (Nextsum[i]==total) Maxcut=max(Maxcut,fadis[i]); return Maxcost<=Maxcut; } int main(){ read(n),read(m); e.set(); int Csum=0; for (int i=1;i<n;i++){ int a,b,c; read(a),read(b),read(c); e.add(a,b,c); e.add(b,a,c); Csum+=c; } deep[0]=-1,dis[1]=fadis[1]=0; memset(Anst,0,sizeof Anst); Build_Tree(0,1); for (int i=1;i<=m;i++){ read(q[i].x),read(q[i].y); q[i].LCA=LCA(q[i].x,q[i].y); q[i].cost=dis[q[i].x]+dis[q[i].y]-dis[q[i].LCA]*2; } int le=0,ri=Csum,mid,ans=0; while (le<=ri){ mid=(le+ri)>>1; if (check(mid)) ri=mid-1,ans=mid; else le=mid+1; } printf("%d",ans); return 0; }
代码2:
UOJ:100分 AC*20
Vijos:100分 AC*20#pragma comment(linker, "/STACK:10240000,10240000") #include <cstring> #include <algorithm> #include <cstdio> #include <cstdlib> #include <cmath> #include <vector> using namespace std; const int N=300000+5,M=N*2,Inf=N*1000; void read(int &x){ x=0; char ch=getchar(); while (!('0'<=ch&&ch<='9')) ch=getchar(); while ('0'<=ch&&ch<='9'){ x=x*10+ch-48; ch=getchar(); } } struct Edge{ int cnt,y[M],z[M],nxt[M],fst[N]; void set(){ cnt=0; memset(y,0,sizeof y); memset(z,0,sizeof z); memset(nxt,0,sizeof nxt); memset(fst,0,sizeof fst); } void add(int a,int b,int c){ cnt++; y[cnt]=b,z[cnt]=c; nxt[cnt]=fst[a],fst[a]=cnt; } }e; int n,m; vector <int> Tree[N]; int father[N],son[N],deep[N],dis[N],fadis[N],bh[N],bhtot; int Anst[N][20];//Ancestor struct Query{ int x,y,LCA,cost; }q[N]; int Nextsum[N]; void Build_Tree(int prev,int rt){ bh[++bhtot]=rt; Tree[rt].clear(),deep[rt]=deep[prev]+1,son[rt]=0,father[rt]=prev; for (int i=e.fst[rt];i;i=e.nxt[i]) if (e.y[i]!=prev){ son[rt]++,Tree[rt].push_back(e.y[i]), fadis[e.y[i]]=e.z[i],dis[e.y[i]]=dis[rt]+e.z[i]; Build_Tree(rt,e.y[i]); } } void LCA_Prepare(){ memset(Anst,0,sizeof Anst); for (int i=1;i<=n;i++){ int rt=bh[i]; Anst[rt][0]=father[rt]; for (int i=1;(1<<i)<=deep[rt];i++) Anst[rt][i]=Anst[Anst[rt][i-1]][i-1]; } } int LCA(int a,int b){ if (deep[a]>deep[b]) swap(a,b); for (int i=deep[b]-deep[a],j=0;i>0;i>>=1,j++) if (i&1) b=Anst[b][j]; if (a==b) return a; int k; for (k=0;(1<<k)<=deep[a];k++); for (;k>=0;k--) if ((1<<k)<=deep[a]&&Anst[a][k]!=Anst[b][k]) a=Anst[a][k],b=Anst[b][k]; return Anst[a][0]; } bool check(int t){ int total=0,Maxcost=0,Maxcut=0; memset(Nextsum,0,sizeof Nextsum); for (int i=1;i<=m;i++) if (q[i].cost>t){ Maxcost=max(Maxcost,q[i].cost-t); total++; Nextsum[q[i].x]++; Nextsum[q[i].y]++; Nextsum[q[i].LCA]-=2; } for (int i=n;i>=1;i--) Nextsum[father[bh[i]]]+=Nextsum[bh[i]]; for (int i=1;i<=n;i++) if (Nextsum[i]==total) Maxcut=max(Maxcut,fadis[i]); return Maxcost<=Maxcut; } int main(){ scanf("%d%d",&n,&m); e.set(); for (int i=1;i<n;i++){ int a,b,c; read(a),read(b),read(c); e.add(a,b,c); e.add(b,a,c); } bhtot=0; deep[0]=-1,dis[1]=fadis[1]=0; Build_Tree(0,1); LCA_Prepare(); for (int i=1;i<=m;i++){ read(q[i].x),read(q[i].y); q[i].LCA=LCA(q[i].x,q[i].y); q[i].cost=dis[q[i].x]+dis[q[i].y]-dis[q[i].LCA]*2; } int le=0,ri=Inf,mid,ans=0; while (le<=ri){ mid=(le+ri)>>1; if (check(mid)) ri=mid-1,ans=mid; else le=mid+1; } printf("%d",ans); return 0; }
-
12016-10-25 23:36:07@
#include <iostream>
#include <cstdio>
#include <cmath>
#include <cstring>
#include <algorithm>
#include <vector>
using namespace std;
const int MAXN = 300000 + 10;int read(){
char c = getchar();
int x = 0;
while(!isdigit(c)) c = getchar();
while(isdigit(c)) x = x*10 + c - '0', c = getchar();
return x;
}struct Edge{
int to, w;
Edge* p;
Edge(int a=0, int b=0):to(a), w(b){}
}map[MAXN], bri[MAXN * 3];int cnt = 0;
void add(int a, int b, int c){
Edge& now = bri[++cnt];
now.to = b;
now.w = c;
now.p = map[a].p;
map[a].p = &now;
}int n, m, maxd;
int dep[MAXN], f[MAXN][21], u[MAXN], v[MAXN], lca[MAXN], dist[MAXN], dis[MAXN], we[MAXN], sum[MAXN];int dfs(int x, int fa, int deep){
dep[x] = deep;
for(Edge* i = map[x].p; i != NULL; i = i->p){
if(i->to == fa) continue;
we[i->to] = i->w;
f[i->to][0] = x;
dist[i->to] = dist[x] + i->w;
dfs(i->to, x, deep + 1);
}
}int LCA(int a, int b){
if(dep[a] < dep[b]) swap(a, b);
for(int i = 19; i >= 0; i--)
if(f[a][i] && dep[f[a][i]] >= dep[b])
a = f[a][i];
if(a == b) return a;
for(int i = 19; i >= 0; i--)
if(f[a][i] != f[b][i])
a = f[a][i], b = f[b][i];
return f[a][0];
}int maxe;
void dfs2(int x, int father, int tot){
for(Edge* i = map[x].p; i != NULL; i = i->p){
if(i->to == father) continue;
dfs2(i->to, x, tot);
sum[x] += sum[i->to];
}
if(tot == sum[x]) maxe = max(maxe, we[x]);
}bool check(int mid){
int tot = 0;
maxe = 0;
for(int i = 1; i <= n; i++) sum[i] = 0;
for(int i = 1; i <= m; i++)//查分优化!
if(dis[i] > mid){
sum[u[i]]++;
sum[v[i]]++;
sum[lca[i]] -= 2;
tot++;
}
dfs2(1, -1, tot);
if(maxd - maxe <= mid) return 1;
return 0;
}int main()
{
n = read(), m = read();
for(int i = 1; i <= n; i++) map[i].p = NULL;
int l = 0, r = 0;
for(int i = 1; i < n; i++){
int a = read(), b = read(), c = read();
add(a, b, c);
add(b, a, c);
}
dfs(1, -1, 1);
for(int j = 1; j < 19; j++)
for(int i = 1; i <= n; i++)
f[i][j] = f[f[i][j-1]][j-1];
for(int i = 1; i <= m; i++){
u[i] = read(), v[i] = read();
lca[i] = LCA(u[i], v[i]);
dis[i] = dist[u[i]] + dist[v[i]] - 2*dist[lca[i]];
r = max(dis[i], r);
}
maxd = r;
while(l < r){
int mid = (l + r) >> 1;
if(check(mid))
r = mid;
else
l = mid + 1;
}
printf("%d", l);
return 0;
}
LCA +查分+二分 -
12016-10-08 14:24:41@
二分答案+链剖求lca+树上的差分序列
#include<cstdio>
#include<cstring>
#include<cctype>
#include<algorithm>
using namespace std;
#define rep(i,s,t) for(int i=s;i<=t;i++)
#define dwn(i,s,t) for(int i=s;i>=t;i--)
#define clr(x,c) memset(x,c,sizeof(x))
#define qwq(x) for(edge *o=head[x];o;o=o->next)
#define lson l,mid,x<<1
#define rson mid+1,r,x<<1|1
int read(){
int x=0;char c=getchar();
while(!isdigit(c)) c=getchar();
while(isdigit(c)) x=x*10+c-'0',c=getchar();
return x;
}
const int nmax=3e5+5;
const int inf=0x7f7f7f7f;
struct edge{
int to,dist;edge *next;
};
edge es[nmax<<1],*pt=es,*head[nmax];
struct node{
int u,v,lca,dist;
bool operator<(const node&rhs)const{
return dist>rhs.dist;}
};node ns[nmax];
void add(int u,int v,int d){
pt->to=v;pt->dist=d;pt->next=head[u];head[u]=pt++;
pt->to=u;pt->dist=d;pt->next=head[v];head[v]=pt++;
}
int fa[nmax],dep[nmax],tp[nmax],id[nmax],idx[nmax],son[nmax],sz[nmax],n,m,dist[nmax],num[nmax],w[nmax];
void dfs(int x,int ta){
sz[x]=1;
qwq(x) if(o->to!=ta){
dep[o->to]=dep[x]+1;fa[o->to]=x;w[o->to]=o->dist;dist[o->to]=dist[x]+o->dist;
dfs(o->to,x);sz[x]+=sz[o->to];
if(sz[o->to]>sz[son[x]]||!son[x]) son[x]=o->to;
}
}
void DFS(int x,int top){
tp[x]=top;id[++id[0]]=x;idx[x]=id[0];
if(son[x]) DFS(son[x],top);
qwq(x) if(!idx[o->to]) DFS(o->to,o->to);
}
void query(int a,int b,int cur){
ns[cur].dist=dist[a]+dist[b];
while(tp[a]!=tp[b]){
if(dep[tp[a]]>dep[tp[b]]) swap(a,b);
b=fa[tp[b]];
}
if(dep[a]>dep[b]) swap(a,b);
ns[cur].dist-=dist[a]*2;ns[cur].lca=a;
return ;
}
void get_cnt(int x,int fa){
qwq(x) if(o->to!=fa){
get_cnt(o->to,x);
num[x]+=num[o->to];
}
}
bool check(int x){
int cnt=0;
rep(i,1,m) if(ns[i].dist>x) ++cnt;else break;
if(!cnt) return 1;
clr(num,0);
rep(i,1,cnt) num[ns[i].u]++,num[ns[i].v]++,num[ns[i].lca]-=2;
get_cnt(1,0);
int tmax=0;
rep(i,1,n) if(num[i]==cnt) tmax=max(tmax,w[i]);
return ns[1].dist-tmax<=x;
}
int main(){
n=read(),m=read();int u,v,d,tm=0;
rep(i,1,n-1) u=read(),v=read(),d=read(),add(u,v,d),tm+=d;
dfs(1,0);DFS(1,1);
//rep(i,1,n) printf("%d %d %d %d %d %d %d %d\n",dep[i],fa[i],son[i],sz[i],tp[i],id[i],idx[i],w[i]);
rep(i,1,m) ns[i].u=read(),ns[i].v=read(),query(ns[i].u,ns[i].v,i);
sort(ns+1,ns+m+1);
//rep(i,1,m) printf("%d %d\n",ns[i].lca,ns[i].dist);
int l=1,r=tm,mid,ans=0;
while(l<=r){
mid=(l+r)>>1;
if(check(mid)) ans=mid,r=mid-1;
else l=mid+1;
}
printf("%d\n",ans);
return 0;
}
-
12016-09-05 19:27:27@
//noip 2015 day2 t3
//rt http://www.cnblogs.com/yanlifneg/p/5559491.html
// 二分答案{
倍增求出路径长度判断: 判断有多少条路径>mid t 同时求出最大差值
对于每一个点求不合法的路径求经过它的次数,如果 =t
且如果它的父边>=差值 则 删去 exit(true)判断次数(sum[i]+1, sum[j]+1, sum[lca(i,j)]-2) 查分
然后遍历加和。
O(nlogn)
( 常数很大,因此第20个点是卡着过的,, 在其他oj都tle,, 而且,中间我调试时又一次错误非常明显,结果还是95.。。 )}
uses math; type re=record v,len,next:longint; end; rea=record u,v:longint; len:longint; end; var n,m,u,v,z,tot,i,j,max_deep,l,r,mid,max_cz:longint; last,dis_head,deep,sum,father,f_b:array[0..300000]of longint; lj:array[0..300000]of rea; f:array[0..300000,0..20]of longint; t:array[0..300000*2]of re; procedure add(u,v,z:longint); begin inc(tot); t[tot].v:=v; t[tot].len:=z; t[tot].next:=last[u]; last[u]:=tot; end; procedure In_Dfs(i,fat:longint); var x,tox:longint; begin x:=last[i]; while x<>0 do begin tox:=t[x].v; if tox<>fat then begin deep[tox]:=deep[i]+1; father[tox]:=i; f_b[tox]:=x; dis_head[tox]:=dis_head[i]+t[x].len; max_deep:=max(max_deep,deep[tox]); f[tox][0]:=i; In_Dfs(tox,i); end; x:=t[x].next; end; end; procedure Init_f; var i,j:longint; begin for j:=1 to trunc(ln(max_deep)/ln(2)) do for i:=1 to n do if 1<<j<=deep[i] then f[i][j]:=f[f[i][j-1]][j-1]; end; procedure swap(var x,y:longint); var tmp:longint; begin tmp:=x; x:=y; y:=tmp; end; function lca(x,y:longint):longint; var p,k:longint; begin if deep[x]<deep[y] then swap(x,y); p:=deep[x]-deep[y]; k:=0; while p<>0 do begin if (p and 1=1) then x:=f[x][k]; inc(k); p:=p>>1; end; if x=y then exit(x); k:=0; while k>=0 do begin if f[x][k]<>f[y][k] then begin x:=f[x][k]; y:=f[y][k]; inc(k); end else dec(k); end; exit(f[x][0]); end; procedure Init_lj; var i:longint; begin for i:=1 to m do begin readln(lj[i].u,lj[i].v); lj[i].len:=(dis_head[lj[i].u]+dis_head[lj[i].v]-2*dis_head[lca(lj[i].u,lj[i].v)]); r:=max(r,lj[i].len); end; end; procedure Init; var i,j:longint; begin readln(n,m); for i:=1 to n-1 do begin readln(u,v,z); add(u,v,z); add(v,u,z); end; In_Dfs(1,0); Init_f; Init_lj; end; procedure Dfs_sum(i,fat:longint); var x,tox:longint; begin x:=last[i]; while x<>0 do begin tox:=t[x].v; if tox<>fat then begin Dfs_sum(tox,i); sum[i]:=sum[i]+sum[tox]; end; x:=t[x].next; end; end; function check(mid:longint):boolean; var i,j,max_cz,tot,x,zx:longint; begin tot:=0; max_cz:=0; for i:=1 to n do sum[i]:=0; for i:=1 to m do if lj[i].len>mid then begin inc(tot); max_cz:=max(max_cz,lj[i].len-mid); inc(sum[lj[i].u]); inc(sum[lj[i].v]); zx:=lca(lj[i].u,lj[i].v); dec(sum[zx],2) end; Dfs_sum(1,0); for i:=1 to n do if (sum[i]=tot)and(t[f_b[i]].len>=max_cz) then exit(true); exit(false); end; begin Init; while l<>r do begin mid:=(l+r)>>1; if check(mid) then r:=mid else l:=mid+1; end; writeln(l); end.```
-
12016-08-30 15:55:10@
同下,大牛程序看不懂系列
```c++
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <map>
#include <stack>
#include <queue>
#include <algorithm>
#include <cmath>
#include <fstream>
#include <time.h>
#include <cctype>
#include <vector>
#include <set>
#define pb push_back
#define N 700100using namespace std;
int dp,pre[N],p[N],tt[N],ww[N],fa[N],deep[N],v[N],A[N],B[N],LCA[N];
int s[N][20],n,m,a,b,c,i,sum[N],ans,cnt,dis[N],dist[N];
void link(int x,int y,int z)
{
dp++;
pre[dp]=p[x];
p[x]=dp;
tt[dp]=y;
ww[dp]=z;
}void dfs(int x)
{
int i;
i=p[x];
while (i)
{
if (tt[i]!=fa[x])
{
deep[tt[i]]=deep[x]+1;
fa[tt[i]]=x;
v[tt[i]]=ww[i];
dis[tt[i]]=dis[x]+ww[i];
dfs(tt[i]);
}
i=pre[i];
}
}int lca(int x,int y)
{
if(deep[x]>deep[y])x^=y^=x^=y;
int i;
for(i=19;i>=0;i--)
{
if(deep[y]-deep[x]>=(1<<i))
{
y=s[y][i];
}
}
if(x==y)return x;
for(i=19;i>=0;i--)
{
if(s[x][i]!=s[y][i])
{
x=s[x][i];
y=s[y][i];
}
}
return fa[x];
}void gao(int x)
{
int i=p[x];
while (i)
{
if (tt[i]!=fa[x])
{
gao(tt[i]);
sum[x]+=sum[tt[i]];
}
i=pre[i];
}
}int check(int x)
{
int cnt=0,dec=0;
for (i=1;i<=n;i++)
sum[i]=0;
for (i=1;i<=m;i++)
if (dist[i]>x)
{
cnt++;
dec=max(dec,dist[i]-x);
sum[A[i]]++;
sum[B[i]]++;
sum[LCA[i]]-=2;
}
gao(1);
for (i=1;i<=n;i++)
if ((sum[i]==cnt)&&(v[i]>=dec)) return 1;
return 0;
}int main()
{
scanf("%d%d",&n,&m);
for (i=1;i<n;i++)
{
scanf("%d%d%d",&a,&b,&c);
link(a,b,c);
link(b,a,c);
}
dfs(1);
for(i=1;i<=n;i++)s[i][0]=fa[i];
for(int h=1;h<20;h++)
{
for(i=1;i<=n;i++)
{
s[i][h]=s[s[i][h-1]][h-1];
}
}
for (i=1;i<=m;i++)
{
scanf("%d%d",&A[i],&B[i]);
LCA[i]=lca(A[i],B[i]);
dist[i]=dis[A[i]]+dis[B[i]]-2*dis[LCA[i]];
}
int L=0,R=0;
for (i=1;i<=m;i++)
R=max(R,dist[i]);
int mid;
while (L<=R)
{
mid=(L+R)>>1;
if (check(mid))
R=mid-1;
else
L=mid+1;
}
printf("%d\n",L);
return 0;
}
``` -
12016-08-14 14:45:34@
评测结果
编译成功测试数据 #0: Accepted, time = 15 ms, mem = 80400 KiB, score = 5
测试数据 #1: Accepted, time = 0 ms, mem = 80400 KiB, score = 5
测试数据 #2: Accepted, time = 0 ms, mem = 80400 KiB, score = 5
测试数据 #3: Accepted, time = 0 ms, mem = 80396 KiB, score = 5
测试数据 #4: Accepted, time = 0 ms, mem = 80420 KiB, score = 5
测试数据 #5: Accepted, time = 15 ms, mem = 80448 KiB, score = 5
测试数据 #6: Accepted, time = 15 ms, mem = 80484 KiB, score = 5
测试数据 #7: Accepted, time = 0 ms, mem = 80396 KiB, score = 5
测试数据 #8: Accepted, time = 0 ms, mem = 80400 KiB, score = 5
测试数据 #9: Accepted, time = 15 ms, mem = 80396 KiB, score = 5
测试数据 #10: Accepted, time = 109 ms, mem = 80552 KiB, score = 5
测试数据 #11: Accepted, time = 171 ms, mem = 80592 KiB, score = 5
测试数据 #12: RuntimeError, time = 78 ms, mem = 82420 KiB, score = 0
测试数据 #13: RuntimeError, time = 93 ms, mem = 82416 KiB, score = 0
测试数据 #14: RuntimeError, time = 78 ms, mem = 82416 KiB, score = 0
测试数据 #15: RuntimeError, time = 109 ms, mem = 82416 KiB, score = 0
测试数据 #16: Accepted, time = 265 ms, mem = 80548 KiB, score = 5
测试数据 #17: Accepted, time = 265 ms, mem = 80572 KiB, score = 5
测试数据 #18: Accepted, time = 312 ms, mem = 80584 KiB, score = 5
测试数据 #19: TimeLimitExceeded, time = 1125 ms, mem = 80988 KiB, score = 0
TimeLimitExceeded, time = 2665 ms, mem = 82420 KiB, score = 75
#include <iostream>
#include <cstring>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <algorithm>
#include <queue>
#include <stack>
#define LL long long
#define inf 1000000000
using namespace std;
const int N=300000+5;
const int M=300000+5;
struct edge
{
int v,w;
edge *next;
}E[M*2],*p=E,*point[N*2];
struct node
{
int a1,b1,anc,dis;
}lca[N*2];
inline void add(int x,int y,int z)
{
p++; p->v=y; p->w=z; p->next=point[x];
point[x]=p;
}
int anc[N][31],fa[N],dep[N],dis[N],sum[N];
void dfs(int x)
{
anc[x][0]=fa[x];
for (int i=1; i<=30; i++)
anc[x][i]=anc[anc[x][i-1]][i-1];
for (edge *j=point[x]; j; j=j->next)
if (j->v!=fa[x])
{
fa[j->v]=x;
dep[j->v]=dep[x]+1;
dis[j->v]=dis[x]+j->w;
dfs(j->v);
}
}
int LCA(int x,int y)
{
if (dep[x]<dep[y]) swap(x,y);
for (int i=30; i>=0; i--)
if (dep[y]<=dep[anc[x][i]]) x=anc[x][i];
if (x==y) return x;
for (int i=30; i>=0; i--)
if (anc[x][i]!=anc[y][i])
{
x=anc[x][i];
y=anc[y][i];
}
return anc[x][0];
}
void get_sum(int x)
{
for (edge *j=point[x]; j; j=j->next)
if (j->v!=fa[x])
{
get_sum(j->v);
sum[x]+=sum[j->v];
}
}
int n,m,l=0,r=0,mid;
bool check(int limit)
{
int tot=0, p=0;
memset(sum,0,sizeof(sum));
for (int i=1; i<=m; i++)
if (lca[i].dis>limit)
{
tot++;
sum[lca[i].a1]++;
sum[lca[i].b1]++;
sum[lca[i].anc]-=2;
p=max(p,lca[i].dis-limit);
}
get_sum(1);
for (int i=1; i<=n; i++)
if (sum[i]==tot)
for (edge *j=point[i]; j; j=j->next)
if (j->v==fa[i]&&j->w>=p) return 1;
return 0;
}
void solve()
{
scanf("%d%d",&n,&m);
for (int i=1; i<n; i++)
{
int x,y,z;
scanf("%d%d%d",&x,&y,&z);
add(x,y,z); add(y,x,z);
}
dfs(1);
for (int i=1; i<=m; i++)
{
scanf("%d%d",&lca[i].a1,&lca[i].b1);
lca[i].anc=LCA(lca[i].a1,lca[i].b1);
lca[i].dis=dis[lca[i].a1]+dis[lca[i].b1]-2*dis[lca[i].anc];
r=max(r,lca[i].dis);
}
while (l<=r)
{
mid=(l+r)>>1;
if (check(mid)) r=mid-1;
else l=mid+1;
}
printf("%d\n",r+1);
}
int main()
{
solve();
return 0;
}
求指教。。。。。。。。。为什么会运行时错误,运行时错误是smg。。。。。。 -
12015-11-14 11:59:24@
###大牛的程序我看不懂!!!
记录信息
评测状态 Accepted
题目 P1983 运输计划
递交时间 2015-11-14 11:19:34
代码语言 C++
评测机 VijosEx
消耗时间 1605 ms
消耗内存 44876 KiB
评测时间 2015-11-14 11:19:39
评测结果
编译成功测试数据 #0: Accepted, time = 0 ms, mem = 44872 KiB, score = 5
测试数据 #1: Accepted, time = 0 ms, mem = 44872 KiB, score = 5
测试数据 #2: Accepted, time = 0 ms, mem = 44868 KiB, score = 5
测试数据 #3: Accepted, time = 0 ms, mem = 44876 KiB, score = 5
测试数据 #4: Accepted, time = 0 ms, mem = 44872 KiB, score = 5
测试数据 #5: Accepted, time = 0 ms, mem = 44868 KiB, score = 5
测试数据 #6: Accepted, time = 0 ms, mem = 44872 KiB, score = 5
测试数据 #7: Accepted, time = 0 ms, mem = 44872 KiB, score = 5
测试数据 #8: Accepted, time = 15 ms, mem = 44868 KiB, score = 5
测试数据 #9: Accepted, time = 15 ms, mem = 44872 KiB, score = 5
测试数据 #10: Accepted, time = 46 ms, mem = 44872 KiB, score = 5
测试数据 #11: Accepted, time = 78 ms, mem = 44872 KiB, score = 5
测试数据 #12: Accepted, time = 109 ms, mem = 44876 KiB, score = 5
测试数据 #13: Accepted, time = 125 ms, mem = 44872 KiB, score = 5
测试数据 #14: Accepted, time = 125 ms, mem = 44868 KiB, score = 5
测试数据 #15: Accepted, time = 140 ms, mem = 44872 KiB, score = 5
测试数据 #16: Accepted, time = 109 ms, mem = 44872 KiB, score = 5
测试数据 #17: Accepted, time = 125 ms, mem = 44872 KiB, score = 5
测试数据 #18: Accepted, time = 156 ms, mem = 44872 KiB, score = 5
测试数据 #19: Accepted, time = 562 ms, mem = 44872 KiB, score = 5
Accepted, time = 1605 ms, mem = 44876 KiB, score = 100
代码
#include<algorithm>
#include <iostream>
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <math.h>
#include <time.h>
#include <vector>
#include <queue>
#include <map>
#include <set>
using namespace std;const int N=300005;
#define next _Next
struct Route
{
int u,v,l;
}R[N];inline bool operator<(const Route& A,const Route& B)
{
return A.l>B.l;
}int n,m,e[N<<1],L[N<<1],next[N<<1],G[N],tot,dep[N],fa[N][20],dst[N],Rou[N],Rtot,RouL[N],Mark[N],Max[N];
void adde(int u,int v,int l)
{
e[++tot]=v;next[tot]=G[u];G[u]=tot;L[tot]=l;
e[++tot]=u;next[tot]=G[v];G[v]=tot;L[tot]=l;
}void bfs()
{
static int q[N]={},a=1,b=1;
q[1]=1;dep[1]=1;
while(a<=b)
{
int u=q[a++];
for(int i=0;fa[fa[u][i]][i];i++)
fa[u][i+1]=fa[fa[u][i]][i];
for(int i=G[u];i;i=next[i])
if(!dep[e[i]])
dep[e[i]]=dep[u]+1,dst[e[i]]=dst[u]+L[i],fa[e[i]][0]=u,q[++b]=e[i];
}
}inline int Lca(int u,int v)
{
if(dep[u]<dep[v])
swap(u,v);
for(int i=19;~i;i--)
if(dep[fa[u][i]]>=dep[v])
u=fa[u][i];
if(u==v)
return u;
for(int i=19;~i;i--)
if(fa[u][i]!=fa[v][i])
u=fa[u][i],v=fa[v][i];
return fa[u][0];
}inline int dis(int u,int v)
{
return dst[u]+dst[v]-2*dst[Lca(u,v)];
}void get_rou(int u,int v)
{
int lca=Lca(u,v);
Rtot=dep[u]+dep[v]-2*dep[lca]+1;
int Now=0;
while(u!=lca)
{
Rou[++Now]=u;RouL[Now]=dst[u]-dst[fa[u][0]];u=fa[u][0];
}
Now=Rtot;
Rou[Now]=v;
while(v!=lca)
{
u=v;v=fa[u][0];Rou[--Now]=v;RouL[Now]=dst[u]-dst[v];
}
}void mark_all()
{
int q[N]={},a=1,b=0;
for(int i=1;i<=Rtot;i++)
Mark[Rou[i]]=i,q[++b]=Rou[i];
while(a<=b)
{
int u=q[a++];
for(int i=G[u];i;i=next[i])
if(!Mark[e[i]])
Mark[e[i]]=Mark[u],q[++b]=e[i];
}
}int main()
{
scanf("%d%d",&n,&m);
for(int i=1,u,v,l;i<n;i++)
scanf("%d%d%d",&u,&v,&l),adde(u,v,l);
bfs();
for(int i=1;i<=m;i++)
scanf("%d%d",&R[i].u,&R[i].v),R[i].l=dis(R[i].u,R[i].v);
sort(R+1,R+m+1);
if(R[1].l==0)
{
cout<<0<<endl;return 0;
}
get_rou(R[1].u,R[1].v);
mark_all();
int Lnow=1,Rnow=Rtot;
for(int i=1;i<=m;i++)
{
int Le=Mark[R[i].u],Ri=Mark[R[i].v];
if(Le>Ri)
swap(Le,Ri);
if(Lnow<Le)
{
for(int j=Lnow;j<Rnow&&j<Le;j++)
Max[j]=R[i].l;
Lnow=Le;
}
if(Ri<Rnow)
{
for(int j=max(Lnow,Ri);j<Rnow;j++)
Max[j]=R[i].l;
Rnow=Ri;
}
if(Lnow>=Rnow)
break;
}
int Ans=R[1].l;
for(int i=1;i<Rtot;i++)
Ans=min(Ans,max(R[1].l-RouL[i],Max[i]));
cout<<Ans<<endl;
return 0;
}
###我的60‘
记录信息
评测状态 Time Limit Exceeded
题目 P1983 运输计划
递交时间 2015-11-14 11:36:33
代码语言 C++
评测机 VijosEx
消耗时间 4476 ms
消耗内存 23424 KiB
评测时间 2015-11-14 11:36:39
评测结果
编译成功foo.cpp: In function 'int main()':
foo.cpp:76:24: warning: unknown conversion type character 'l' in format [-Wformat=]
printf("%lld\n",ans);
^
foo.cpp:76:24: warning: too many arguments for format [-Wformat-extra-args]
foo.cpp:61:13: warning: 'now' may be used uninitialized in this function [-Wmaybe-uninitialized]
int pre,now;
^
测试数据 #0: Accepted, time = 0 ms, mem = 21408 KiB, score = 5
测试数据 #1: Accepted, time = 0 ms, mem = 21408 KiB, score = 5
测试数据 #2: Accepted, time = 15 ms, mem = 21408 KiB, score = 5
测试数据 #3: Accepted, time = 0 ms, mem = 21412 KiB, score = 5
测试数据 #4: Accepted, time = 0 ms, mem = 21444 KiB, score = 5
测试数据 #5: Accepted, time = 31 ms, mem = 21492 KiB, score = 5
测试数据 #6: Accepted, time = 78 ms, mem = 21544 KiB, score = 5
测试数据 #7: Accepted, time = 0 ms, mem = 21412 KiB, score = 5
测试数据 #8: Accepted, time = 0 ms, mem = 21412 KiB, score = 5
测试数据 #9: Accepted, time = 15 ms, mem = 21412 KiB, score = 5
测试数据 #10: Accepted, time = 46 ms, mem = 21644 KiB, score = 5
测试数据 #11: Accepted, time = 62 ms, mem = 21700 KiB, score = 5
测试数据 #12: RuntimeError, time = 46 ms, mem = 23420 KiB, score = 0
测试数据 #13: RuntimeError, time = 46 ms, mem = 23424 KiB, score = 0
测试数据 #14: RuntimeError, time = 31 ms, mem = 23424 KiB, score = 0
测试数据 #15: RuntimeError, time = 46 ms, mem = 23420 KiB, score = 0
测试数据 #16: TimeLimitExceeded, time = 1015 ms, mem = 21628 KiB, score = 0
测试数据 #17: TimeLimitExceeded, time = 1015 ms, mem = 21652 KiB, score = 0
测试数据 #18: TimeLimitExceeded, time = 1015 ms, mem = 21692 KiB, score = 0
测试数据 #19: TimeLimitExceeded, time = 1015 ms, mem = 22292 KiB, score = 0
TimeLimitExceeded, time = 4476 ms, mem = 23424 KiB, score = 60
代码
#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;
int nxt[600123],fir[300123],fa[300123][2],val[600123],t[600123],dep[300123],cnt;
long long to[600123];
int n,m;
long long mem[2][300123];
long long ans;
void sw(int &a,int &b)
{
a=a^b;
b=a^b;
a=a^b;
}
void add(int a,int b,int v)
{
nxt[++cnt]=fir[a];
val[cnt]=v;
to[cnt]=b;
fir[a]=cnt;
}
void search(int node,int pre)
{ dep[node]=dep[pre]+1;
for(int i=fir[node];i;i=nxt[i])
{ if(to[i]==pre)continue;
fa[to[i]][0]=node;
fa[to[i]][1]=i;
search(to[i],node);
}
}
long long DO_IT(int x,int y,int k)
{
long long tot=0;
if(dep[x]<dep[y])sw(x,y);
while(dep[x]>dep[y]){
t[fa[x][1]]=k;
tot+=val[fa[x][1]];
x=fa[x][0];
}
while(x!=y){
t[fa[x][1]]=k;
tot+=val[fa[x][1]];
t[fa[y][1]]=k;
tot+=val[fa[y][1]];
x=fa[x][0];
y=fa[y][0];
}
return tot;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<n;i++){
int a,b,c;
scanf("%d%d%d",&a,&b,&c);
add(a,b,c);
add(b,a,c);
}
search(1,1);
int pre,now;
for(int i=1;m--;i++)
{ pre=(i+1)%2;
now=i%2;
int a,b;
scanf("%d%d",&a,&b);
long long tot=DO_IT(a,b,i+2*m+1);
for(int j=1;j<=cnt;j++)
{ int ss=(t[j]==i+2*m+1)?val[j]:0;
mem[now][j]=max(mem[pre][j],tot-ss);
}
}
ans=2333333333;
for(int i=1;i<=cnt;i++)
ans=min(ans,mem[now][i]);
printf("%lld\n",ans);
return 0;
} -
12015-11-10 21:48:36@
评测结果
编译成功
测试数据 #0: Accepted, time = 0 ms, mem = 6536 KiB, score = 5
测试数据 #1: Accepted, time = 0 ms, mem = 6532 KiB, score = 5
测试数据 #2: WrongAnswer, time = 0 ms, mem = 6532 KiB, score = 0
测试数据 #3: Accepted, time = 15 ms, mem = 6536 KiB, score = 5
测试数据 #4: Accepted, time = 15 ms, mem = 6588 KiB, score = 5
测试数据 #5: Accepted, time = 46 ms, mem = 6644 KiB, score = 5
测试数据 #6: Accepted, time = 78 ms, mem = 6708 KiB, score = 5
测试数据 #7: Accepted, time = 15 ms, mem = 6532 KiB, score = 5
测试数据 #8: Accepted, time = 0 ms, mem = 6536 KiB, score = 5
测试数据 #9: Accepted, time = 15 ms, mem = 6536 KiB, score = 5
测试数据 #10: Accepted, time = 671 ms, mem = 6760 KiB, score = 5
测试数据 #11: Accepted, time = 1000 ms, mem = 6808 KiB, score = 5
测试数据 #12: TimeLimitExceeded, time = 1015 ms, mem = 6532 KiB, score = 0
测试数据 #13: TimeLimitExceeded, time = 1015 ms, mem = 6532 KiB, score = 0
测试数据 #14: TimeLimitExceeded, time = 1015 ms, mem = 6536 KiB, score = 0
测试数据 #15: TimeLimitExceeded, time = 1015 ms, mem = 6536 KiB, score = 0
测试数据 #16: TimeLimitExceeded, time = 1015 ms, mem = 6828 KiB, score = 0
测试数据 #17: TimeLimitExceeded, time = 1015 ms, mem = 6864 KiB, score = 0
测试数据 #18: TimeLimitExceeded, time = 1015 ms, mem = 6532 KiB, score = 0
测试数据 #19: RuntimeError, time = 46 ms, mem = 6536 KiB, score = 0
TimeLimitExceeded, time = 9006 ms, mem = 6864 KiB, score = 55
首先声明,这个程序是
乱搞的!
乱搞的!
乱搞的!
考场上正解写不出来。。。毕竟蒟蒻一个,连LCA都不会。。然而居然骗这么多分,我感觉到爽翻了。。。
做法:贪心:对于每个计划DFS一遍,找到计划中最大边删掉,如果此时的距离最大,更新MAXDIS,最后输出MAXDIS顺便说一下,样例2都没过。。。
###Block Code
#include<iostream>
#include<cstdio>
#include<cstring>
#include<string>
#include<cmath>
using namespace std;
int u[200001],v[200001],w[200001],first[200001],cnt[200001],pu[200001],pv[200001],t;
int dis[200001],maxdis=0,maxdisn=0;
void quicksort(int left,int right) {
if (left>=right) return;
int i=left,j=right,t=0;
while (i<j) {
while (i<j && u[left]<=u[j]) j--;
while (i<j && u[i]<=u[left]) i++;
if (i<j) {
t=w[i];
w[i]=w[j];
w[j]=t;
t=u[i];
u[i]=u[j];
u[j]=t;
t=v[i];
v[i]=v[j];
v[j]=t;
}
}
t=w[left];
w[left]=w[i];
w[i]=t;
t=u[left];
u[left]=u[i];
u[i]=t;
t=v[left];
v[left]=v[i];
v[i]=t;
quicksort(left,i-1);
quicksort(i+1,right);
return;
}
int dfs(int x,int mubiao,int maxnow) {
int target=first[x]+cnt[x];
int nowcost=dis[x];
for (int i=first[x];i<target;i++) {
if (dis[v[i]]==-1) {
dis[v[i]]=nowcost+w[i];
if (v[i]==mubiao) {
return max(maxnow,w[i]);
}
int re=dfs(v[i],mubiao,max(maxnow,w[i]));
if (re!=0) return re;
}
}
return 0;
}
int main() {
int n,m;
// freopen("transport.in","r",stdin);
// freopen("transport.out","w",stdout);
scanf("%d%d",&n,&m);
for (int i=1;i<n;i++) {
int j=i<<1;
int a,b,r;
scanf("%d%d%d",&a,&b,&r);
u[j-1]=a;
v[j-1]=b;
w[j-1]=r;
v[j]=a;
u[j]=b;
w[j]=r;
cnt[a]++;
cnt[b]++;
}
t=(n-1)<<1;
quicksort(1,t);
int now=-1;
for (int i=1;i<=t;i++) {
if (u[i]!=now) {
now=u[i];
first[now]=i;
}
}
// printf("read & sort complete!");
if (m*n>1000000) m=10000;
for (int i=1;i<=m;i++) {
scanf("%d%d",&pu[i],&pv[i]);
for (int j=1;j<=n;j++)
dis[j]=-1;
dis[pu[i]]=0;
int maxnow=dfs(pu[i],pv[i],0);
// printf("%d->%d:%d",pu[i],pv[i],dis[pv[i]]);
if (maxdis<dis[pv[i]]-maxnow) {
maxdis=dis[pv[i]]-maxnow;
}
}
printf("%d",maxdis);
return 0;
} -
02017-04-06 11:42:42@
其实2个dfs就完了的说
。。。。#include <cstdio> #include <cstdlib> #include <algorithm> #include <vector> #include <cstring> #include <queue> using namespace std; //AC const int maxn=2*300000+50; const int maxm=2*300000+50; const int max_log_n=20; inline int read() { char c;int x=1,res=0; c=getchar(); while(!isdigit(c)) { if(c=='-') x=-1; c=getchar(); } while(isdigit(c)) { res=res*10+c-'0'; c=getchar(); } return res*x; } int rnk[maxn];/////////////// int sz=0; int head[maxn]; struct edge { int to,cost,next; }; edge es[maxn]; int cnt=0; void addedge(int from,int to,int cost) { es[++cnt].to=to;es[cnt].next=head[from];es[cnt].cost=cost;head[from]=cnt; } int pa[max_log_n][maxn]; int n,m; int depth[maxn]; int dis[maxn]; struct plan{int id,from,to,cost,anse;}; plan ps[maxm]; int maxcost=-1; vector<int > tar; int add[maxn]; int sum[maxn];//子树标记之和 int prevc[maxn]; void dfs(int v,int fa) { depth[v]=depth[fa]+1; pa[0][v]=fa; int k=1; if(pa[k-1][v]!=0) { while(pa[k-1][pa[k-1][v]]!=0) { pa[k][v]=pa[k-1][pa[k-1][v]]; k++; } } for(int i=head[v];i>0;i=es[i].next) { edge &e=es[i]; if(e.to!=fa) { dis[e.to]=dis[v]+e.cost; prevc[e.to]=e.cost; dfs(e.to,v); } } rnk[++sz]=v; } int lca(int u,int v) { if(depth[u]>depth[v]) { swap(u,v); } for(int k=max_log_n-1;k>=0;k--) { if(depth[v]-depth[u]>>k&1) { v=pa[k][v]; } } if(v==u) return u; else { for(int k=max_log_n-1;k>=0;k--) { if(pa[k][u]!=pa[k][v]) { u=pa[k][u];v=pa[k][v]; } } return pa[0][u]; } } void dfs2(int v,int fa) { sum[v]=add[v]; for(int j=head[v];j>0;j=es[j].next) { edge &e=es[j]; if(e.to!=fa) { dfs2(e.to,v); sum[v]+=sum[e.to]; } } } bool cal(int x,bool debug=false) { /*/ if(x==70) { debug=true; } /*/ tar.clear(); for(int i=1;i<=m;i++) { if(ps[i].cost>x) { tar.push_back(i); } } int maxx=-1; memset(add,0,sizeof(add)); memset(sum,0,sizeof(sum)); for(unsigned j=0;j<tar.size();j++) { int num=tar[j]; int v=ps[num].from;int u=ps[num].to; add[v]+=1; add[u]+=1; add[ps[num].anse]-=2; maxx=max(ps[num].cost-x,maxx); } dfs2(1,0); /* for(int i=1,u;i<=n;i++) { u=rnk[i]; add[pa[0][u]]+=add[u]; } */ if(debug) { printf("sum:\n"); for(int i=1;i<=n;i++) { printf("sum[%d]=%d\n",i,sum[i]); } printf("\n"); for(int i=1;i<=n;i++) { printf("sum[%d]=%d\n",i,sum[i]); } printf("maxdel=%d\n",maxx); printf("\n"); printf("total=%d\n",tar.size()); printf("prevc:\n"); for(int i=1;i<=n;i++) { printf("%d\n",prevc[i]); } printf("\n"); } int dec=-1; for(int i=1;i<=n;i++) { if(sum[i]==(int)tar.size()) { //i为要删除的边的下点 if(prevc[i]>=maxx) { dec=max(dec,prevc[i]); } } } return dec>=maxx; } inline int solve() { /* for(int i=1;i<=sz;i++) { printf("rnk[%d]=%d\n",i,rnk[i]); } printf("\n"); */ for(int i=1;i<=m;i++) { int u=ps[i].from; int v=ps[i].to; if(depth[u]>depth[v]) swap(u,v); ps[i].anse=lca(u,v);int zu=ps[i].anse; ps[i].cost=dis[u]+dis[v]-2*dis[zu]; maxcost=max(maxcost,ps[i].cost); } int l=-1;int r=3*100000000; while(l<r-1) { int mid=l+(r-l)/2; //printf("%d\n",mid); if(cal(mid)) { r=mid; } else l=mid; } return r; } int main() { /* #ifdef ONLINE_JUDGE freopen("1.out","w",stdout); #endif */ //freopen("WA.out","w",stdout); n=read();m=read(); for(int i=0;i<n-1;i++) { int from,to,cost; from=read();to=read();cost=read(); addedge(from,to,cost); addedge(to,from,cost); } dfs(1,0); for(int i=1;i<=m;i++) { ps[i].from=read();ps[i].to=read(); } printf("%d\n",solve()); return 0; }
-
02016-10-21 12:24:58@
看到题解中一堆大神写树剖萌新瑟瑟发抖...然而学完了树剖发现也可以用Tarjan求LCA...→_→不过树剖边权变点权的思想还是有用的。
LCA+二分答案+树上差分序列检验
差分序列的具体实现:
遍历路径长度大于二分中值的(不合法的)询问,记其总数为tot,对其中每个节点u指向节点v的询问:cnt[u]++; cnt[v]++; cnt[LCA(u,v)]-=2;
然后DFS,对于节点u,加深,记rtn[u]=cnt[u]+所有子节点的rtn值之和,这样rtn[u]即表示经过节点u(所对应的原树中的边)的(不合法的)路径的条数,若==tot则表示所有不合法的边都经过这一路径。
在此基础上,如果改造这条边为虫洞能使最长的路径<=二分中值(此时所有其他路径也<=二分中值),则可行。我使用的是Tarjan离线算法预先求出所有询问的原路径长度和LCA。
特别提示C++党不要用vector建图,否则只能过一半的点。
测试数据 #0: Accepted, time = 0 ms, mem = 34608 KiB, score = 5 测试数据 #1: Accepted, time = 15 ms, mem = 34608 KiB, score = 5 测试数据 #2: Accepted, time = 0 ms, mem = 34608 KiB, score = 5 测试数据 #3: Accepted, time = 15 ms, mem = 34608 KiB, score = 5 测试数据 #4: Accepted, time = 15 ms, mem = 34648 KiB, score = 5 测试数据 #5: Accepted, time = 15 ms, mem = 34692 KiB, score = 5 测试数据 #6: Accepted, time = 0 ms, mem = 34744 KiB, score = 5 测试数据 #7: Accepted, time = 0 ms, mem = 34612 KiB, score = 5 测试数据 #8: Accepted, time = 15 ms, mem = 34612 KiB, score = 5 测试数据 #9: Accepted, time = 15 ms, mem = 34604 KiB, score = 5 测试数据 #10: Accepted, time = 93 ms, mem = 34840 KiB, score = 5 测试数据 #11: Accepted, time = 109 ms, mem = 34904 KiB, score = 5 测试数据 #12: Accepted, time = 140 ms, mem = 37884 KiB, score = 5 测试数据 #13: Accepted, time = 156 ms, mem = 38356 KiB, score = 5 测试数据 #14: Accepted, time = 218 ms, mem = 38824 KiB, score = 5 测试数据 #15: Accepted, time = 218 ms, mem = 39296 KiB, score = 5 测试数据 #16: Accepted, time = 156 ms, mem = 34844 KiB, score = 5 测试数据 #17: Accepted, time = 203 ms, mem = 34868 KiB, score = 5 测试数据 #18: Accepted, time = 203 ms, mem = 34900 KiB, score = 5 测试数据 #19: Accepted, time = 687 ms, mem = 35508 KiB, score = 5 Accepted, time = 2273 ms, mem = 39296 KiB, score = 100 #include<iostream> #include<cstring> #include<vector> #include<cstdio> #include<algorithm> #define MAXN 300010 using namespace std; int N,M; struct edge { int u,v; int w; } E[MAXN<<1]; int EC=0,Ef[MAXN<<1],En[MAXN<<1]; void inline ADD_EDGE(int u,int v,int w) { E[++EC].u=u; E[EC].v=v; E[EC].w=w; En[EC]=Ef[u]; Ef[u]=EC; } struct query { int u,v; int LCA; int l; bool operator > (const query& q) const { return l>q.l; } } Q[MAXN<<1]; int Qf[MAXN<<1],Qn[MAXN<<1]; void inline ADD_QUERY(int u,int v,int i) { Q[i].u=u; Q[i].v=v; Q[i].LCA=Q[i].l=0; Qn[(i<<1)]=Qf[u]; Qf[u]=(i<<1); Qn[(i<<1)|1]=Qf[v]; Qf[v]=(i<<1)|1; } int dist[MAXN],P[MAXN],visited[MAXN],nw[MAXN],fa[MAXN]; int FIND_SET(int u) { if(P[u]==u) return u; else return P[u]=FIND_SET(P[u]); } void DFS(int u) { P[u]=u; for(int i=Ef[u];i;i=En[i]) { edge& e=E[i]; if(e.v!=fa[u]) { dist[e.v]=dist[u]+e.w; nw[e.v]=e.w; fa[e.v]=u; DFS(e.v); P[e.v]=u; } } visited[u]=true; for(int i=Qf[u];i;i=Qn[i]) { query& q=Q[i>>1]; if(!q.LCA) { if(q.v==u) swap(q.u,q.v); if(visited[q.v]) { q.LCA=FIND_SET(q.v); q.l=dist[u]+dist[q.v]-2*dist[FIND_SET(q.v)]; } } } } int inline Q_FIND(int minl) { int l=1,u=M; while(l<u) { int m=(l+u+1)>>1; if(Q[m].l>minl) l=m; else u=m-1; } return l; } int tot,cnt[MAXN],rtn[MAXN],min_nw; bool CHECK_DFS(int u) { rtn[u]=cnt[u]; for(int i=Ef[u];i;i=En[i]) { edge& e=E[i]; if(e.v!=fa[u]) { //fa[]在预处理DFS时已建立完成 if(CHECK_DFS(e.v)) return true; rtn[u]+=rtn[e.v]; } } return (rtn[u]>=tot&&nw[u]>=min_nw); } bool inline CHECK(int t) { memset(cnt,0,sizeof(cnt)); tot=Q_FIND(t); for(int i=1;i<=tot;i++) { cnt[Q[i].u]++; cnt[Q[i].v]++; cnt[Q[i].LCA]-=2; } min_nw=Q[1].l-t; return CHECK_DFS(1); } int MAX_W=0; int main() { memset(Ef,0,sizeof(Ef)); memset(Qf,0,sizeof(Qf)); scanf("%d%d",&N,&M); int u,v,w,l,m; for(int i=1;i<N;i++) { scanf("%d%d%d",&u,&v,&w); MAX_W=max(MAX_W,w); ADD_EDGE(u,v,w); ADD_EDGE(v,u,w); } for(int i=1;i<=M;i++) { scanf("%d%d",&u,&v); ADD_QUERY(u,v,i); } memset(visited,false,sizeof(visited)); dist[1]=0; nw[1]=0; fa[1]=1; DFS(1); sort(Q+1,Q+1+M,greater<query>()); l=Q[1].l-MAX_W; u=Q[1].l-1; while(l<u) { m=(l+u)>>1; if(CHECK(m)) u=m; else l=m+1; } printf("%d\n",l); return 0; }
-
02016-09-13 17:33:46@
T了一个点……我理解为写dfs在vijos上爆栈了吧,然而不想改了……bzoj AC。
二分答案 + 树链剖分 + 差分标记验证。#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <ctime>
#include <cstdlib>
#include <algorithm>
#include <queue>
#include <string>
using namespace std;#define MAXN 300005
int ch=0;
inline void read(int &a) {
while (ch<'0' || ch>'9') ch=getchar();
a=0;
while (ch>='0' && ch<='9') {a=(a<<3)+(a<<1)+ch-'0';ch=getchar();}
}struct edge{
int next,to,w;
}e[MAXN<<1];int n,m,maxl=0,ecnt=0,head[MAXN],rs[MAXN],rt[MAXN],rd[MAXN],len[MAXN];
int ss[MAXN],maxs[MAXN],son[MAXN],fa[MAXN],dep[MAXN],dis[MAXN];
int top[MAXN],tid[MAXN],dfn=0;int t[MAXN],rlen[MAXN];
inline void add(int u,int v,int w) {
e[++ecnt].to=v;
e[ecnt].next=head[u];
e[ecnt].w=w;
head[u]=ecnt;
e[++ecnt].to=u;
e[ecnt].next=head[v];
e[ecnt].w=w;
head[v]=ecnt;
}inline void input() {
read(n);read(m);
for (int i=1,u,v,w;i<n;i++) {
read(u);read(v);read(w);
add(u,v,w);
}
for (int i=1;i<=m;i++) {
read(rs[i]);read(rt[i]);
}
}void finds(int x) {
son[x]=maxs[x]=0;
ss[x]=1;
for (int now=head[x],v;now;now=e[now].next) {
v=e[now].to;
if (v!=fa[x]) {
rd[v]=e[now].w;
dis[v]=dis[x]+e[now].w;
fa[v]=x;
dep[v]=dep[x]+1;
finds(v);
ss[x]+=ss[v];
if (ss[v]>maxs[x]) {
maxs[x]=ss[v];
son[x]=v;
}
}
}
}void link(int x,int anc) {
tid[x]=++dfn;
rlen[dfn]=rd[x];
top[x]=anc;
if (son[x]) {
link(son[x],anc);
for (int now=head[x],v;now;now=e[now].next) {
v=e[now].to;
if (v!=fa[x] && v!=son[x]) {
link(v,v);
}
}
}
}inline void init() {
input();
fa[1]=1;
dis[1]=0;
dep[1]=1;
finds(1);
link(1,1);
}int getLCA(int u,int v) {
while (top[u]!=top[v]) {
if (dep[top[u]]>dep[top[v]]) {
swap(u,v);
}
v=fa[top[v]];
}
if (dep[u]<dep[v]) return u;
return v;
}inline void getlen() {
for (int i=1;i<=m;i++) {
int lca=getLCA(rs[i],rt[i]);
len[i]=dis[rs[i]]+dis[rt[i]]-(dis[lca]<<1);
maxl=max(maxl,len[i]);
}
}inline void mark(int x) {
int u=rs[x],v=rt[x];
while (top[u]!=top[v]) {
if (dep[top[u]]>dep[top[v]]) {
swap(u,v);
}
// mark top[v] + ~ v +
t[tid[top[v]]]++,t[tid[v]+1]--;
v=fa[top[v]];
}
if (dep[u]>dep[v]) {
swap(u,v);
}
// mark u - ~ v +
t[tid[u]+1]++,t[tid[v]+1]--;
}inline bool check(int x) {
memset(t,0,sizeof(t));
int cnt=0;
for (int i=1;i<=m;i++) {
if (len[i]>x) {
mark(i);
cnt++;
}
}
for (int i=1;i<=n;i++) {
t[i]+=t[i-1];
if (t[i]>=cnt && maxl-x<=rlen[i]) return 1;
}
return 0;
}int main() {
//freopen("transport.in","r",stdin);
//freopen("transport.out","w",stdout);
init();
getlen();
int l=0,r=maxl,mid;
while (l<r) {
mid=(l+r)>>1;
if (check(mid)) r=mid;
else l=mid+1;
}
printf("%d\n",l);
return 0;
} -
-12016-07-22 10:29:51@
同样的程序,bzoj上ac,这tle= =
-
-12015-11-14 19:09:41@
看到这题秒想出LCA。可是...我考试前5 6天才接触的LCA。。。然后我已经前年考过了就懒了下...我擦,然后我就哭了。
于是只好打了个spfa+快排预先读入每个点计算所有计划所需要的时间。用了类似底楼的贪心...希望能骗点分吧。呜呜呜呜
day2炸的飞起。还在浙江....
-
-12015-11-11 09:18:06@
二分答案之后求树上路径交……树剖+差分数组比较好办呐……考场上狂撸倍增QWQ
- 1
信息
- ID
- 1983
- 难度
- 8
- 分类
- (无)
- 标签
- 递交数
- 2442
- 已通过
- 333
- 通过率
- 14%
- 被复制
- 9
- 上传者