T1

朴素dp很好想,设 \(dp_{u,0/1}\) ,表示以 \(u\) 为根的子树,选/不选 \(u\) 所产生的最大贡献。

转移方程则有,

\[dp_{u,0} = \prod_{v\in son_{u}}\max(dp_{v,0},dp_{v,1})
\]
\[dp_{u,1} = w_{u}\prod_{v\in son_{u}}dp_{v,0}
\]

发现在dp过程中直接取模,大小关系无法确定。

考虑对每个难度取 \(\log\) ,在dp的过程中记录dp到当前值的 \(\log\) 和,dp转移时的比较大小时由比值改成比 \(\log\) 即可,dp数组仍然记录当前最大值,dp时就可以取模了。

当值很大,比较大小时,直接计算不太好搞尤其是再带个取模,可以通过取 \(log\) 将乘法变为加法。

当然,也可以用高精,1e9+7进制,最后一位就是答案,但毕竟不是人人都是 太虚真人

Code
#include<cmath>
#include<cstdio>
#define MAX 200020
#define re register
#define int long long
int n,w[MAX];
double sum[MAX][2];
struct graph
{
int next;
int to;
}edge[MAX<<1];
int cnt=1,head[MAX];
inline void add(int u,int v)
{ edge[++cnt] = (graph){head[u],v},head[u] = cnt; }
namespace read
{
struct stream
{
template<typename type>inline stream &operator >>(type &s)
{
int w=1; s=0; char ch=getchar();
while(ch<'0'||ch>'9'){ if(ch=='-')w=-1; ch=getchar(); }
while(ch>='0'&&ch<='9'){ s=s*10+ch-'0'; ch=getchar(); }
return s*=w,*this;
}
}cin;
}using read::cin;
namespace OMA
{
int dp[MAX][2];
const int p = 1e9+7;
inline int max(int a,int b)
{ return a>b?a:b; }
inline void dfs(int u,int fa)
{
dp[u][1] = w[u],dp[u][0] = 1;
for(re int i=head[u],v; i; i=edge[i].next)
{
v = edge[i].to;
if(v!=fa)
{
dfs(v,u);
(dp[u][1] *= dp[v][0]) %= p;
sum[u][1] += sum[v][0];
//(dp[u][0] *= max(dp[v][0],dp[v][1])) %= p;
if(sum[v][0]>sum[v][1])
{ (dp[u][0] *= dp[v][0]) %= p,sum[u][0] += sum[v][0]; }
else
{ (dp[u][0] *= dp[v][1]) %= p,sum[u][0] += sum[v][1]; }
}
}
}
signed main()
{
//system("./data");
//freopen("node.in","r",stdin);
cin >> n;
for(re int i=1; i<=n; i++)
{ cin >> w[i]; sum[i][1] = log(w[i]); }
for(re int i=2,u,v; i<=n; i++)
{ cin >> u >> v; add(u,v),add(v,u); }
dfs(1,0);
//printf("%0.6lf %0.6lf\n",sum[1][0],sum[1][1]);
//printf("%lld %lld\n",dp[1][0],dp[1][1]);
printf("%lld\n",sum[1][0]>sum[1][1]?dp[1][0]:dp[1][1]);
return 0;
}
}
signed main()
{ return OMA::main(); }

T2

考虑将 \(n\) 划分成形如 \(p,2p,,,,2^{k}p\) 的形式,其中 \(p\) 为奇数。

举个例子,\(n=8\) , 则划分成了:

  1. 1 2 4 8
  2. 3 6
  3. 5
  4. 7

不难发现,链长都是 \(log\) 级别的,每次都是乘上2出来下一个数。且 \(A,B\) 在同一条链上肯定是间隔着选的。

考虑将链分奇偶来考虑。

  • 对于链长为奇数的,则必然是 \(A\) 比 \(B\) 多一个,或者 \(B\)比 \(A\) 多一个,所以最终 \(A\) 的大小会在一个区间 \([l,r]\) 内,\(l\) 全少个1,\(r\) 全多个1。 比 \(l\) 多的那部分必然都是由奇数链贡献来的,所以最后计算组合数 \(\tbinom{k_{1}}{m-l}\) ,其中,\(k_{1}\) 表示奇数链的个数。

  • 对于链长为偶数的,则必然是给 \(A\) 分一半, \(B\) 分一半,则对答案会产生2的贡献。每条链都能产生2的贡献,设一共有 \(k_{2}\) 条偶数链,则对答案产生的贡献为 \(2^{k_{2}}\)

