并不对劲的多项式求ln,exp
ln
解释
设\(g(x)=ln(f(x))\),两边同时求导,则有:\(g'(x)=ln'(f(x))*f'(x)=f^{-1}(x)*f'(x)\)(1)
因为\(f(x)\)是个多项式,所以设\(f(x)=\sum_{i=0}^{n}a_i*x^i\),则有\(f'(x)=\sum_{i=0}^{n-1}a_{i+1}*(i+1)*x^i\)
设\(h(x)=f^{-1}(x)*f'(x)\),对(1)式两边同时求积分,得:\(g(x)=\int h(x)\space dx\)
因为\(h(x)\)是个多项式,设\(h(x)=\sum_{i=0}^{n}b_i*x^i\)就有:\(\int h(x)\space dx=\sum_{i=1}^{n}\frac{b_{i-1}}{i}x^i\)
代码
#include<algorithm>
#include<cmath>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<ctime>
#include<iomanip>
#include<iostream>
#include<map>
#include<queue>
#include<set>
#include<stack>
#include<vector>
#define rep(i,x,y) for(register int i=(x);i<=(y);++i)
#define dwn(i,x,y) for(register int i=(x);i>=(y);--i)
#define maxn 800010
#define LL long long
#define mo(x) (x>=mod?x-mod:(x<0?x+mod:x))
using namespace std;
int read()
{
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)&&ch!='-')ch=getchar();
if(ch=='-')f=-1,ch=getchar();
while(isdigit(ch))x=(x<<1)+(x<<3)+ch-'0',ch=getchar();
return x*f;
}
void write(int x)
{
if(x==0){putchar('0');return;}
int f=0;char ch[20];
if(x<0)putchar('-'),x=-x;
while(x)ch[++f]=x%10+'0',x/=10;
while(f)putchar(ch[f--]);
return;
}
const LL mod=998244353;
int f[maxn],g[maxn],h[maxn],nowlen,nown,r[maxn],tmp[maxn],n;
int mul(int x,int y){int res=1;while(y){if(y&1)res=(LL)res*(LL)x%mod;x=(LL)x*(LL)x%mod,y>>=1;}return res;}
void dnt(int * u,int fh)
{
rep(i,0,nown-1)r[i]=(r[i>>1]>>1)|((i&1)<<(nowlen-1));
rep(i,0,nown-1)if(i<r[i])swap(u[i],u[r[i]]);
for(int i=1;i<nown;i<<=1)
{
int wn=mul(3,(mod-1)/(i<<1));
if(fh==-1)wn=mul(wn,mod-2);
for(int j=0;j<nown;j+=(i<<1))
{
int w=1;
rep(k,0,i-1){int x=u[j+k],y=(LL)w*(LL)u[i+j+k]%mod;u[j+k]=mo(x+y),u[i+j+k]=mo(x-y),w=(LL)w*(LL)wn%mod;}
}
}
if(fh==-1)
{
int inv=mul(nown,mod-2);
rep(i,0,nown-1)u[i]=(LL)u[i]*(LL)inv%mod;
}
}
void getny(int * u,int * v,int nlen)
{
v[0]=mul(u[0],mod-2);
for(int len=0,tmpn=1;tmpn<nlen;len++,tmpn<<=1)
{
nown=tmpn<<1,nowlen=len+1;
rep(i,0,nown-1)tmp[i]=u[i];
nown<<=1,nowlen++;
rep(i,(tmpn<<1),nown-1)tmp[i]=0;
dnt(tmp,1),dnt(v,1);
rep(i,0,nown-1)v[i]=mo(2ll-(LL)tmp[i]*(LL)v[i]%mod)*(LL)v[i]%mod;
dnt(v,-1);
rep(i,(tmpn<<1),nown-1)v[i]=0;
}
rep(i,nlen,nown)v[i]=0;
}
void getup(int * u,int * v,int nlen)
{
rep(i,1,nlen-1)v[i-1]=(LL)i*(LL)u[i]%mod;v[nlen-1]=0;
}
void getdx(int * u,int * v,int nlen)
{
rep(i,1,nlen-1)v[i]=(LL)u[i-1]*(LL)mul(i,mod-2)%mod;v[0]=0;
}
void getln(int * u,int nlen)
{
getup(u,h,nlen),getny(u,g,nlen);
for(nowlen=0,nown=1;nown<(nlen+nlen);nowlen++,nown<<=1);
dnt(h,1),dnt(g,1);
rep(i,0,nown-1)h[i]=(LL)h[i]*(LL)g[i]%mod;
rep(i,0,nown-1)g[i]=0;
dnt(h,-1);getdx(h,g,nlen);
rep(i,nlen,nown)g[i]=0;
}
int main()
{
n=read();
rep(i,0,n-1)f[i]=read();
getln(f,n);
rep(i,0,n-1)write(g[i]),putchar(' ');
return 0;
}
exp
解释
这个人(点这里)讲得很清楚
代码
#include<algorithm>
#include<cmath>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<ctime>
#include<iomanip>
#include<iostream>
#include<map>
#include<queue>
#include<set>
#include<stack>
#include<vector>
#define rep(i,x,y) for(register int i=(x);i<=(y);++i)
#define dwn(i,x,y) for(register int i=(x);i>=(y);--i)
#define maxn 800010
#define LL long long
#define mo(x) (x>=mod?x-mod:(x<0?x+mod:x))
using namespace std;
int read()
{
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)&&ch!='-')ch=getchar();
if(ch=='-')f=-1,ch=getchar();
while(isdigit(ch))x=(x<<1)+(x<<3)+ch-'0',ch=getchar();
return x*f;
}
void write(int x)
{
if(x==0){putchar('0');return;}
int f=0;char ch[20];
if(x<0)putchar('-'),x=-x;
while(x)ch[++f]=x%10+'0',x/=10;
while(f)putchar(ch[f--]);
return;
}
const LL mod=998244353;
int f[maxn],g[maxn],h[maxn],ans[maxn],nowlen,nown,r[maxn],tmp[maxn],tmp2[maxn],n;
int mul(int x,int y){int res=1;while(y){if(y&1)res=(LL)res*(LL)x%mod;x=(LL)x*(LL)x%mod,y>>=1;}return res;}
void dnt(int * u,int fh)
{
rep(i,0,nown-1)r[i]=(r[i>>1]>>1)|((i&1)<<(nowlen-1));
rep(i,0,nown-1)if(i<r[i])swap(u[i],u[r[i]]);
for(int i=1;i<nown;i<<=1)
{
int wn=mul(3,(mod-1)/(i<<1));
if(fh==-1)wn=mul(wn,mod-2);
for(int j=0;j<nown;j+=(i<<1))
{
int w=1;
rep(k,0,i-1){int x=u[j+k],y=(LL)w*(LL)u[i+j+k]%mod;u[j+k]=mo(x+y),u[i+j+k]=mo(x-y),w=(LL)w*(LL)wn%mod;}
}
}
if(fh==-1)
{
int inv=mul(nown,mod-2);
rep(i,0,nown-1)u[i]=(LL)u[i]*(LL)inv%mod;
}
}
void getny(int * u,int * v,int nlen)
{
v[0]=mul(u[0],mod-2);
for(int len=0,tmpn=1;tmpn<nlen;len++,tmpn<<=1)
{
nown=tmpn<<1,nowlen=len+1;
rep(i,0,nown-1)tmp[i]=u[i];
nown<<=1,nowlen++;
rep(i,(tmpn<<1),nown-1)tmp[i]=0;
dnt(tmp,1),dnt(v,1);
rep(i,0,nown-1)v[i]=mo(2ll-(LL)tmp[i]*(LL)v[i]%mod)*(LL)v[i]%mod;
dnt(v,-1);
rep(i,(tmpn<<1),nown-1)v[i]=0;
}
rep(i,0,nown)tmp[i]=0;
rep(i,nlen,nown)v[i]=0;
}
void getup(int * u,int * v,int nlen)
{
rep(i,1,nlen-1)v[i-1]=(LL)i*(LL)u[i]%mod;v[nlen-1]=0;
}
void getdx(int * u,int * v,int nlen)
{
rep(i,1,nlen-1)v[i]=(LL)u[i-1]*(LL)mul(i,mod-2)%mod;v[0]=0;
}
void getln(int * u,int * v,int nlen)
{
getup(u,h,nlen),getny(u,g,nlen);
for(nowlen=0,nown=1;nown<(nlen+nlen);nowlen++,nown<<=1);
dnt(h,1),dnt(g,1);
rep(i,0,nown-1)h[i]=(LL)h[i]*(LL)g[i]%mod;
rep(i,0,nown-1)g[i]=0;
dnt(h,-1);getdx(h,v,nlen);
rep(i,nlen,nown)v[i]=0;
}
void getexp(int * u,int * v,int nlen)
{
rep(i,0,(nlen<<2))v[i]=0;
v[0]=1;
for(int len=0,tmpn=1;tmpn<nlen;len++,tmpn<<=1)
{
rep(i,0,(tmpn<<1))tmp2[i]=0;
getln(v,tmp2,(tmpn<<1));
nown=(tmpn<<2),nowlen=len+2;
rep(i,(tmpn<<1),nown)tmp2[i]=0;
rep(i,0,(tmpn<<1)-1)tmp[i]=u[i];
rep(i,(tmpn<<1),nown)tmp[i]=0;
dnt(tmp2,1),dnt(v,1),dnt(tmp,1);
rep(i,0,nown-1)v[i]=(LL)v[i]*(LL)mo(mo(1ll-tmp2[i])+tmp[i])%mod;
dnt(v,-1);
rep(i,(tmpn<<1),nown)v[i]=0;
}
rep(i,nlen,nown)v[i]=0;
}
int main()
{
n=read();
rep(i,0,n-1)f[i]=read();
getexp(f,ans,n);
rep(i,0,n-1)write(ans[i]),putchar(' ');
return 0;
}
并不对劲的多项式求ln,exp的更多相关文章
- 多项式求ln,求exp,开方,快速幂 学习总结
按理说Po姐姐三月份来讲课的时候我就应该学了 但是当时觉得比较难加上自己比较懒,所以就QAQ了 现在不得不重新弄一遍了 首先说多项式求ln 设G(x)=lnF(x) 我们两边求导可以得到G'(x)=F ...
- [BZOJ3456]城市规划(生成函数+多项式求逆+多项式求ln)
城市规划 时间限制:40s 空间限制:256MB 题目描述 刚刚解决完电力网络的问题, 阿狸又被领导的任务给难住了. 刚才说过, 阿狸的国家有n个城市, 现在国家需要在某些城市对之间建立一 ...
- BZOJ 3456: 城市规划 [多项式求逆元 组合数学 | 生成函数 多项式求ln]
3456: 城市规划 题意:n个点组成的无向连通图个数 以前做过,今天复习一下 令\(f[n]\)为n个点的无向连通图个数 n个点的完全图个数为\(2^{\binom{n}{2}}\) 和Bell数的 ...
- 指数型生成函数 及 多项式求ln
指数型生成函数 我们知道普通型生成函数解决的是组合问题,而指数型生成函数解决的是排列问题 对于数列\(\{a_n\}\),我们定义其指数型生成函数为 \[G(x) = a_0 + a_1x + a_2 ...
- BZOJ3456 城市规划 【多项式求ln】
题目链接 BZOJ3456 题解 真是一道经典好题,至此已经写了分治\(NTT\),多项式求逆,多项式求\(ln\)三种写法 我们发现我们要求的是大小为\(n\)无向联通图的数量 而\(n\)个点的无 ...
- 【BZOJ】3456: 城市规划(多项式求ln)
题解 在我写过分治NTT,多项式求逆之后 我又一次写了多项式求ln 我们定义一个数列的指数型生成函数为 \(\sum_{i = 0}^{n} \frac{A_{i}}{i!} x^{i}\) 然后这个 ...
- 【BZOJ3456】轩辕朗的城市规划 EGF+多项式求ln
我们构造$f(i)$和$g(i)$. 其中$f(x)$表示由$x$个节点构成的无向简单连通图的个数. $g(x)$表示有$x$个节点构成的无向简单图(不要求连通)的个数. 显然,由$x$个节点构成的无 ...
- 洛谷5月月赛T30212 玩游戏 【分治NTT + 多项式求ln】
题目链接 洛谷T30212 题解 式子很容易推出来,二项式定理展开后对于\(k\)的答案即可化简为如下: \[k!(\sum\limits_{i = 0}^{k} \frac{\sum\limits_ ...
- bzoj 3456 城市规划——分治FFT / 多项式求逆 / 多项式求ln
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3456 分治FFT: 设 dp[ i ] 表示 i 个点时连通的方案数. 考虑算补集:连通的方 ...
随机推荐
- AC日记——砝码称重 洛谷 P2347
题目描述 设有1g.2g.3g.5g.10g.20g的砝码各若干枚(其总重<=1000), 输入输出格式 输入格式: 输入方式:a1 a2 a3 a4 a5 a6 (表示1g砝码有a1个,2g砝 ...
- python在webservice接口测试中的应用
接口测试第二波,webservice接口来咯,欢迎各位小伙伴吐槽~本次拿免费的互联网国内手机号码归属地查询WEB服务webservice接口做例子,当然有很多免费webservice接口可以供大家使用 ...
- 深究Spring中Bean的生命周期
前言 这其实是一道面试题,是我在面试百度的时候被问到的,当时没有答出来(因为自己真的很菜),后来在网上寻找答案,看到也是一头雾水,直到看到了<Spring in action>这本书,书上 ...
- Ionic 学习笔记
本文为原创,转载请注明出处: cnzt 文章:cnzt-p http://www.cnblogs.com/zt-blog/p/7831153.html 注: 本篇学习笔记基于Ionic 3 ...
- iOS中创建自定义的圆角按钮
iOS中很多时候都需要用到指定风格的圆角按钮,尽管UIButton提供了一个方式创建圆角按钮: + (id)buttonWithType:(UIButtonType)buttonType;//指定bu ...
- Go -- PipleLine
1.pipeline的产生 从一个现象说起,有一家咖啡吧生意特别好,每天来的客人络绎不绝,客人A来到柜台,客人B紧随其后,客人C排在客人B后面,客人D排在客人C后面,客人E排在客人D后面,一直排到店面 ...
- Go -- type 和断言 interface{}转换
摘要 类型转换在程序设计中都是不可避免的问题.当然有一些语言将这个过程给模糊了,大多数时候开发者并不需要去关 注这方面的问题.但是golang中的类型匹配是很严格的,不同的类型之间通常需要手动转换,编 ...
- 如何通过SQL注入获取服务器本地文件
写在前面的话 SQL注入可以称得上是最臭名昭著的安全漏洞了,而SQL注入漏洞也已经给整个网络世界造成了巨大的破坏.针对SQL漏洞,研究人员也已经开发出了多种不同的利用技术来实施攻击,包括非法访问存储在 ...
- 源代码方式向openssl中加入新算法完整具体步骤(演示样例:摘要算法SM3)【非engine方式】
openssl简单介绍 openssl是一个功能丰富且自包括的开源安全工具箱.它提供的主要功能有:SSL协议实现(包括SSLv2.SSLv3和TLSv1).大量软算法(对称/非对称/摘要).大数运算. ...
- groovy入门 第05章 基本输入输出
基本输入输出 5.1基本输出 print XXX //同一行输出 println XXX //换行输出 输出字符串: def message ="My name is Michael& ...