/*
可以并查集维护
可以发现,某个联通快出现大于等于2个环,一定无法分配。
有解要么一个环,要么没有环。
一个环时答案等于点数乘2(顺时针或逆时针)。
没有环是树,对于一个n个点的树,方案一定有n种(不连某个点)。
*/
#include<iostream>
#include<cstdio>
#include<cstring> #define N 100007
#define mod 1000000007
#define ll long long using namespace std;
ll n,m,ans,cnt;
ll fa[N],siz[N],num[N];
bool vis[N]; inline ll read()
{
ll x=,f=;char c=getchar();
while(c>''||c<''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
} ll find(ll x){return x==fa[x]?x:fa[x]=find(fa[x]);} void merge(ll x,ll y)
{
fa[y]=x;
siz[x]+=siz[y];num[x]+=num[y];
} int main()
{
ll x,y;ans=;
n=read();m=read();
for(ll i=;i<=n;i++) fa[i]=i,siz[i]=;
for(ll i=;i<=m;i++)
{
x=read();y=read();
ll r1=find(x),r2=find(y);
if(r1!=r2) merge(r1,r2);
else num[r1]++;
}
for(ll i=;i<=n;i++)
{
ll now=find(i);
if(vis[now]) continue;vis[now]=;
if(num[now]>) continue;
if(num[now]==) ans=(ans*)%mod;
if(!num[now]) ans=(ans*siz[now])%mod;
}
printf("%lld\n",ans%mod);
return ;
}

/*
若两数k进制下相同
那么k进制后它们一定偶数位为0,奇数位为0~k-1
从高位递推可能的情况累加答案即可。
*/
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath> #define N 100
#define ll long long using namespace std;
ll n,k,ans,cnt,len;
ll Pow[N];
int a[N]; int main()
{
scanf("%lld%lld",&n,&k);
while(n){Pow[++len]=n%k;n/=k;}
if(len%==)
{
ans=pow(k,len>>);
printf("%lld\n",ans);
}
else
{
for(int i=len;i>=;i--)
{
if(Pow[i])
{
if(i%==)
{
ans+=1ll*pow(k,i/);
break;
}
else ans+=1ll*Pow[i]*pow(k,i/);
if(i==) ++ans;
}
}
}
printf("%lld\n",ans);
return ;
}

旅行

/*
嗯,需要维护每个点到根的距离。
首先开始的时候选取叶子结点一定比中间节点优。
当选择了一条链的时候,会对哪些点有影响呢?
答案当然是在这条链上的点的子树。把这个点的子树权值减掉这个点的权值就好。
看到子树,想到dfs序。又因为要查询最大值,所以可以想到用线段树实现。
线段树每个节点维护原树每个点到根的距离的最大值和原树每个节点dfs序所对应的点的编号。
每次查询区间最大值,然后删去这条链,每次删的时候更新子树权值(区间减法)。
删除把这个点的权值赋值为0就好。然后往上走,走的时候到某个点权值为0那么就停。
因为如果某个点权值为0,那么他到根的路径上所有点权都为零。恩。
*/
#include<iostream>
#include<cstdio>
#include<cstring> #define N 200007
#define ll long long using namespace std;
ll n,m,ans,cnt,tot;
ll head[N],dis[N],fa[N];
ll S[N],pos[N],T[N],a[N];
struct edge{
int u,v,net,w;
}e[N<<];
struct tree{
ll l,r,mx,pos,flag;
}tr[N<<]; namespace seg
{
void pushup(int k)
{
if(tr[k<<].mx>tr[k<<|].mx) tr[k].mx=tr[k<<].mx,tr[k].pos=tr[k<<].pos;
else tr[k].mx=tr[k<<|].mx,tr[k].pos=tr[k<<|].pos;
}
void pushdown(int k)
{
tr[k<<].flag+=tr[k].flag;tr[k<<|].flag+=tr[k].flag;
tr[k<<].mx+=tr[k].flag;tr[k<<|].mx+=tr[k].flag;
tr[k].flag=;
}
void build(int k,int l,int r)
{
tr[k].l=l;tr[k].r=r;
if(l==r)
{
tr[k].mx=dis[pos[l]],tr[k].pos=pos[l];
return;
}
int mid=l+r>>;
build(k<<,l,mid);build(k<<|,mid+,r);
pushup(k);
}
void update(int k,int l,int r,int z)
{
if(tr[k].l==l && tr[k].r==r)
{
tr[k].mx+=z;tr[k].flag+=z;
return;
}
pushdown(k);
int mid=tr[k].l+tr[k].r>>;
if(r<=mid) update(k<<,l,r,z);
else if(l>mid) update(k<<|,l,r,z);
else update(k<<,l,mid,z),update(k<<|,mid+,r,z);
pushup(k);
} }using namespace seg; inline void add(int u,int v)
{
e[++cnt].v=v;e[cnt].net=head[u];head[u]=cnt;
} inline ll read()
{
ll x=,f=;char c=getchar();
while(c>''||c<''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
} void dfs(int u,int last,ll sum)
{
S[u]=++tot;pos[tot]=u;dis[u]=sum;
for(int i=head[u];i;i=e[i].net)
{
int v=e[i].v;
if(v==last) continue;
fa[v]=u;dfs(v,u,sum+a[v]);
}T[u]=tot;
} void change(int u)
{
while(a[u])
{
update(,S[u],T[u],-a[u]);
a[u]=;u=fa[u];
}
} int main()
{
freopen("tour.in","r",stdin);
freopen("tour.out","w",stdout);
int x,y;
n=read();m=read();
for(int i=;i<=n;i++) a[i]=read();
for(int i=;i<n;i++)
{
x=read();y=read();
add(x,y);add(y,x);
}ans=a[];a[]=;dfs(,,);
build(,,n);
while(m--)
{
tree Tr=tr[];ans+=Tr.mx;
change(Tr.pos);
}
printf("%lld\n",ans);
return ;
}

清北考前刷题day3下午好的更多相关文章

  1. 清北考前刷题day1下午好

    水题(water) Time Limit:1000ms   Memory Limit:128MB 题目描述 LYK出了道水题. 这个水题是这样的:有两副牌,每副牌都有n张. 对于第一副牌的每张牌长和宽 ...

  2. 清北考前刷题da7下午好

    三向城 /* 原图一定是一棵完全二叉树. 根节点是x,左节点是x*2,右节点是x*2+1 转化为二进制往左右走就很明显了. */ #include<iostream> #include&l ...

  3. 清北考前刷题day6下午好

    /* 贪心 负数一定不取 枚举最高位是1 且答案取为0的 位置, 更新答案. */ #include<iostream> #include<cstdio> #include&l ...

  4. 清北考前刷题da5下午好

    /* (4,1)*(3,1)*(2,1)的话1变成2然后一直是2 2变成1然后变成3 3变成1然后变成4 4变成1 */ #include<iostream> #include<cs ...

  5. 清北考前刷题day4下午好

    /* 辗转相除,每次计算多出现了几个数. */ #include<iostream> #include<cstdio> #include<cstring> #inc ...

  6. 清北考前刷题day2下午好

    #include<iostream> #include<cstdio> #include<cstring> #include<stack> #defin ...

  7. 清北考前刷题day4早安

      LI /* 没有考虑次大值有大于一个的情况 */ #include<iostream> #include<cstdio> #include<cstring> # ...

  8. 清北考前刷题day7早安

  9. 清北考前刷题day6早安

    #include<iostream> #include<cstdio> #include<cstring> #include<algorithm> #d ...

随机推荐

  1. 【Codeforces 231C】To Add or Not to Add

    [链接] 我是链接,点我呀:) [题意] 题意 [题解] 把数组排个序, 显然优先用大的且小于枚举的数字a[i]的数字变成a[i] 那么肯定有一个范围j. 然后a[j~i-1]都能在k花费以内变成a[ ...

  2. mysql ab主从复制出错及解决过程

    一.mysql主从服务器报错描述:Slave_IO_Running=NO,Slave_SQL_Running=YES,Last_Errno=0 mysql slave stop ; mysql sla ...

  3. Linux下汇编语言学习笔记4 ---

    这是17年暑假学习Linux汇编语言的笔记记录,参考书目为清华大学出版社 Jeff Duntemann著 梁晓辉译<汇编语言基于Linux环境>的书,喜欢看原版书的同学可以看<Ass ...

  4. [bzoj1510][POI2006]Kra-The Disks_暴力

    Kra-The Disks bzoj-1510 POI-2006 题目大意:题目链接. 注释:略. 想法:不难发现其实只有前缀最小值是有效的. 进而我们把盘子一个一个往里放,弄一个自底向上的指针往上蹦 ...

  5. Spring Cloud体系实现标签路由

    如果你正在使用Spring Cloud体系,在实际使用过程中正遇到以下问题,可以阅读本文章的内容作为后续你解决这些问题的参考,文章内容不保证无错,请务必仔细思考之后再进行实践. 问题: 1,本地连上开 ...

  6. Servlet表单数据处理

    以下内容引用自http://wiki.jikexueyuan.com/project/servlet/form-data.html: 当需要从浏览器到Web服务器传递一些信息并最终传回到后台程序时,一 ...

  7. base64加解密字符串

    import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOExceptio ...

  8. TCP网络库:Acceptor、TcpServer、TcpConnection

    Acceptor类:用于接收新的TCP连接,该类是内部class,供TcpServer使用,生命期由TcpServer控制 类成员: class Acceptor : boost::noncopyab ...

  9. go语言中的timer 和ticker定时任务

    https://mmcgrana.github.io/2012/09/go-by-example-timers-and-tickers.html --------------------------- ...

  10. go-import下划线的作用

    原文:http://studygolang.com/articles/4356 ------------------------------------------------------------ ...