Min_25篩

来源:https://www.cnblogs.com/wxyww/archive/2019/12/25/Min_25.html
-Advertisement-
Play Games

簡介 Min_25篩~~據說~~可以在$O(\frac{n^{\frac{3}{4}}}{logn})$處理出含有以下性質的函數f的首碼和: 1.$f(ab)=f(a)f(b)$,即f是一個積性函數 2.$f(p^k)$可以快速計算。 PS:本文沒有關於複雜度的證明。。。 預處理 首先要預處理兩個東 ...


簡介

Min_25篩據說可以在\(O(\frac{n^{\frac{3}{4}}}{logn})\)處理出含有以下性質的函數f的首碼和:
1.\(f(ab)=f(a)f(b)\),即f是一個積性函數
2.\(f(p^k)\)可以快速計算。

PS:本文沒有關於複雜度的證明。。。

預處理

首先要預處理兩個東西,一個是\(\sqrt{n}\)(n為詢問的值域)內的質數。直接線性篩就好了。用\(pri[i]\)表示第i個質數。設共有\(m\)個質數

另一個是\(g(n,j)\),表示所有\(x\in[1,n]\)中滿足x最小質因數大於\(pri[j]\)或者x是質數的\(f(x)\)之和。

這樣一來,\(g(n,m)\)表示的就是\([1,n]\)中所有質數的f值之和。這個東西後面會用到。

那麼來看一下這個\(g\)值該如何求。

顯然,如果\(pri_j^2>n\)那麼\(g(n,j)=g(n,j-1)\)。因為這時的\(g(n,j-1)\)已經只表示\([1,n]\)中所有質數的f之和。\(g(n,j)\)並不會比\(g(n,j-1)\)多刪除掉任何東西。

如果\(pri_j^2\le n\)呢?我們可以理解為埃氏篩法的過程,\(g(n,j)\)\(g(n,j-1)\)的差別就是篩掉了\(pri_j\)的倍數。那麼就好像可以轉移了。問題就在於如何計算出所有\(pri_j\)的倍數所產生的貢獻。前面說到這是一個積性函數,所以我們將這些要刪除的數全都提出來一個\(pri_j\),那麼剩下的就是\([1,\frac{n}{pri_j}]\)了。因為需要\(pri_j\)是這些數字的最小質因數,所以實際上區間\([1,pri_j-1]\)內的數字是不可以的,所以要刪除的區間實際上是\([pri_j,\frac{n}{pri_j}]\)所以要刪除的數字就是\(f(pri_j)[g(\frac{n}{pri_j},j-1)-g(pri_j-1,j-1)]\)。也就是說\(g(n,j)=g(n,j-1)-f(pri_j)[g(\frac{n}{pri_j},j-1)-g(pri_j-1,j-1)]\)

因為最終我們需要的只有\(g(\lfloor\frac{n}{i}\rfloor,m),1\in [1,n]\)。所以空間只需要開一維,每次處理複雜度是\(O(m)\)的(實際上並不到),類似於整除分塊,我們知道\(\frac{n}{i}\)只有\(\sqrt{n}\)級別種取值。複雜度據說是\(O(\frac{n^{\frac{3}{4}}}{logn})\)

計算答案

上面的東西預處理完了,那麼有什麼用呢??

我們再定義一個函數\(S(n,j)\)表示\(x\in[1,n]\)中滿足\(x\)的最小質因數大於等於\(pri_j\)\(f(x)\)之和。

最終我們要求的答案就是\(S(n,1)+f(1)\)

上面說到,\(g(n,m)(pri_m^2>n)\)可以表示\([1,n]\)中所有質數的\(f\)值之和。

所以我們將\(S(n,j)\)分為質數和合數兩塊來處理。

質數的一塊顯然就是\(g(n,m)-\sum\limits_{k=1}^{j-1}f(pri_k)\)。為什麼要減掉後面這一塊??因為小於\(pri_j\)的質數不包含在\(S(n,j)\)裡面呀~

