BZOJ3413: 匹配(尾碼自動機 線段樹合併)

来源:https://www.cnblogs.com/zwfymqz/archive/2019/02/20/10409186.html
-Advertisement-
Play Games

題意 "題目鏈接" Sol 神仙題Orz 尾碼自動機 + 線段樹合併。。。 首先可以轉化一下模型(想不到qwq):問題可以轉化為統計$B$中每個首碼在$A$中出現的次數。(畫一畫就出來了) 然後直接對$A$串建SAM,線段樹合併維護一下siz就行了 cpp include using namespa ...


題意

題目鏈接

Sol

神仙題Orz

尾碼自動機 + 線段樹合併。。。

首先可以轉化一下模型(想不到qwq):問題可以轉化為統計\(B\)中每個首碼在\(A\)中出現的次數。(畫一畫就出來了)

然後直接對\(A\)串建SAM,線段樹合併維護一下siz就行了

#include<bits/stdc++.h>
using namespace std;
const int MAXN = 4e5 + 10, SS = 1e7 + 10;
int N, M;
char S[MAXN], T[MAXN];
int fa[MAXN], len[MAXN], ch[MAXN][11], root = 1, las = 1, tot = 1;
vector<int> par[MAXN];
int insert(int x) {
    int now = ++tot, pre = las; las = now; len[now] = len[pre] + 1;
    for(; pre && !ch[pre][x]; pre = fa[pre]) ch[pre][x] = now;
    if(!pre) fa[now] = root;
    else {
        int q = ch[pre][x];
        if(len[pre] + 1 == len[q]) fa[now] = q;
        else {
            int nq = ++tot; fa[nq] = fa[q]; len[nq] = len[pre] + 1;
            memcpy(ch[nq], ch[q], sizeof(ch[q]));
            for(; pre && ch[pre][x] == q; pre = fa[pre]) ch[pre][x] = nq;
            fa[q] = fa[now] = nq;
        }
    }
    return las;
}
void Build() {
    for(int i = 1; i <= tot; i++) par[fa[i]].push_back(i);
}
int rt[SS], ls[SS], rs[SS], sum[SS], cnt;
void update(int k) {
    sum[k] = sum[ls[k]] + sum[rs[k]];
}
void Modify(int &k, int l, int r, int p, int v) {
    if(!k) k = ++cnt;
    if(l == r) {sum[k]++; return ;}
    int mid = l + r >> 1;
    if(p <= mid) Modify(ls[k], l, mid, p, v);
    else Modify(rs[k], mid + 1, r, p, v);
    update(k);
}
int Merge(int x, int y) {
    if(!x || !y) return x ^ y;
    int nw = ++cnt;
    if(!ls[x] && !rs[x]) {sum[nw] = sum[x] + sum[y]; return nw;}
    ls[nw] = Merge(ls[x], ls[y]);
    rs[nw] = Merge(rs[x], rs[y]);
    update(nw);
    return nw;
}
int Get(int k, int l, int r) {
    if(!k) return N;
    if(l == r) return l;
    int mid = l + r >> 1;
    if(sum[ls[k]]) return Get(ls[k], l, mid);
    else return Get(rs[k], mid + 1, r);
}
int Query(int k, int l, int r, int ql, int qr) {
    if(!k || (l > r) || (ql > qr)) return 0;
    if(ql <= l && r <= qr) 
        return sum[k];
    int mid = l + r >> 1;
    if(ql > mid) return Query(rs[k], mid + 1, r, ql, qr);
    else if(qr <= mid) return Query(ls[k], l, mid, ql, qr);
    else return Query(ls[k], l, mid, ql, qr) + Query(rs[k], mid + 1, r, ql, qr);
}
void dfs(int x) {
    for(auto &to : par[x]) {
        dfs(to);
        rt[x] = Merge(rt[x], rt[to]);
    }
}
void solve() {
    int n = strlen(T + 1), now = root, flag = 0, Lim = 0;
    for(int i = 1; i <= n; i++) {
        int nxt = T[i] - '0';
        if(!ch[now][nxt]) {flag = 1; break;}
        now = ch[now][nxt];
        if(i == n) 
            Lim = Get(rt[now], 1, N) - n;//µÚÒ»´Î³öÏÖµÄλÖà 
    }
    int ans = 0;
    if(flag) ans = N;
    else ans = Lim + n;
    now = root;
    for(int i = 1; i <= n; i++) {
        int nxt = T[i] - '0';
        if(!ch[now][nxt]) break;
        now = ch[now][nxt];
        if(flag) ans += Query(rt[now], 1, N, 1, N);
        else ans += Query(rt[now], 1, N, 1, Lim + i - 1);
    }
    cout << ans << '\n';
}
int main() {
    //freopen("1.in", "r", stdin); freopen("b.out", "w", stdout);
    cin >> N;
    scanf("%s", S + 1);
    for(int i = 1; i <= N; i++) 
        Modify(rt[insert(S[i] - '0')], 1, N, i, 1);
    Build();
    dfs(root);
    cin >> M;
    for(int i = 1; i <= M; i++) {
        scanf("%s", T + 1);
        solve();
    }
    return 0;
}
/*
7
1090901
4
0901
87650
109
090
*/

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