则对于每次询问,最后答案即为 \(2^{k_{2}}\tbinom{k_{1}}{m-l}\) ,要用到 \(lucas\) 。

Code
#include<cstdio>
#define MAX 100100
#define re register
#define int long long
int n,q,l,pow=1,tub;
const int p = 10000019;
int c[p+1],inv[p+1];
namespace read
{
struct stream
{
template<typename type>inline stream &operator >>(type &s)
{
int w=1; s=0; char ch=getchar();
while(ch<'0'||ch>'9'){ if(ch=='-')w=-1; ch=getchar(); }
while(ch>='0'&&ch<='9'){ s=s*10+ch-'0'; ch=getchar(); }
return s*=w,*this;
}
}cin;
}using read::cin;
namespace OMA
{
inline int quickpow(int a,int b)
{
int ans = 1;
while(b)
{
if(b&1)
{ ans = ans*a%p; }
a = a*a%p;
b >>= 1;
}
return ans;
}
inline int C(int n,int m)
{ return m>n?0:c[n]*inv[m]%p*inv[n-m]%p; }
inline int lucas(int n,int m)
{ return (m>n||m<0)?0:(!m?1:lucas(n/p,m/p)*C(n%p,m%p)%p); }
signed main()
{
cin >> n >> q;
c[0] = inv[0] = 1;
for(re int i=1; i<=n; i++)
{ c[i] = i*c[i-1]%p; }
inv[n] = quickpow(c[n],p-2);
for(re int i=n-1; i; i--)
{ inv[i] = (i+1)*inv[i+1]%p; }
int last = n,cnt = 1;
for(re int i=0; ~i; i++)
{
if(!last)
{ break ; }
int tmp = last-n/cnt;
l += (tmp+(last&1))/2*(i/2);
if(i&1)
{ (tub += (tmp+(last&1))/2) %= p; }
else
{ pow = pow*quickpow(2,(tmp+(last&1))/2)%p; }
last = n/cnt;
cnt *= 2;
}
for(re int i=1,m; i<=q; i++)
{ cin >> m; printf("%lld\n",pow*lucas(tub,m-l)%p); }
return 0;
}
}
signed main()
{ return OMA::main(); }

T3

阴间dp

首先,对于本题的dp,有两种,

  1. 设 \(f_{i,j}\) 表示当前dp到数字 \(i\) 即最大数为 \(i\) ,总和为 \(j\) 的方案数。转移则有,\(f_{i,j}=f_{i-1,j}+f_{i-1,j-k}\;,k\in[1,j]\) 。

  2. 设 \(g_{i,j}\) 表示划分成了\(i\) 个数,总和为 \(j\) 的方案数。转移则有,\(g_{i,j}=g_{i-1,j}+g_{i,j-i}\) 。

两种dp单独搞都是 \(O(n^{2})\) 。

有一个比较没见过的常见的套路然而我并不知道,将 \(n\) 按 \(\sqrt{n}\) 分成两部分,分别使用两种dp,可做到 \(O(n\sqrt{n})\) 。

Code
#include<cmath>
#include<cstdio>
#include<cstring>
#define MAX 100010
#define re register
namespace OMA
{
int x,y,n,p;
int f[MAX],g[MAX],sum[MAX];
inline int max(int a,int b)
{ return a>b?a:b; }
inline int DP(int a)
{
memset(f,0,sizeof(f));
memset(g,0,sizeof(g));
memset(sum,0,sizeof(sum));
f[0] = g[0] = sum[0] = 1;
int b = max(sqrt(n),a);
for(re int i=a; i<=b; i++)
{
for(re int j=i; j<=n; j++)
{ (f[j] += f[j-i]) %= p; }
}
for(re int i=1; i<=n/b; i++)
{
int c = i*b+i;
for(re int j=i; j+c<=n; j++)
{ (g[j] += g[j-i]) %= p; }
for(re int j=0; j+c<=n; j++)
{ (sum[j+c] += g[j]) %= p; }
}
int ans = 0;
for(re int i=0; i<=n; i++)
{ (ans += 1LL*f[i]*sum[n-i]%p) %= p; }
return ans;
}
signed main()
{
scanf("%d%d%d%d",&x,&y,&n,&p);
printf("%d\n",((DP(x)-DP(y+1))%p+p)%p);
return 0;
}
}
signed main()
{ return OMA::main(); }

