题目传送门

这题自己(真正)思考了很久(欣慰)。

(轻而易举)地发现这是一棵树后,打算从Dfs序中下功夫,推敲了很久规律,没看出来(太弱了)。

开始手动枚举距离为2的情况,模模糊糊有了一些概念,但没有总结。(敲黑板:题目中发现规律与重要性质注意总结!

其实,距离为2的情况只有两种:祖父/兄弟。

一个小时后放弃治疗。开始想暴力,很好想,我们对于每个点,枚举他的出边,再在每个出边中的出边中进行枚举,储存距离为2 的点。期望得分60pts.

大力交了一下:40pts,AC*6,WA*2,MLE*4.

MLE还有情可缘,vector开动态数组可能炸了,WA的那两个喵喵喵?

 #include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector> using namespace std;
typedef long long ll; int n,tot;
ll sum,ans,p=;
int head[];
ll val[];
bool vis[];
struct node{
int to,next;
}edge[];
vector<int>law[]; void add(int x,int y)
{
edge[++tot].to=y;
edge[tot].next=head[x];
head[x]=tot;
} void dfs(int u)
{
for(int i=head[u];i;i=edge[i].next)
{
int v=edge[i].to;
for(int j=head[v];j;j=edge[j].next)
{
int g=edge[j].to;
if(g==u) continue;
law[u].push_back(g);
}
}
} int main()
{
scanf("%d",&n);
for(int i=;i<=n-;i++)
{
int x=,y=;
scanf("%d%d",&x,&y);
add(x,y);
add(y,x);
}
for(int i=;i<=n;i++) scanf("%d",&val[i]);
// dfs_pre(1);
// memset(vis,0,sizeof(vis));
// dfs(1,0);
for(int i=;i<=n;i++)
dfs(i);
for(int i=;i<=n;i++)
{
for(int j=;j<law[i].size();j++)
{
int u=i,v=law[i][j];
ll tmp=val[u]%p*val[v]%p;
(sum+=tmp)%=p;
(ans=max(ans,tmp))%=p;
}
}
printf("%lld %lld",ans,sum);
return ;
}

后来经过冷静分析看题解发现并不需要存儿子,当时每次更新一下就行了。而且最大值并不需要取膜。

再交一下60pts。TLE4个点,正常。

 #include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector> using namespace std;
typedef long long ll; int n,tot;
ll sum,ans,p=;
int head[];
ll val[];
struct node{
int to,next;
}edge[]; void add(int x,int y)
{
edge[++tot].to=y;
edge[tot].next=head[x];
head[x]=tot;
} void dfs(int u)
{
for(int i=head[u];i;i=edge[i].next)
{
int v=edge[i].to;
for(int j=head[v];j;j=edge[j].next)
{
int g=edge[j].to;
if(g==u) continue;
ans=max(ans,val[u]*val[g]);
(sum+=val[u]*val[g])%=p;
}
}
} int main()
{
scanf("%d",&n);
for(int i=;i<=n-;i++)
{
int x=,y=;
scanf("%d%d",&x,&y);
add(x,y);
add(y,x);
}
for(int i=;i<=n;i++) scanf("%d",&val[i]);
// dfs_pre(1);
// memset(vis,0,sizeof(vis));
// dfs(1,0);
for(int i=;i<=n;i++)
dfs(i);
printf("%lld %lld",ans,sum);
return ;
}

正解:我们只需要枚举每个点与他相连的每一条边即可,统计出与每个点相邻的最大点值与次大点值,全局最值用(最大点值*次大点值)更新,全局和用“乘法分配律“”维护。

 #include<cstdio>
#include<algorithm> using namespace std;
typedef long long ll; int n,tot;
ll p=,sum,ans;
int head[],val[];
struct node{
int to,next;
}edge[]; void add(int x,int y)
{
edge[++tot].to=y;
edge[tot].next=head[x];
head[x]=tot;
} ll llmax(ll a,ll b)
{
if(a>=b) return a;
else return b;
} void update(int x)
{
int maxx=,maxs=;
ll cnt=;
for(int i=head[x];i;i=edge[i].next)
{
int y=edge[i].to;
if(val[y]>maxx) maxs=maxx,maxx=val[y];
else if(val[y]>maxs) maxs=val[y];
(sum+=cnt*val[y])%=p;
(cnt+=val[y])%=p;
}
ans=llmax(ans,maxs*maxx);
} int main()
{
scanf("%d",&n);
for(int i=;i<=n-;i++)
{
int x=,y=;
scanf("%d%d",&x,&y);
add(x,y);add(y,x);
}
for(int i=;i<=n;i++) scanf("%d",&val[i]);
for(int i=;i<=n;i++) update(i);
printf("%lld %lld",ans,*sum%p);
return ;
}

最新文章

  1. PS教程:20个新鲜出炉的 Photoshop 中级教程
  2. BestCoder Round #75 解题报告
  3. 总有你需要的之 ios 小技巧 (下)
  4. TDirectory.GetParent获取指定目录的父目录
  5. LDAP缓存命令
  6. spring实现数据库读写分离
  7. Xcode打包framework脚本
  8. Windows 10 IoT Serials 6 - 如何修改IoTStartupOnBoot.cmd文件
  9. 5_find grep sed awk 详解
  10. Atcoder 乱做
  11. Git命令执行漏洞
  12. 上海市2019年公务员录用考试第一轮首批面试名单(A类)
  13. 并发编程---IO模型
  14. Windows 下VC++6.0制作、使用动态库和静态库
  15. patch-test-and-proc
  16. SVN previous operation has not finished
  17. [ 转载 ] Http详解
  18. [LeetCode] 257. Binary Tree Paths_ Easy tag: DFS
  19. Replication--复制事务和复制命令
  20. Global Average Pooling Layers for Object Localization

热门文章

  1. Spring基于注解的配置概述
  2. JavaOne Online Hands-on Labs
  3. 【scrapy】Item及Spider
  4. Ansible 详细用法说明(一)
  5. BZOJ 1091([SCOI2003]分割多边形-分割直线)
  6. ext.net 2.5 导出excel的使用方法
  7. [LeetCode][Java] Roman to Integer
  8. Servlet访问Javabean并传结果给jsp
  9. [IT学习]Python pandas 学习
  10. iOS开发——高级篇——多线程dispatch_apply