QOJ.ac
QOJ
ID | Problem | Submitter | Result | Time | Memory | Language | File size | Submit time | Judge time |
---|---|---|---|---|---|---|---|---|---|
#610371 | #8941. Even or Odd Spanning Tree | ucup-team4975# | TL | 4ms | 26280kb | C++23 | 3.6kb | 2024-10-04 15:44:55 | 2024-10-04 15:44:55 |
Judging History
answer
#define _CRT_SECURE_NO_WARNINGS
#include<bits/stdc++.h>
using namespace std;
const int N=500010;
const int inf=1000000000;
typedef long long ll;
int n,m,istree[N],fa[N],f[N][20],g[N][20],_fa[N][20],llg[N]; //f odd edge g even edge
unordered_map<ll,int>id;
inline ll _(int x,int y){return 1ll*(1ll*x-1)*n+y;}
inline void init(int x){for(int i=1;i<=x;i++)fa[i]=i;}
inline int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);}
struct Edge{
int x,y,z,id;
inline bool operator<(const Edge that){
return z<that.z;
}
}a[N];
struct Graph{
int tot,head[N],suiv[N],edge[N],ver[N],dep[N];
inline void init(int x)
{
for(int i=1;i<=x;i++)
head[i]=0;
tot=0;
}
inline void lnk(int x,int y,int z)
{
ver[++tot]=y;
edge[tot]=z;
suiv[tot]=head[x];
head[x]=tot;
}
inline void dfs(int x,int fa=-1)
{
for(int i=head[x];i;i=suiv[i])
{
int y=ver[i],z=edge[i];
if(y==fa)continue;
if(z&1)f[y][0]=z,g[y][0]=-inf;
else g[y][0]=z,f[y][0]=-inf;
_fa[y][0]=x;
dep[y]=dep[x]+1;
dfs(y,x);
}
}
inline void cal()
{
for(int i=1;i<=n;i++)
for(int j=1;1<<j<=dep[i];j++)
_fa[i][j]=_fa[_fa[i][j-1]][j-1];
for(int i=1;i<=n;i++)
for(int j=1;1<<j<=dep[i];j++)
f[i][j]=max(f[i][j-1],f[_fa[i][j-1]][j-1]),
g[i][j]=max(g[i][j-1],g[_fa[i][j-1]][j-1]);
}
inline int query(int x,int y,int op) //op==0 query odd(f) op==1 query even(g)
{
int res=-inf;
if(dep[x]<dep[y])swap(x,y);
while(dep[x]>dep[y])
{
if(!op)res=max(res,f[x][llg[dep[x]-dep[y]]]);
else res=max(res,g[x][llg[dep[x]-dep[y]]]);
x=_fa[x][llg[dep[x]-dep[y]]];
}
if(x==y)return res;
for(int i=llg[dep[x]];i>=0;i--)
if(_fa[x][i]!=_fa[y][i])
{
if(!op)res=max({res,f[x][i],f[y][i]});
else res=max({res,g[x][i],g[y][i]});
x=_fa[x][i];
y=_fa[y][i];
}
if(!op)return max({res,f[x][0],f[y][0]});
else return max({res,g[x][0],g[y][0]});
}
}e;
signed main()
{
for(int i=1;i<N;i++)
llg[i]=log2(i);
int T;
scanf("%d",&T);
while(T--)
{
id.clear();
ll evenans=0,oddans=0;
scanf("%d%d",&n,&m);
for(int i=1;i<=m;i++)
scanf("%d%d%d",&a[i].x,&a[i].y,&a[i].z);
sort(a+1,a+1+m);
for(int i=1;i<=m;i++)
a[i].id=i;
init(n);
int cnt=0;
for(int i=1;i<=m;i++)
{
int fx=find(a[i].x),fy=find(a[i].y);
if(fx==fy)continue;
fa[fx]=fy;
istree[a[i].id]=1;
evenans+=1ll*a[i].z;
id[_(a[i].x,a[i].y)]=id[_(a[i].y,a[i].x)]=a[i].id;
e.lnk(a[i].x,a[i].y,a[i].z);
e.lnk(a[i].y,a[i].x,a[i].z);
cnt++;
}
if(cnt<n-1)
{
puts("-1 -1");
for(int i=1;i<=m;i++)
istree[i]=0;
e.init(n);
continue;
}
//for(int i=1;i<=n;i++)
// for(int j=0;j<20;j++)
// f[i][j]=g[i][j]=-inf;
_fa[1][0]=1;
e.dep[1]=1;
e.dfs(1);
e.cal();
//while(1)
//{
// int x,y,z;
// cin>>x>>y>>z;
// cout<<e.query(x,y,z)<<endl;
//}
//cout<<evenans<<endl;
ll add=inf;
for(int i=1,treeedge;i<=m;i++)
{
int res=-inf;
if(istree[a[i].id])continue;
if((treeedge=id[_(a[i].x,a[i].y)]))
{
if((a[treeedge].z%2)==(a[i].z%2))continue;
add=min(add,1ll*abs(a[treeedge].z-a[i].z));
continue;
}
if(a[i].z&1)//odd
res=e.query(a[i].x,a[i].y,1);
else res=e.query(a[i].x,a[i].y,0);
if(res!=-inf)
add=min(add,abs(res-1ll*a[i].z));
}
//cout<<"!"<<endl;
if(add!=inf)oddans=evenans+add;
else oddans=-1;
if(evenans&1)swap(evenans,oddans);
printf("%lld %lld\n",evenans,oddans);
for(int i=1;i<=m;i++)
istree[i]=0;
e.init(n);
}
return 0;
}
Details
Tip: Click on the bar to expand more detailed information
Test #1:
score: 100
Accepted
time: 4ms
memory: 26280kb
input:
3 2 1 1 2 5 3 1 1 3 1 4 4 1 2 1 1 3 1 1 4 1 2 4 2
output:
-1 5 -1 -1 4 3
result:
ok 6 numbers
Test #2:
score: -100
Time Limit Exceeded
input:
10000 16 50 1 2 649251632 2 3 605963017 3 4 897721528 4 5 82446151 5 6 917109168 6 7 79647183 7 8 278566178 7 9 573504723 3 10 679859251 8 11 563113083 1 12 843328546 10 13 594049160 11 14 997882839 7 15 569431750 2 16 244494799 16 7 960172373 13 4 317888322 13 3 446883598 9 3 678142319 5 8 43208692...