@雅礼集训01/06 - T3@ math
@description@
给出 n, m, x,你需要求出下列式子的值:
\]
其中 ki 为正整数。由于答案非常大,你只需要输出答案(保证不为 0)的正负(如果是负数输出负号,否则输出正号)和从左往右第一个非 0 数位上的数字即可。
input
第一行一个整数 T 表示数据组数。
对于每组数据,每行有两个整数 m,n 和一个两位小数 x。
对于 100%的数据,T ≤ 10,n ≤ 10^9,m ≤ 30。
output
输出共 T 行,每行两个字符表示答案。
sample input
2
3 5 0.01
3 6 0.02
sample output
+2
+4
@solution@
其实题目要你求解的说白了就是卷积。即对于多项式 \(f(p) = \sum_{i=1}^{+\infty}\sin(i*x)p^i\),求 \(f^m(p)\) 中第 n 项的系数为多少。然而 n 最大可以为 10^9,FFT 再怎么优化也过不了。
我们发现对于这个卷积,我们想要得到的其实只有第 n 项的系数,其他项的系数并不重要,而 FFT 必然要求出所有的系数,所以时间复杂度肯定降不下来。我们不得不换一种思路。
如果是一般多项式,FFT 就是时间复杂度的下限。但是我们的系数是三角函数,也就是说,我们要利用三角函数的一些性质。
数据范围 n <= 10^9,且对象是三角函数。如果熟悉的话,很容易想到三角函数的递推式:
\]
边界为已知量 \(\sin 0, \sin x\)。 \(\cos x\) 是一个我们可以预先知道的常数,这样子就可以和矩阵乘法扯上关系了。
这个式子是怎么来的呢?可以参考下面的推导过程(但是如果你很熟悉了可以直接跳过)。
首先是三角函数的和差公式:
\]
这个式子中的 \(\cos (k-1)x\) 是我们想要消去的,因此再对它使用一次和差公式。
\]
又出现了 \(\cos (k-2)x\)。但是,注意到其实可以把 \(\sin x\cos(k-2)x\) 当作一个整体,而这个整体出现在 \(\sin (k-1)x\) 的和差公式中。
因此,我们把 \(\sin (k-1)x = \sin x\cos (k-2)x+\cos x\sin (k-2)x\) 变形代入上式。
\]
恒等变形,就可以得到我们的结果:
\]
好的我们推完式子再回到我们刚刚的思路。
考虑几种特殊情况吧。
当 m = 1 时,就是求 \(\sin nx\), 直接矩阵加速即可(当然最直接的还是暴力算)。
当 m = 2 时,令 \(g[i] = \sum_{j=1}^{i-1}(\sin(j*x)*\sin((i-j)*x))\),我们相当于是求 \(g[n]\)。
我们尝试建立递推关系。代入三角函数的递推式得到(注意边界情况的存在):
=\sin x*\sin((i-1)*x)+2*\cos x*g[i-1]-g[i-2]\]
其中 \(\sin((i-1)*x)\) 虽然不是常数,但是也可以通过矩阵乘法得到。
更一般地,令 \(dp[i][j]\) 表示 j 个多项式卷积第 i 项的系数,我们可以得到如下的递推关系:
dp[i][j] = \sin x & i = 1 且 j = 1\\
dp[i][j] = 0 & otherwise
\end{cases}\]
然后就可以矩阵加速了。
@accepted code@
不知道为什么,求解非零位时必须按照标程写才能过。
如果 p 一直作除法改成 ans 一直作乘法就过不了。
是因为后者浮点误差太大了吗?如果有知道的拜托请评论在下面告诉我好吗 QAQ。
#include<cstdio>
#include<cmath>
const int MAXN = 30;
struct matrix{
int r, c;
double m[MAXN*2 + 5][MAXN*2 + 5];
}M, B;
matrix operator * (matrix A, matrix B) {
matrix C; C.r = A.r, C.c = B.c;
for(int i=0;i<C.r;i++)
for(int j=0;j<C.c;j++) {
C.m[i][j] = 0;
for(int k=0;k<A.c;k++)
C.m[i][j] += A.m[i][k]*B.m[k][j];
}
return C;
}
/*
void Print(matrix M) {
puts("");
for(int i=0;i<M.r;i++) {
for(int j=0;j<M.c;j++)
printf("%lf ", M.m[i][j]);
puts("");
}
puts("");
}
*/
matrix qpow(matrix A, int p) {
matrix ret; ret.r = A.r, ret.c = A.c;
for(int i=0;i<ret.r;i++)
for(int j=0;j<ret.c;j++)
ret.m[i][j] = (i == j);
while( p ) {
if( p & 1 ) ret = ret * A;
A = A * A;
p >>= 1;
}
return ret;
}
void solve() {
int m, n; double x;
scanf("%d%d%lf", &m, &n, &x);
B.r = 2*m, B.c = 1, B.m[0][0] = 1;
for(int i=0;i<2*m;i++)
B.m[i][0] = 0;
B.m[m-1][0] = sin(x);
M.r = M.c = 2*m;
for(int i=0;i<2*m;i++)
for(int j=0;j<2*m;j++)
M.m[i][j] = 0;
for(int i=0;i<m;i++) {
M.m[i][i] = 2*cos(x);
M.m[i][m + i] = -1;
M.m[m + i][i] = 1;
}
for(int i=2;i<=m;i++)
M.m[i-2][i-1] = sin(x);
//Print(M); Print(B);
//Print(qpow(M, n-1)*B);
double ans = (qpow(M, n-1)*B).m[0][0];
putchar(ans > 0 ? '+' : '-');
ans = fabs(ans);
if( ans > 1 ) {
while( ans >= 10 ) ans /= 10;
printf("%d\n", int(ans));
}
else {
double p = 0.1;
while( p > ans ) p /= 10;
printf("%d\n", int(ans/p));
}
}
int main() {
int T; scanf("%d", &T);
for(int i=1;i<=T;i++) solve();
}
@details@
不得不说这道题还真的挺容易让人走错方向来着。
首先这是一个卷积形式,一开始肯定是思考能不能用 FFT(特别是像我一样最近才入多项式这个坑什么都想先来 FFT 一下)。
然后 m 个正整数的和为 n,又是一个组合数学的经典问题。这又是一个大坑。
然后 sin(x),一样是因为最近学了多项式,搞得我都想泰勒展开了……
当然如果你是神犇肯定不会像我一样去想上面那些错误思路而是一眼就秒出了这道题的正解。
@雅礼集训01/06 - T3@ math的更多相关文章
- @雅礼集训01/13 - T1@ union
目录 @description@ @solution@ @part - 1@ @part - 2@ @part - 3@ @accepted code@ @details@ @description@ ...
- @雅礼集训01/10 - T1@ matrix
目录 @description@ @solution@ @accepted code@ @details@ @description@ 给定一个矩阵.求它的所有子矩阵中本质不同的行的个数之和. inp ...
- [LOJ 6030]「雅礼集训 2017 Day1」矩阵
[LOJ 6030] 「雅礼集训 2017 Day1」矩阵 题意 给定一个 \(n\times n\) 的 01 矩阵, 每次操作可以将一行转置后赋值给某一列, 问最少几次操作能让矩阵全为 1. 无解 ...
- 雅礼集训1-9day爆零记
雅礼集训1-9day爆零记 先膜一下虐爆我的JEFF巨佬 Day0 我也不知道我要去干嘛,就不想搞文化科 (文化太辣鸡了.jpg) 听李总说可以去看(羡慕)各路大佬谈笑风声,我就报一个名吧,没想到还真 ...
- LOJ_6045_「雅礼集训 2017 Day8」价 _最小割
LOJ_6045_「雅礼集训 2017 Day8」价 _最小割 描述: 有$n$种减肥药,$n$种药材,每种减肥药有一些对应的药材和一个收益. 假设选择吃下$K$种减肥药,那么需要这$K$种减肥药包含 ...
- 雅礼集训【Day6-1】字符串
雅礼集训[Day6-1]字符串 假设我们有串\(a\),我们设\(a'\)为\(a\)翻转后按为取反过后的串. 我们只考虑前一半的,长为\(m\)的串.如果前半截匹配了\(a\)或者\(a'\),则\ ...
- 「雅礼集训 2017 Day7」事情的相似度
「雅礼集训 2017 Day7」事情的相似度 题目链接 我们先将字符串建后缀自动机.然后对于两个前缀\([1,i]\),\([1,j]\),他们的最长公共后缀长度就是他们在\(fail\)树上对应节点 ...
- 「雅礼集训 2017 Day2」解题报告
「雅礼集训 2017 Day2」水箱 我怎么知道这种题目都能构造树形结构. 根据高度构造一棵树,在树上倍增找到最大的小于约束条件高度的隔板,开一个 \(vector\) 记录一下,然后对于每个 \(v ...
- 「雅礼集训 2017 Day1」 解题报告
「雅礼集训 2017 Day1」市场 挺神仙的一题.涉及区间加.区间除.区间最小值和区间和.虽然标算就是暴力,但是复杂度是有保证的. 我们知道如果线段树上的一个结点,\(max=min\) 或者 \( ...
随机推荐
- DTD约束与schema约束的不同
本篇笔记了解------Schema约束的语法 ------可以参考W3school之Schema教程. Schema:是基于 XML 的 DTD 替代者,用于描述XML文档结构.支持XML 命名空间 ...
- docker-4-Dockerfile配置文件详解
Dockerfile简单一点就是描述你这个镜像安装了哪些软件包,有哪些操作,创建了什么东西.有些人喜欢用 docker commit 命令去打包镜像,这样是不好的,首先commit出来的镜像比你使 ...
- 出现$(#form).validate is not a function的问题
最近为项目写cms系统,在新增/编辑文章的页面,一些input诸如文章题目,作者等等需要验证是否已经填写,于是使用jquery.validate.js来做这个工作,自己写了个验证的validate.j ...
- LINUX配置文件介绍
每个 Linux 程序都是一个可执行文件,它含有操作码列表,CPU 将执行这些操作码来完成特定的操作.例如,ls 命令是由 /bin/ls 文件提供的,该文件含有机器指令的列表,在屏幕上显示当前目录中 ...
- linux性能监视工具sar
sar是一个优秀的一般性能监视工具,它可以输出Linux所完成的几乎所有工作的数据.sar命令在sysetat rpm中提供.示例中使用sysstat版本5.0.5,这是稳定的最新版本之一.关于版本和 ...
- leetcode 31-40 easy
38.Count and Say The count-and-say sequence is the sequence of integers with the first five terms as ...
- 怎样判断一个exe可执行程序(dll文件)是32位的还是64位的
看到一个比较简单粗暴的方式,做个记录. 直接用记事本或者notepad++(文本编辑软件都可)打开exe文件(dll文件), 会有很多乱码,接下来只需要在第二段中找到PE两个字母,在其后的不远出会出现 ...
- 人不能同时在两个地方做猪(Scrum Team)
在一个神奇的国度里生活着许多动物, 其中有猪, 鸡, 和鹦鹉. 它们每天搞头脑风暴, 琢磨如何创业, 最后鹦鹉提议它们合伙开一个早餐店: 具体分工如下: 猪: 提供猪肉, 做熏猪肉 (bacon) 鸡 ...
- nested exception is org.hibernate.MappingException解决方案
1.可能是因为映射文件( Order.hbm.xm)配置的class路径出错 <hibernate-mapping> <class name="com.web.bean.O ...
- DirectX11笔记(二)--Direct3D初始化1之基本概念
原文:DirectX11笔记(二)--Direct3D初始化1之基本概念 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/u010333737/art ...