[51nod 1847]奇怪的数学题
【 51nod 1847 】奇怪的数学题
题目
点这里看题目。
分析
是挺奇怪的......
以下定义质数集合为\(P\),\(p_i\)为第\(i\)个质数。
定义\(mp(x)\)为\(x\)的最小质因子,则可以得到:
\]
这个比较显然。然后可以娴熟地变换式子得到:
\]
后面的求\(\varphi\)的前缀和的部分可以用杜教筛等筛法高速求出。这里推荐杜教筛,因为它可以方便地记忆化。
并且可以发现,\(\varphi\)的前缀和的上限实际上可以整除分块。所以,如果可以告诉求出\(\left(\frac d{mp(d)}\right)^k\),我们就可以整除分块了。这也是杜教筛比较占优势的原因——一次杜教筛就可以筛出所有需要的值,并且存下来。
问题变成了如何解决前一部分的求和。考虑使用 min_25。
min_25 首先需要算出在质数位置的贡献,不难发现这些位置的贡献为\(1^k=1\)。因此只需要筛出质数数量即可。
其它的位置需要筛。首先有常见操作:
设\(f(x)=x^k\),\(g(a,b)\)为前\(a\)个数进行了\(b\)轮埃筛之后的\(f\)的总贡献。
转移略。这是常见操作。
不过仔细想想,我们会发现转移出现的\(g(\lfloor\frac a{p_b}\rfloor, b-1)-g(p_{b-1},b-1)\)实际上就是 " 最小质因数为\(p_b\)的数的贡献 " (最小质因数少乘上一次)。因此我们实际操作的时候就不用写 min_25 的第二步,而是直接每轮累加起来,就可以得到合数的贡献。
而质数的贡献已经知道是质数个数了。所以对于每次询问,我们可以直接将合数和质数的贡献加起来回答。
但是,\(g(a,0)=\sum_{i=2}^a i^k\),这个该怎么计算呢?
可以用第二类斯特林数来处理:
\sum_{i=1}^n i^k
&=\sum_{i=1}^n \sum_{j=1}^k {k\brace j} i^{\underline j}\\
&=\sum_{j=1}^k {k\brace j}\sum_{i=1}^n i^{\underline j}\\
&=\sum_{j=1}^k{k\brace j}\frac{(n+1)^{\underline{j+1}}}{j+1}
\end{aligned}\]
其中一步推导用到了 " 离散微积分 " 的东西,我不会了,可以参考[第二类斯特林数]自然数幂求和。
需要注意的是,由于这道题取模方法是自然溢出,因此不能求逆元。但由于求幂和的时候结果一定是整数,所以在\((n+1)^{\underline{j+1}}\)里面一定有一个\(j+1\)的倍数。我们可以先用余数把它找出来,除掉\(j+1\)之后再将剩下的乘起来即可。
代码
#include <map>
#include <cmath>
#include <cstdio>
using namespace std;
typedef long long LL;
typedef unsigned int ui;
const int MAXN = 1e5 + 5, MAXK = 55;
template<typename _T>
void read( _T &x )
{
x = 0;char s = getchar();int f = 1;
while( s > '9' || s < '0' ){if( s == '-' ) f = -1; s = getchar();}
while( s >= '0' && s <= '9' ){x = ( x << 3 ) + ( x << 1 ) + ( s - '0' ), s = getchar();}
x *= f;
}
template<typename _T>
void write( _T x )
{
if( x < 0 ){ putchar( '-' ); x = ( ~ x ) + 1; }
if( 9 < x ){ write( x / 10 ); }
putchar( x % 10 + '0' );
}
map<int, ui> mp;
ui G[MAXN << 1], g1[MAXN << 1], gk[MAXN << 1];
ui ps[MAXN], pks[MAXN], pk[MAXN];
ui S[MAXK][MAXK];
int val[MAXN << 1];
int id1[MAXN], id2[MAXN];
int phi[MAXN], prime[MAXN], pn;
int N, K, s, cnt;
bool isPrime[MAXN];
int& ID( const int x ) { return x <= s ? id1[x] : id2[N / x]; }
ui qkpow( ui base, int indx )
{
ui ret = 1;
while( indx )
{
if( indx & 1 ) ret *= base;
base *= base, indx >>= 1;
}
return ret;
}
void EulerSieve( const int siz )
{
phi[1] = 1, isPrime[1] = true;
for( int i = 2 ; i <= siz ; i ++ )
{
if( ! isPrime[i] ) prime[++ pn] = i, phi[i] = i - 1;
for( int j = 1 ; j <= pn && 1ll * i * prime[j] <= siz ; j ++ )
{
isPrime[i * prime[j]] = true;
if( ! ( i % prime[j] ) ) { phi[i * prime[j]] = phi[i] * prime[j]; break; }
phi[i * prime[j]] = phi[i] * ( prime[j] - 1 );
}
}
for( int i = 1 ; i <= pn ; i ++ ) pks[i] = pks[i - 1] + ( pk[i] = qkpow( prime[i], K ) );
for( int i = 1 ; i <= siz ; i ++ ) ps[i] = ps[i - 1] + phi[i];
}
ui getS( const int a )
{
ui ret = 0, t;
for( int i = 1, tmp ; i <= K ; i ++ )
{
tmp = ( a + 1 ) % ( i + 1 ), t = S[K][i];
for( int j = 1 ; j <= tmp ; j ++ ) t *= ( a - j + 2 );
t *= ( a - tmp + 1 ) / ( i + 1 );
for( int j = tmp + 2 ; j <= i + 1 ; j ++ ) t *= ( a - j + 2 );
ret += t;
}
return ret;
}
ui SPhi( const int n )
{
if( n <= s ) return ps[n];
if( mp[n] ) return mp[n];
ui ret;
if( n & 1 ) ret = ( ui ) ( n + 1 ) / 2 * n;
else ret = ( ui ) n / 2 * ( n + 1 );
for( int l = 2, r ; l <= n ; l = r + 1 )
{
r = n / ( n / l );
ret -= ( r - l + 1 ) * SPhi( n / l );
}
return mp[n] = ret;
}
int main()
{
read( N ), read( K );
s = sqrt( N );
EulerSieve( s );
for( int i = 1 ; i <= K ; i ++ ) S[i][i] = 1, S[i][0] = 0;
for( int i = 2 ; i <= K ; i ++ )
for( int j = 1 ; j <= K ; j ++ )
S[i][j] = S[i - 1][j - 1] + ( ui ) j * S[i - 1][j];
for( int l = 1, r, v ; l <= N ; l = r + 1 )
{
r = N / ( v = N / l ), val[++ cnt] = v;
g1[ID( v ) = cnt] = v - 1, gk[cnt] = getS( v ) - 1;
}
for( int j = 1, k ; j <= pn ; j ++ )
for( int i = 1 ; i <= cnt && 1ll * prime[j] * prime[j] <= val[i] ; i ++ )
{
k = ID( val[i] / prime[j] );
g1[i] -= g1[k] - ( j - 1 );
gk[i] -= ( ui ) pk[j] * ( gk[k] - pks[j - 1] );
G[i] += gk[k] - pks[j - 1];
}
ui ans = 0, pre = 0, nxt;
for( int l = 2, r, v ; l <= N ; l = r + 1 )
{
r = N / ( v = N / l );
nxt = G[ID( l )] + g1[ID( l )];
ans += ( ui ) ( 2u * SPhi( v ) - 1 ) * ( nxt - pre ), pre = nxt;
}
write( ans ), putchar( '\n' );
return 0;
}
[51nod 1847]奇怪的数学题的更多相关文章
- 【51NOD 1847】奇怪的数学题(莫比乌斯反演,杜教筛,min_25筛,第二类斯特林数)
[51NOD 1847]奇怪的数学题(莫比乌斯反演,杜教筛,min_25筛,第二类斯特林数) 题面 51NOD \[\sum_{i=1}^n\sum_{j=1}^nsgcd(i,j)^k\] 其中\( ...
- 【51nod 1847】奇怪的数学题
题目描述 给出 N,K ,请计算下面这个式子: \(∑_{i=1}^N∑_{j=1}^Nsgcd(i,j)^k\) 其中,sgcd(i, j)表示(i, j)的所有公约数中第二大的,特殊地,如果gcd ...
- 【51nod 1874】 奇怪的数学题
题目 求 \[\sum_{i=1}^n\sum_{j=1}^nsgcd(i,j)^k\] 首先这个次大公约数显然就是\(gcd\)除一下最小质因子了 于是 \[\sum_{i=1}^n\sum_{j= ...
- 51nod 1965 奇怪的式子——min_25筛
题目:http://www.51nod.com/Challenge/Problem.html#!#problemId=1965 考虑 \( \prod_{i=1}^{n}\sigma_0^i \) \ ...
- [51nod1847]奇怪的数学题
description 51nod 求\[\sum_{i=1}^{n}\sum_{j=1}^{n}sgcd(i,j)^k\]其中\(sgcd(i,j)\)表示\(i,j\)的次大公约数,如果\(gcd ...
- 51nod 1965 奇怪的式子 —— min_25筛
题目:http://www.51nod.com/Challenge/Problem.html#!#problemId=1965 推式子就同这里:https://www.cnblogs.com/yoyo ...
- 【51NOD1847】奇怪的数学题 min_25筛
题目描述 记\(sgcd(i,j)\)为\(i,j\)的次大公约数. 给你\(n\),求 \[ \sum_{i=1}^n\sum_{j=1}^n{sgcd(i,j)}^k \] 对\(2^{32}\) ...
- 【51nod1847】 奇怪的数学题
就当我是 A 了此题吧... 首先预备知识有点多(因为题目要处理的东西都挺毒瘤): 杜教筛运用(当然你可以用其他筛?) 第二类斯特林数相关定理 下降阶乘幂相关定理 min25 筛运用 好了可以关掉本题 ...
- 【51Nod1847】奇怪的数学题
记\(f(x)=\)\(x\)的次大因数,那么\(sgcd(i,j)=f(gcd(i,j))\). 下面来推式子: \[ \begin{aligned} \sum_{i=1}^n\sum_{j=1 ...
随机推荐
- Hbase2.0-源码阅读环境
最近准备开始研究Hbase源码,因为第一次研究源码,所以做片笔记,踩坑踩的很耗时. 1.我用的IDE是IDEA,本地window需要配置JDK,MAVEN,HADOOP环境 2.上GitHub下载Hb ...
- jquery 根据 option 的 text 定位选中 option
$('#test option[text="b"]').attr("selected",true); 上面的方法在 jquery 低于 1.4.2 的版本(含) ...
- 一文读懂Java注解
什么是注解 Java官方文档上说,注解是元数据的一种形式,它提供不属于程序一部分的数据,注解对被注解的代码没有直接的影响. 准确上说,注解只不过是一种特殊的注释而已,如果没有解析它的代码,它可能连注释 ...
- Poj1753 翻转棋子
这个题就是用枚举举遍所有情况,然后一个一个深搜看看是不是符合条件,符合条件直接退出,不符合则继续, 由于表格只有16个所以可以得知最多的步数只能是16,所以可以根据步数从0到16依次枚举, 第一个符合 ...
- 杨辉三角(hdu2032)——有待完善
思考:杨辉三角形 #include<stdio.h> #include<cstring> int main() { int n; char d; ][] = {}; while ...
- JavaScript实现单向链表结构
参考资料 一.什么是链表结构? 1.1.简介 链表和数组一样, 可以用于存储一系列的元素, 但是链表和数组的实现机制完全不同,链表中的元素在内存不是连续的空间,链表的每个元素由一个存储元素本身(数据) ...
- TCP实现连接传输案例
使用的类 ServerSocket 服务器端 构造方法:ServerSocket(端口号); 接收端使用 方法:accept(); 返回一个 Socket对象 getI ...
- 永久关闭windows更新步骤
在搜索“web和windows”框中输入“服务” 在搜索结果中点击第一个,那个图标像齿轮的那个!如下图. 在打开的“服务”窗口中,我们找到windows update 找到”windows upd ...
- css变量的使用
css变量的使用 1.介绍:我们也可以在css中定义变量,和less.sass一样,通过--来定义变量 div { /* 开始定义变量 */ --color: red; /* 通过var()函数来使用 ...
- 解码问题--leetcode:91 (2019商汤笔试)
题目:有一种将字母编码成数字的方式:'a'->1, 'b->2', ... , 'z->26'. 现在给一串数字,给出有多少种可能的译码结果. 想法: 该题就是动态规划问题,建议在写 ...