hihocoder #1073 : 光棍节
描述
尽管付出了种种努力,jzp还是得过光棍节。
jzp非常不爽,但也无能为力,只能够哀叹起来他的命运。他想到了一位长者的人生经验:“人的一生,不光要靠自我奋斗,也要考虑历史的进程”。
他终于明白自己只是时运不济,所以又继续开始努力。终于在圣诞节当天把到了妹子。
jzp从此过上了快乐的现充生活,在圣诞节当天,他还和妹子玩起了有趣的游戏:
jzp的家里有一棵非常大的圣诞树,可以看成是一个n个点的无向联通无环图。每个点上都有一个正整数,JZP和妹子每次都会选择树上的一条路径,
这条路径的权值被定义为路径上所有点上数的最大公约数,JZP可以得到这个权值的分数。
JZP玩了一会儿有点腻了,他想知道对于每种可能的权值x,权值为x的不同路径的数量(a到b的路径和b到a的路径算作一种,a到a自身也算作一条路径。)
解题报告:
用时2h30min,N TLE
这题不难,直接暴力点分,复杂度可以证明是对的,因为从一个点下去,gcd只有可以是它的约数,所以是\(\sqrt n\)的,所以直接去重之后两重循环枚举,复杂度是\(O(n)\)的,所以复杂度是对的。但是本人太渣,点分都可以写错,找重心的数组一直没清空(那么居然是0.7s是怎么回事?)然后一交就TLE,还以为是常数太大,就一直在压,浪费了许久,写了两个代码,一个是所有方案-不合法的,另一个是用vector维护已经处理过的子树的gcd,处理完当前子树后,在处理另一个子树直接用当前子树的每一个gcd和vector里的元素匹配.
#include <algorithm>
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <vector>
#include <cmath>
#define RG register
#define il inline
#define iter iterator
#define Max(a,b) ((a)>(b)?(a):(b))
#define Min(a,b) ((a)<(b)?(a):(b))
using namespace std;
typedef long long ll;
const int N=1e5+5;
int n,val[N],head[N],num=0,nxt[N<<1],to[N<<1],sum;
void link(int x,int y){nxt[++num]=head[x];to[num]=y;head[x]=num;}
bool vis[N];int sz[N],root=0,son[N]={N};
il int gi(){
RG int str=0;RG char ch=getchar();
while(ch>'9' || ch<'0')ch=getchar();
while(ch>='0' && ch<='9')
str=(str<<1)+(str<<3)+ch-48,ch=getchar();
return str;
}
il void getroot(int x,int last){
int u;sz[x]=1;son[x]=0;
for(int i=head[x];i;i=nxt[i]){
u=to[i];if(u==last || vis[u])continue;
getroot(u,x);
sz[x]+=sz[u];son[x]=Max(son[x],sz[u]);
}
son[x]=Max(son[x],sum-sz[x]);
if(son[x]<son[root])root=x;
}
il int gcd(int x,int y){
int c;while(x%y){c=y;y=x%y;x=c;}
return y;
}
vector<int>q,s;ll t[N],ton[N];
void count(int x){
int szz=q.size(),gc;
for(int i=0;i<szz;i++){
gc=gcd(q[i],x);
t[gc]+=ton[q[i]];
}
}
int app[N];
il void calc(int x,int last,int g){
int u;
t[g]++;count(g);
if(!app[g])s.push_back(g);app[g]++;
for(int i=head[x];i;i=nxt[i]){
u=to[i];if(u==last || vis[u])continue;
if(g!=1)calc(u,x,gcd(val[u],g));
else calc(u,x,1);
}
}
il void solve(int x){
int u,szz;
for(int i=head[x];i;i=nxt[i]){
u=to[i];if(vis[u])continue;
calc(u,x,gcd(val[u],val[x]));
szz=s.size();
for(int j=0;j<szz;j++){
if(!ton[s[j]])q.push_back(s[j]);
ton[s[j]]+=app[s[j]];app[s[j]]=0;
}
s.clear();
}szz=q.size();
for(int i=0;i<szz;i++)ton[q[i]]=0;
q.clear();
}
il void dfs(int x){
int u;vis[x]=true;solve(x);
for(int i=head[x];i;i=nxt[i]){
u=to[i];if(vis[u])continue;
root=0;sum=sz[u];getroot(u,x);
dfs(root);
}
}
void work()
{
int x,y,lim=0;
scanf("%d",&n);
for(int i=1;i<=n;i++){
val[i]=gi();
lim=Max(lim,val[i]);
t[val[i]]++;
}
for(int i=1;i<n;i++){
x=gi();y=gi();
link(x,y);link(y,x);
}
sum=n;root=0;getroot(1,1);
dfs(root);
for(int i=1;i<=lim;i++)
if(t[i])printf("%d %lld\n",i,t[i]);
}
int main()
{
work();
return 0;
}
再是原来写的,清空数组后也能过
#include <algorithm>
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cmath>
#define RG register
#define il inline
#define iter iterator
#define Max(a,b) ((a)>(b)?(a):(b))
#define Min(a,b) ((a)<(b)?(a):(b))
using namespace std;
typedef long long ll;
const int N=1e5+5;
int n,val[N],head[N],num=0,nxt[N<<1],to[N<<1],sum;
void link(int x,int y){nxt[++num]=head[x];to[num]=y;head[x]=num;}
bool vis[N];int sz[N],root=0,son[N]={N};
il int gi(){
RG int str=0;RG char ch=getchar();
while(ch>'9' || ch<'0')ch=getchar();
while(ch>='0' && ch<='9')
str=(str<<1)+(str<<3)+ch-48,ch=getchar();
return str;
}
il void getroot(int x,int last){
int u;sz[x]=1;son[x]=0;
for(int i=head[x];i;i=nxt[i]){
u=to[i];if(u==last || vis[u])continue;
getroot(u,x);
sz[x]+=sz[u];son[x]=Max(son[x],sz[u]);
}
son[x]=Max(son[x],sum-sz[x]);
if(son[x]<son[root])root=x;
}
il int gcd(int x,int y){
int c;while(x%y){c=y;y=x%y;x=c;}
return y;
}
struct node{
int x;ll cnt;
bool operator <(const node &Pr)const{return x<Pr.x;}
}q[N];
int top=0;bool flag=false;
il void calc(int x,int last,int g){
int u;
if(g==1 && flag){q[++top].x=1;q[top].cnt=sz[x];return ;}
if(top && q[top].x==g)q[top].cnt++;
else q[++top].x=g,q[top].cnt=1;
for(int i=head[x];i;i=nxt[i]){
u=to[i];if(u==last || vis[u])continue;
if(g!=1)calc(u,x,gcd(val[u],g));
else calc(u,x,1);
}
}
ll t[N];
il void solve(int x,int op,int sta){
top=0;calc(x,x,sta);
int Yut=top;top=1;
sort(q+1,q+Yut+1);
RG int i,j;
for(i=2;i<=Yut;i++){
if(q[i].x==q[top].x)q[top].cnt+=q[i].cnt;
else q[++top]=q[i];
}
int tmp;
for(i=1;i<=top;i++){
t[q[i].x]+=q[i].cnt*(q[i].cnt-1)*op>>1;
for(j=i+1;j<=top;j++){
tmp=gcd(q[i].x,q[j].x);
t[tmp]+=q[i].cnt*q[j].cnt*op;
}
}
}
il void dfs(int x){
int u;vis[x]=true;
flag=false;solve(x,1,val[x]);
for(int i=head[x];i;i=nxt[i]){
u=to[i];if(vis[u])continue;
root=0;sum=sz[u];getroot(u,x);
flag=true;solve(u,-1,gcd(val[x],val[u]));
dfs(root);
}
}
void work()
{
int x,y,lim=0;
scanf("%d",&n);
for(int i=1;i<=n;i++){
val[i]=gi();
lim=Max(lim,val[i]);
t[val[i]]++;
}
for(int i=1;i<n;i++){
x=gi();y=gi();
link(x,y);link(y,x);
}
sum=n;root=0;getroot(1,1);
dfs(root);
for(int i=1;i<=lim;i++)
if(t[i])printf("%d %lld\n",i,t[i]);
}
int main()
{
work();
return 0;
}
hihocoder #1073 : 光棍节的更多相关文章
- 11.11光棍节工作心得——github/MVP
11.11光棍节工作心得 1.根据scrum meeting thirdday中前辈的指导进行学习 我在博客中贴了链接,竟然TrackBack引来了原博主,
- hiho 光棍节
描述 尽管付出了种种努力,jzp还是得过光棍节. jzp非常不爽,但也无能为力,只能够哀叹起来他的命运.他想到了一位长者的人生经验:“人的一生,不光要靠自我奋斗,也要考虑历史的进程”. 他终于明白自己 ...
- ABBYY FineReader去他的光棍节,我要我的双十一
今天就是双十一,全民剁手的双十一,一年仅一次的双十一,不只是半价的双十一.....此时此刻,多少钱拿起手机在疯狂购物,又有多少人死守着电脑,不敢怠慢一丁点机会,买着买着购物车就空了,然后才发现,咦!超 ...
- 光棍节程序员闯关秀writeup
答题链接https://1111.segmentfault.com/ 第一关 首先当然是右键查看源码啊 点击链接进入下一关 第二关 还是老样子,右键查看源码 这个key是要放在URL链接里敲回车的 第 ...
- zznu 1073: 海军节上的鸣炮声计算
这是个一不留神就会出错的题目,首先大家可以想到在同一时间可能会有多个炮同时发射,不过观众只能听到一响,所以需要注意! 而且刚开始的时候所有的船都会发射一发,输入的a, b, c,都是时间间隔,
- 11月11日光棍节考试总结hhh
好吧,第一题字符串裸栈就能A 第二题字典序没调完,先写的第一题和第三题暴力,第二题读题感觉自己不会写,其实也能写出来,浪费了好多时间 第三题DP大概是写的暴力,原本可以搞到20分,要交的时候发现自己少 ...
- NYOJ-451-光棍节的快乐
光棍节的快乐 时间限制:1000 ms | 内存限制:65535 KB 难度: 描写叙述 光棍们,今天是光棍节.聪明的NS想到了一个活动来丰富这个光棍节. 规则例如以下: 每一个光棍在一个纸条上写 ...
- “老坛泡新菜”:SOD MVVM框架,让WinForms焕发新春
火热的MVVM框架 最近几年最热门的技术之一就是前端技术了,各种前端框架,前端标准和前端设计风格层出不穷,而在众多前端框架中具有MVC,MVVM功能的框架成为耀眼新星,比如GitHub关注度很高的Vu ...
- quartz CronExpression表达式
一个cron表达式有至少6个(也可能7个)有空格分隔的时间元素.按顺序依次为1.秒(0~59)2.分钟(0~59)3.小时(0~23)4.天(月)(0~31,但是你需要考虑你月的天数)5.月(0~11 ...
随机推荐
- selenium 爬虫
from selenium import webdriver import time driver = webdriver.PhantomJS(executable_path="D:/pha ...
- 大数据技术Hadoop笔试题
Hadoop有高容错性的特点,并且设计用来部署在低廉的(low-cost)硬件上.以下是由应届毕业生网小编J.L为您整理推荐的面试笔试题目和经验,欢迎参考阅读. 单项选择题 1. 下面哪个程序负责 H ...
- 201421123042 《Java程序设计》第13周学习总结
1. 本周学习总结 以你喜欢的方式(思维导图.OneNote或其他)归纳总结多网络相关内容. 答: 2. 为你的系统增加网络功能(购物车.图书馆管理.斗地主等)-分组完成 为了让你的系统可以被多个用户 ...
- jenkins简单安装及配置(Windows环境)
jenkins是一款跨平台的持续集成和持续交付.基于Java开发的开源软件,提供任务构建,持续集成监控的功能,可以使开发测试人员更方便的构建软件项目,提高工作效率. Windows平台下,一般安装方法 ...
- guava-19.0和google-collections-1.0 的 ImmutableSet 类冲突
guava-19.0 google-collections-1.0 都有 ImmutableSet 类,包路径也一致,前者有 copyOf(Collection)? 一.应用报错: 二.解决办法 co ...
- Dijkstra的双栈算术表达式求值算法
这次来复习一下Dijkstra的双栈算术表达式求值算法,其实这就是一个计算器的实现,但是这里用到了不一样的算法,同时复习了栈. 主体思想就是将每次输入的字符和数字分别存储在两个栈中.每遇到一个单次结束 ...
- Opencv出现“_pFirstBlock == pHead”错误的解决方法
先说结论: opencv链接库使用错误. 1,确认VS工程属性中,opencv的链接库路径和版本正确. VS2013应该使用vc12目录,VS2012对应vc11目录.debug版和release版要 ...
- POJ2398【判断点在直线哪一侧+二分查找区间】
题意:同POJ2318 #include<algorithm> #include<cstdio> #include<cstdlib> #include<cst ...
- Java:逐行读、写文件、文件目录过滤的用法
import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; import java.io.I ...
- 'NoneType' object is not iterable
"TypeError: 'NoneType' object is not iterable" 一般是返回值为None同时赋值给了多个变量