首页 > 其他分享 >积性函数

积性函数

时间:2023-02-25 17:44:06浏览次数:31  
标签:Prime 函数 积性 times int 因子

积性函数

引入:

我们在线性筛质数的时候使用的方法是这样的

void GetPrime(int n){
    memset(isPrime, 1, sizeof(isPrime));
    isPrime[1] = 0;//1不是素数
    for(int i = 2; i <= n; i++){
        if(isPrime[i])
            Prime[++cnt] = i;
        for(int j = 1; j <= cnt && i*Prime[j] <= n; j++) {
            isPrime[i*Prime[j]] = 0;
            if(i % Prime[j] == 0)
                break; 
        }
    }
}

在这个程序里面我们保证一些东西:

1. \(i \mod Prime[j] == 0\) 时候 \(Prime[j]\) 一定是 \(i\) 的最小质因子

从小到大枚举 \(Prime[j]\) 这显而易见

2.$i \times Prime[j] $中 \(Prime[j]\) 一定是最小质因子

设 \(x=i\times Prime[j]\)

如果有更小的质因子 \(p\) ,那么一定满足 \(i \mod p=0\) 会在前面直接判断跳出循环

3. \(x\) 一定被筛一次并且被他的最小质因子筛

积性函数:

定义:一个数是积性函数当且仅当:

\[f(nm) = f(n)f(m)~~~~(\gcd(m, n) = 1) \]

而在积性函数中,经常使用到以下几个重要的积性函数(容易证明它们都是积性的):
    • \(\tau(n)\) 表示正整数 \(n\) 的正因子个数。
    • \(\sigma(n)\) 表示正整数 \(n\) 的正因子和。
    • \(\mu(n)\) 表示正整数 \(n\) 的 Mobius 函数值。
    • \(\varphi(n)\) 表示正整数 \(n\) 的欧拉函数值,即 [1, n] 中与 n 互质的数的个数。

积性函数一般搭配线性筛用

我在这里以 \(\tau(n)\) 为例子

\[f(nm) = f(n)f(m)~~~~(\gcd(m, n) = 1) \]

可以直接相乘

当\(i \mod Prime[j] == 0\)

我们设 \(x = i,y=prime[j]\)

那么一定有

\[x=k\times y^m \]

那么可以表示为

\[x\times y=k\times y^{m+1} \]

这样 \(k\) 和 \(y^{m+1}\) 一定互质,可以直接相乘

但假如 \(i = 16,Prime[j]=2\)

那么 \(k=1,y^{m+1}=32\)

我们并没有算出 \(32\) 的函数值,怎么办呢?

我们可以发现 \(y^{m+1}\) 的特性,他的因子一定是

\[y^0,y^1,y^2...y^{m+1} \]

函数值就直接等于 \(m+2\)

这样就有了以下代码

void pre(){
	a[1] = 1;
	for(int i = 2;i <= N; ++i){
		isP[i] = 1;
	}
	for(int i = 2;i <= N; ++i){
		if(isP[i] == 1){
			prime[++tot] = i;
			a[i] = 2;//如果是质数一定有两个因子
		}
		for(int j = 1;j <= tot && prime[j] * i <= N - 10; ++j){
			isP[prime[j] * i] = 0;
			if(i % prime[j] == 0){
				int add = 0,sum = 0,x = i,y(prime[j]);
				while(x % y == 0){
					x = x / y;
					add++;
				} 
				a[prime[j] * i] = a[x] * (2 + add);
				break;
			}
			a[prime[j] * i] = a[prime[j]] * a[i];
		}
	}
}

但时间复杂度变成了 \(O(n\log n)\) ,如何优化?

我们记录 \(add_i\) 表示 \(i\) 的最小质因子,就可以用空间换时间:

具体代码实现如下

void pre(){
	a[1] = 1;
	for(int i = 2;i <= N; ++i){
		isP[i] = 1;
	}
	for(int i = 2;i <= N; ++i){
		if(isP[i] == 1){
			prime[++tot] = i;
			a[i] = 2;//如果是质数一定有两个因子
            add[i] = 1;//注意
		}
		for(int j = 1;j <= tot && prime[j] * i <= N - 10; ++j){
			isP[prime[j] * i] = 0;
			if(i % prime[j] == 0){
				int x = i,y(prime[j]);
                add[i * prime[j]] = add[i] + 1;//注意
				a[prime[j] * i] = a[x/(add[i])] * (1 + add[i]);//注意
				break;
			}
			a[prime[j] * i] = a[prime[j]] * a[i];
            add[i * prime[j]] = 1;//注意
		}
	}
}

标签:Prime,函数,积性,times,int,因子
From: https://www.cnblogs.com/hfjh/p/17154880.html

相关文章

  • 去趋势函数 detrend
     matlab去趋势例程clcclearallcloseall%创建一个模拟数据集并计算其平均值。t=0:300;dailyFluct=gallery('normaldata',size(t),2);sdata=cumsum(dai......
  • C++函数名修饰规则
    C++函数名修饰规则这是啥函数的名字修饰(DecoratedName)就是编译器在编译期间创建的一个字符串。用来指明函数的定义或原型。修饰规则C++的修饰规则为“?+函数名+标......
  • golang中的close函数
    close函数是用于关闭通道的。官方解释(摘自close函数源代码注释):Theclosebuilt-infunctionclosesachannel,whichmustbeeitherbidirectionalorsend-only.Itsho......
  • 可变类型和不可变类型、闭包函数、装饰器+语法糖
    可变类型和不可变类型:  闭包函数:  装饰器+语法糖:   ......
  • map()函数应用
    title:map()函数应用author:杨晓东permalink:map()函数应用date:2021-10-0211:27:04categories:-投篮tags:-demomap()函数应用#正常函数一个参数d......
  • [keil] 将函数定义到RAM运行,和定义无初始化变量(软复位,变量不清空)
    keil链接文件​​一、将函数定义到RAM运行​​​​二、定义无初始化变量(软复位,变量不清空)​​先打开Keil工程配置,选择linker链接文件,取消自动生成,并编辑sct。如上图,定义......
  • avformat_seek_file函数介绍
    在做音视频数据分析的时候,经常会遇到这样的需求,每隔5分钟抽取一帧数据进行分析。在做播放器开发的时候,也会遇到这种情况,就是拖动进度条跳转到某个位置进行播放。如果直接用......
  • 构造函数和析构函数
    类内的构造函数:相当于初始化函数,名字和类名一致,可以在里面写入初始化语句类内的析构函数类的对象调用完所有成员函数,将跳出程序之前释放内存空间,名字是构造函数......
  • 友元函数/类
    在类中添加友元,相当于安插了一个卧底,可以访问类内元素,如下classBox{doublewidth;public:friendvoidprintWidth(Boxbox);friendclassBigBox;......
  • c语言:辗转相除求最大公约数 函数
    #include<stdio.h>//求最大公约数:辗转相除法:辗转相除法是求两个自然数的最大公约数的一种方法,也叫欧几里德算法。//319377:319%377=319377%319=58319%58=2958%29=0......