T4

暴力乱搞10pts。

正解:

要用manacher,不会,所以咕了。

官方题解

noip42的更多相关文章

  1. 2021.8.17考试总结[NOIP42]

    $\huge{取模不能比大小!}$ $\huge{取模不能比大小!}$ $\huge{取模不能比大小!}$ 有了打地鼠的前车之鉴,我深信树规板子是可以出现在联赛题里的. 所以T1十分钟码完直接溜了,后 ...

  2. 性能测试-ORACLE性能监控

    通过lr做性能测试的过程,通过监控Oracle数据库的性能 采用的监控工具:PeOny PeOny安装 1. 安装服务端 1) LINUX平台安装 解压缩peony3.x.0.x.tar.gz文件,b ...

随机推荐

  1. Java | 参数传值机制

    值传递 java中,方法中所有的参数的都是"值传递",就是传递的是原来值的副本,不是原来的参数,因此,改变不会影响到原来的参数. 基本数据类型参数的传值 传递的都是副本,改变以后不 ...

  2. C语言:预定义的宏

    预定义宏就是已经预先定义好的宏,我们可以直接使用,无需再重新定义.ANSI C 规定了以下几个预定义宏,它们在各个编译器下都可以使用: __LINE__:表示当前源代码的行号: __FILE__:表示 ...

  3. VisualEffectGraph基础操作 --创建VEG项目步骤讲解

    一:建立VEG项目步骤 首先打开Unity Hub,  使用unity2020.1 新建项目(本技术博客,默认使用unity2020.1 版本演示),选择HDRP 高清渲染管线,确定项目目录与名称. ...

  4. 给potplayer配置iptv源,看所有你想看的电视

    目录 一.展示: 二.下载 三.播放 一.展示: 二.下载 Github 上的开源项目:iptv-org/iptv 传送门: https://github.com/iptv-org/iptv 该项目包 ...

  5. docker容器技术基础之linux cgroup、namespace

    一.开头 接触过docker的同学多多少少听过这样一句话"docker容器通过linux namespace.cgroup特性实现资源的隔离与限制".今天我们来尝试学习一下这两个东 ...

  6. Jenkins 进阶篇 - 权限案例实践

    为了演示方便,我这里创建了如下的一些任务,各个任务由包含在对应的目录中树形结构大概如下: jenkins-authority 目录类型任务 ├── dev-project-group 目录类型任务 │ ...

  7. Leetcode6. Z 字形变换

    > 简洁易懂讲清原理,讲不清你来打我~ 输入字符串,按下右上下右上排列后输出字符串![在这里插入图片描述](https://img-blog.csdnimg.cn/4578280a7c1848c ...

  8. vue(22)Vuex的安装与使用

    前言 每一个 Vuex 应用的核心就是 store(仓库).store基本上就是一个容器,它包含着你的应用中大部分的状态 (state).Vuex 和单纯的全局对象有以下两点不同: Vuex 的状态存 ...

  9. 测试笔记01-Git

    Git工具 Git 一个分布式版本管理工具, 学习资料: https://www.liaoxuefeng.com/wiki/896043488029600/1317161920364578 其中:解决 ...

  10. 《鸟哥Linux私房菜》 全套视频和PDF资料—— 老段带你学鸟哥Linux视频课程

    <鸟哥的Linux私房菜-基础篇-服务器篇(第三版)>学习Linux极为经典的入门资料,但是还是很多同学难以坚持系统的看完整本书,最终以放弃而告终. 为了帮助大家更容易入门Linux,老段 ...