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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...