然後考慮合數的一塊該如何求,我們枚舉一下這些合數的最小質因數\(k\in[pri_j,pri_m]\)\(k\)的指數\(e\)。於上方求g的方法類似的,我們可以提出來一個\(pri_k^e\),那麼剩下的就是\([1,\frac{n}{pri_k^e}]\),他們的f之和就是\(S(\frac{n}{pri_k^e},k)\)。發現這樣無法轉移,那麼我們只好從\(S(\frac{n}{pri_k^e},k+1)\)轉移過來,但是這樣\(f(pri_k^{e+1})\)就沒被計算,單獨加上就好了。

綜上所述,

\[S(n,j)=g(n,m)-\sum\limits_{k=1}^{j-1}f(pri_k)+\sum\limits_{k=j}^m\sum\limits_{e=1}^{pri_k^{e+1}\le n}(f(pri_k^{e+1})+f(pri_k^e)S(\frac{n}{pri_k^e},k+1)\]

然後遞歸計算即可。這裡的複雜度據說也是\(O(\frac{n^{\frac{3}{4}}}{logn})\)

經典例題

loj6053簡單的函數

定義函數\(f(x)\)滿足以下性質,
1.\(f(1)=1\)
2.\(f(p^c)=p\otimes c\)(p為質數)
3.\(f(ab)=f(a)f(b)(a,b互質)\)

\(\sum\limits_{i=1}^nf(i)\)\(n\le 10^{10}\)

思路

發現這些性質恰好吻合了我們一開始要求的性質。
發現除2外所有的質數均為奇數,所以就有\(f(p)=p-1\)(p為奇質數)。然後發現這個東西並不是積性函數,沒法預處理g了。怎麼辦?
那就把它拆開,拆成\(f_1(p)=p,f_2(p)=1\),那麼就有\(f(p)=f_1(p)-f_2(p)\)。然後按照上述方法分別預處理除關於\(f_1\)\(g(n,m)\)。關於\(f_2\)\(h(n,m)\)
要說明的是,我們一開始將所有的合數全都當成奇質數來處理,因為最後都要“篩”掉的,所以沒有影響。

具體細節:

預處理g時,有個式子是\(g(pri_j-1,j-1)\),也就是前\(j-1\)個質數的首碼和。所以預處理質數時同時預處理一下首碼和。

code

/*
* @Author: wxyww
* @Date:   2019-12-22 17:42:00
* @Last Modified time: 2019-12-24 21:58:42
*/
#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<queue>
#include<vector>
#include<ctime>
using namespace std;
typedef long long ll;
const int N = 2000010,mod = 1e9 + 7;
ll read() {
    ll x = 0,f = 1;char c = getchar();
    while(c < '0' || c > '9') {
        if(c == '-') f = -1; c = getchar();
    }
    while(c >= '0' && c <= '9') {
        x = x * 10 + c - '0'; c = getchar();
    }
    return x * f;
}
int tot;
ll n,m,w[N],pri[N],sum[N],js,vis[N],g[N],h[N];
void pre() {//篩出質數
    for(int i = 2;i <= m;++i) {
        if(!vis[i]) {
            pri[++js] = i;
            sum[js] = sum[js - 1] + i;
            sum[js] %= mod;
        }
        for(int j = 1;j <= js && i * pri[j] <= m;++j) {
            vis[pri[j] * i] = 1;
            if(i % pri[j] == 0) break;
        }
    }
}
int id1[N],id2[N];
ll S(ll now,int x) {
    if(now <= 1 || pri[x] > now) return 0;

    // printf("%lld %d\n",now,x);

    int k;
    if(now <= m) k = id1[now];
    else k = id2[n / now];

    ll ret = (g[k] - h[k] - sum[x - 1] + x - 1) % mod;

    if(x == 1) ret += 2;//f(2)當作1計算,實際上f(2)=3 

    for(int k = x;k <= js && pri[k] * pri[k] <= now;++k) {
        ll p = pri[k];
        for(int e = 1;p * pri[k] <= now;p = p * pri[k],++e) {
            ret += (pri[k] ^ e) * S(now / p,k + 1) % mod + (pri[k] ^ (e + 1));
            ret %= mod;
        }
    }
    return ret;
}

int main() {
    // freopen("1.in","w",stdout);
    n = read();
    m = sqrt(n);
    pre();

    // puts("!!!");
    for(ll l = 1,r;l <= n;l = r + 1) {
        r = n / (n / l);
        ll tmp = n / l;
        w[++tot] = tmp;
        if(tmp <= m) id1[tmp] = tot;//數組不夠大,通過id1和id2來映射到sqrt(n)以內
        else id2[n / tmp] = tot;

        g[tot] = (tmp + 2) % mod * ((tmp - 1) % mod) % mod;
        
        if(g[tot] & 1) g[tot] += mod;
        
        g[tot] /= 2;
        // g[tot] %= mod;
        h[tot] = tmp - 1;
    }

    // for(int i = 1;i <= tot;++i) printf("%lld ",g[i]);

    for(int j = 1;j <= js;++j) {
        for(int i = 1;i <= tot  && pri[j] * pri[j] <= w[i];++i) {//枚舉順序不能顛倒
            ll tmp = w[i] / pri[j];
            int k;
            if(tmp <= m) k = id1[tmp];
            else k = id2[n / tmp];
            g[i] -= pri[j] * (g[k] - sum[j - 1]) % mod;//枚舉順序不能顛倒的原因
            g[i] %= mod;
            h[i] -= (h[k] - (j - 1));
            h[i] %= mod;
        }
    }


    // cout<<g[tot - 2]<<endl;

    cout<<(S(n,1) + 1 + mod) % mod;//單獨把1算上 
    return 0;
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 磁力搜索嗅探器裝成BT ague-dht ague-dht 是一個磁力鏈接嗅探器,它偽裝成BT下載客服端,加入DHT網路,嗅探磁力鏈接.每秒發送1000條請求時,平均3秒收到1次帶有infohash的announce_peer請求;10次get_peer請求. 環境要求 需要JDK11,MAVEN3 ...
  • 測試的意義 人們針對一個具體問題,通過分析和設計,最後用編程語言寫出了一個程式,如果它通過了語言解釋器(編譯器)的檢查,可以運行了,那麼下一步的工作就是設法確認它確實滿足了我們需求。這篇文章就是討論怎麼確認程式是否滿足用戶提出的需求。 滿足需求,換言之就是功能正常,確認功能正常可以從以下幾個方面確認 ...
  • 這份資源是我經過多年積累才整理歸類出來,有很多電子書我覺質量還是非常高的,由於電子書太多我也是用業餘時間挑著看的,這麼多資源自己保存著也是浪費,就想著現在把資源分享出來,希望能真正幫到大家; 資源我都整理在網盤了,之前分享出來的鏈接沒過幾天就自動取消,我就在文章底部放了二維碼,需要的添加好友就行了, ...
  • 一、僅做瞭解 //用戶組的處理 public class TestUserGroup { private ProcessEngine processEngine=ProcessEngines.getDefaultProcessEngine(); //創建用戶和用戶組 @Test public voi ...
  • 流程中的某個任務由指定的group來完成,其中group由多個user組成。 一、直接指定辦理組 1、流程圖 2、部署和啟動流程 //部署流程 @Test public void bushu() { InputStream inputStream = this.getClass().getResou ...
  • 一、流程圖 二、分配個人任務的方式 1、直接指定辦理人 說明:這樣分配辦理人不夠靈活,因為項目開發中任務的辦理人不要放置在XML當中實際開發中辦理人是不固定的。 2、使用流程變數指定辦理人 2.1 流程圖中的設置 說明:在Activiti中#{username}和${username}的意義是一樣的 ...
  • 一、接收任務 接收任務(ReceiveTask)即等待任務,接收任務是一個簡單任務,它會等待對應消息的到達。當前,官方只實現了這個任務的java語義。 當流程達到接收任務,流程狀態會保存到資料庫中。在任務創建後,意味著流程會進入等待狀態,直到引擎接收了一個特定的消息, 這會觸發流程穿過接收任務繼續執 ...
  • Java JDBC 資料庫鏈接小結隨筆 一、鏈接資料庫的步驟 二、 關於Statement 和 PrepareStatement 兩者區別 用法 三、關於 ResultSet 的一些小結 四、自定義工具類的封裝 五、一些異常的解釋 一、鏈接資料庫的步驟 註冊驅動 獲得鏈接對象 創建sql容器 執行s ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...