#NTT,DP#U138580 简单的打击
题目
给出两个等长的序列\(a,b\),
重排序列\(b\),使得\(a+b\)众数出现的次数最多
分析
设\(f[i]\)表示众数为\(i\)的贡献,那么
\(f[i]=\sum_{j<i}min(A[j],B[i-j])\)
其中大写的\(a,b\)表示次数,但是这个东西很难做,
考虑把它变成正常的卷积的形式,那么就转换成判定,
判定\(f[i]\)是否能够达到阈值,那么显然就变成了只有\(0,1\)的卷积,
但是这样会超时,考虑阈值仅限于不超过一个常数就可以了
代码
#include <cstdio>
#include <cctype>
#include <queue>
#define rr register
using namespace std;
const int mod=998244353,N=100011,inv3=332748118; priority_queue<pair<int,int> >q;
int a[N],b[N],A[N],B[N],ff[N<<2],n,TOT[2],lim,gg[N<<2],ans[N<<2],Ans,Gmi[31],Imi[31];
inline signed iut(){
rr int ans=0; rr char c=getchar();
while (!isdigit(c)) c=getchar();
while (isdigit(c)) ans=(ans<<3)+(ans<<1)+(c^48),c=getchar();
return ans;
}
inline signed min(int a,int b){return a<b?a:b;}
inline signed mo1(int x,int y){return x+y>=mod?x+y-mod:x+y;}
inline signed mo2(int x,int y){return x<y?x-y+mod:x-y;}
inline signed ksm(int x,int y){
rr int ans=1;
for (;y;y>>=1,x=1ll*x*x%mod)
if (y&1) ans=1ll*ans*x%mod;
return ans;
}
namespace Theoretic{
int rev[N<<2],LAST,tt[N<<2];
inline void Pro(int n){
if (LAST==n) return; LAST=n;
for (rr int i=0;i<n;++i)
rev[i]=(rev[i>>1]>>1)|((i&1)?n>>1:0);
}
inline void NTT(int *f,int n,int op){
Pro(n);
for (rr int i=0;i<n;++i)
if (i<rev[i]) swap(f[i],f[rev[i]]);
rr int p=2,len=1;
for (rr int o=1;p<=n;++o){
rr int W=(op==1)?Gmi[o]:Imi[o];
for (rr int i=0;i<n;i+=p){
rr int t=1;
for (rr int j=i;j<i+len;++j){
rr int z=1ll*f[len+j]*t%mod;
f[len+j]=mo2(f[j],z),f[j]=mo1(f[j],z);
t=1ll*t*W%mod;
}
}
p<<=1,len<<=1;
}
}
inline void Cb(int *f,int *g,int n){
for (rr int i=0;i<n;++i) f[i]=1ll*f[i]*g[i]%mod;
}
inline void times(int *f,int *g,int len,int lim){
rr int n=1,invn; for (;n<lim;n<<=1);
for (rr int i=0;i<len;++i) tt[i]=g[i];
for (rr int i=len;i<n;++i) tt[i]=0;
NTT(f,n,1),NTT(tt,n,1),Cb(f,tt,n),NTT(f,n,-1);
for (rr int i=lim;i<n;++i) f[i]=0;
for (rr int i=0;i<n;++i) tt[i]=0;
invn=ksm(n,mod-2);
for (rr int i=0;i<lim;++i)
f[i]=1ll*f[i]*invn%mod;
}
}
inline void GmiImi(){
for (rr int i=0;i<31;++i) Gmi[i]=ksm(3,(mod-1)/(1<<i));
for (rr int i=0;i<31;++i) Imi[i]=ksm(inv3,(mod-1)/(1<<i));
}
inline void Prep(int lim){
for (rr int i=1;i<=n;++i) ff[i]=a[i]>=lim;
for (rr int i=1;i<=n;++i) gg[i]=b[i]>=lim;
Theoretic::times(ff,gg,n,n*2);
for (rr int i=1;i<=n;++i) ans[i]+=ff[i];
}
signed main(){
n=iut(),GmiImi();
for (rr int i=1;i<=n;++i) ++a[iut()];
for (rr int i=1;i<=n;++i) ++b[iut()];
for (rr int i=1;i<N;++i) if (a[i]) q.push(make_pair(a[i],0));
for (rr int i=1;i<N;++i) if (b[i]) q.push(make_pair(b[i],1));
while (!q.empty()){
rr int t=q.top().second; q.pop();
if ((++TOT[t])*TOT[t^1]>1e8) break;
}
if (!q.empty()) lim=q.top().first+1;
for (rr int i=1;i<=lim;++i) Prep(i);
for (rr int i=1;i<=n;++i) if (a[i]>lim) A[++A[0]]=i;
for (rr int i=1;i<=n;++i) if (b[i]>lim) B[++B[0]]=i;
for (rr int i=1;i<=A[0];++i)
for (rr int j=1;j<=B[0];++j)
ans[A[i]+B[j]]+=min(a[A[i]],b[B[j]])-lim;
for (rr int i=1;i<=2*n;++i)
if (Ans<ans[i]) Ans=ans[i];
return !printf("%d",Ans);
}
#NTT,DP#U138580 简单的打击的更多相关文章
- ZOJ 4257 MostPowerful(状压DP,简单)
题目大意:不超过10种气体,两两之间相互碰撞可以产生一定的能量,如a碰b,那么b气体就消失,自身不能碰自身,问最后所能得到的最大能量. 原代码链接:http://blog.csdn.net/accry ...
- 数位dp 的简单入门
时间紧张,就不讲那么详细了. 之前一直被深搜代码误解,以为数位dp 其实就是记忆化深搜...(虽说爆搜确实很舒服而且还好想) 但是后来发现数位dp 的标准格式其实是 预处理 + dp ...... 数 ...
- 斜率优化dp 的简单入门
不想写什么详细的讲解了...而且也觉得自己很难写过某大佬(大米饼),于是建议把他的 blog 先看一遍,然后自己加了几道题目以及解析...顺便建议看看算法竞赛(蓝皮书)的 0x5A 斜率优化(P294 ...
- PKU 1458 Common Subsequence(最长公共子序列,dp,简单)
题目 同:ZJU 1733,HDU 1159 #include <stdio.h> #include <string.h> #include <algorithm> ...
- DP的简单应用
Problem A:简单的图形覆盖 Time Limit:1000MS Memory Limit:65536KTotal Submit:201 Accepted:104 Description 有一 ...
- dp优化简单总结
1.二分优化 (使用二分查找优化查找效率) 典型例题:LIS dp[i]保存长度为 i 的上升子序列中最小的结尾,可以用二分查找优化到nlogn 2.数学优化 (通过数学结论减少状态数) 例题1:hd ...
- HDU 1024 Max Sum Plus Plus(DP的简单优化)
Problem Description Now I think you have got an AC in Ignatius.L's "Max Sum" problem. To b ...
- dp的简单递推笔记1
(1)转自rockZ的博文 UVa 10328 - Coin Toss (递推) 题意:给你一个硬币,抛掷n次,问出现连续至少k个正面向上的情况有多少种. 原题中问出现连续至少k个H的情况,很难下手. ...
- 2017-5-14 湘潭市赛 Similar Subsequence 分析+四维dp+一些简单优化
Similar Subsequence Accepted : Submit : Time Limit : MS Memory Limit : KB Similar Subsequence For gi ...
- 树形DP(简单题)(Y HDU4705)
题意:给出一个n个节点的树形图,统计{A,B,C}的数量,其中ABC分别是树上三个不同的节点,并且这三个节点不能被一条路径覆盖 分析:对于下图 进行dfs深搜统计,num[u]统计回溯到当前节点u,并 ...
随机推荐
- 学习go语言编程之错误处理
error接口 Golang中有一个关于错误处理的标准模式,即:error接口. type error interface { Error() string } 对于大多数函数,如果要返回错误,大致上 ...
- 【MySQL】数据库设计(一)三大范式
三大范式 1NF 第一范式 强调列的原子性,即列不可分 例如: 2NF 第二范式 前提是1NF,另外包含两个部分: 表必须具有一个主键: 没有包含在主键中的列必须完全依赖于主键,而不是只依赖主键的一部 ...
- Bind DNS Server的基础配置
1.访问https://192.168.3.254:10000 由于SSL证书是不安全的,我用的Firefox浏览器会阻止打开网页: 看到上述界面,先选择"高级", 然后再选择&q ...
- 第一百一十一篇:基本引用类型Date
好家伙,本篇为<JS高级程序设计>第五章的学习笔记 1.基本引用类型 引用值(或者对象)是某个特定引用类型的实例,在ECMAScript中,引用类型是把数据和功能组织到一起的结构,(像 ...
- 【Azure 应用服务】App Service下部署的应用报错 Out of Memory
问题描述 应用部署到App Service后,遇见了Out of Memory的错误. 报错信息:GetData Error:, Exception of type 'System.OutOfMem ...
- debian手册摘要
apt-get source 包名 # 获取源码dpkg --info deb包名 # 查看包信息apt-cache show 包名 # 包信息(含有Depends.Suggests.Section. ...
- 浅入kubernetes(3):namespace、node、pod
目前已经完成三篇关于 kubernetes 的文章: 在 Ubuntu 上安装 K8S教程 浅入kubernetes(1):Kubernetes 入门基础 浅入kubernetes(2):Kubern ...
- Codeforces Round 923 (Div. 3)(A~F)
目录 A B C D E F A #include <bits/stdc++.h> #define int long long #define rep(i,a,b) for(int i = ...
- Prompt进阶系列1:LangGPT(从编程语言反思LLM的结构化可复用提示设计框架)
Prompt进阶系列1:LangGPT(从编程语言反思LLM的结构化可复用提示设计框架) 大语言模型 (Large Language Models, LLMs) 在不同领域都表现出了优异的性能.然而, ...
- URLDNS链分析
一.概述 URLDNS 是ysoserial中利用链的一个名字,通常用于检测是否存在Java反序列化漏洞.该利用链具有如下特点: 不限制jdk版本,使用Java内置类,对第三方依赖没有要求 目标无回显 ...