-Advertisement-
Play Games
更多相關文章
  • 1 搭建springboot 2 配置pom依賴(springboot版本為2.1.3) 3 寫一個controller類 4 SpringBootApplication中增加註解ComponentScan,並啟動 5 啟動測試 http://localhost:8080/index 5.1 開啟驗 ...
  • 在寫代碼過程中,我們修改代碼中寄存器的值,但是有時寄存器的數據較多,手動修改容易出現錯誤而且花費的時間長 這是一段寄存器的配置值: 0x00, 0x34 0x35, 0x25 0x10, 0xd4 0xf5, 0xa5 0x00, 0x34 0x3a, 0xff 0x00, 0x00 0x34, 0 ...
  • 我一直都有一個疑問,豐巢業務服務的生產環境jvm參數設置是禁止system.gc的,也就是開啟設置:-XX:+DisableExplicitGC,但是生產環境卻從來沒有出現過堆外記憶體溢出的情況。說明一下,豐巢使用了阿裡開源的dubbo,而dubbo底層通信預設情況下使用了3.2.5.Final版本的 ...
  • 在實際開發過程中,我們有時候會遇到主線程調用子線程,要等待子線程返回的結果來進行下一步動作的業務。 那麼怎麼獲取子線程返回的值呢,我這裡總結了三種方式: Entity類 主線程等待(這個一看代碼便知曉,沒什麼問題) Join方法阻塞當前線程以等待子線程執行完畢 通過實現Callable介面 這裡又分 ...
  • 一、冒泡排序 冒泡排序(Bubble Sort)是一種交換排序,它的基本思想是:兩兩比較相鄰記錄的關鍵字,如果反序則交換,直到沒有反序的記錄為止。 進一步理解為(假設由小到大排序):對於給定的n個記錄,從第一個記錄開始依次對相鄰的兩個記錄進行比較,當前面的記錄大於後面的記錄時,交換位置,進行一輪比較 ...
  • BUG觸發時的完整報錯內容(本地無關路徑用已經用 隱去): 在解析HTML時,標簽開始部分使用形如 的瀏覽器判斷標識符,結束時結束標簽 (正確的開始和結束標簽應該為 和 )無法正常匹配關閉即可觸發。 觸發BUG的示例代碼如下: 在 Python 3.7.0 版本中,觸發BUG部分的代碼存在於 中的 ...
  • 題目1.7 1 列印沙漏 (20 分) 本題要求你寫個程式把給定的符號列印成沙漏的形狀。例如給定17個“ ”,要求按下列格式列印 所謂“沙漏形狀”,是指每行輸出奇數個符號;各行符號中心對齊;相鄰兩行符號數差2;符號數先從大到小順序遞減到1,再從小到大順序遞增;首尾符號數相等。 給定任意N個符號,不一 ...
  • 1. __new__ 和 __init__ 的區別 python 2.x 老式類(預設繼承type) 老式類中沒有__new__類方法(也就是說定義也不會執行,它不是老式類的類方法),__Init__ 作為構造函數,創建實例對象,並初始化。 過程: 類 => __init__() => 實例(sel ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...