分塊之區間查詢與區間修改

来源:http://www.cnblogs.com/zwfymqz/archive/2017/05/19/6879433.html
-Advertisement-
Play Games

給出一個長為n的數列,以及n個操作,操作涉及區間加法,區間求和。 這題的詢問變成了區間上的詢問,不完整的塊還是暴力;而要想快速統計完整塊的答案,需要維護每個塊的元素和,先要預處理一下。 考慮區間修改操作,不完整的塊直接改,順便更新塊的元素和;完整的塊類似之前標記的做法,直接根據塊的元素和所加的值計算 ...


給出一個長為n的數列,以及n個操作,操作涉及區間加法,區間求和。

 

這題的詢問變成了區間上的詢問,不完整的塊還是暴力;而要想快速統計完整塊的答案,需要維護每個塊的元素和,先要預處理一下。

考慮區間修改操作,不完整的塊直接改,順便更新塊的元素和;完整的塊類似之前標記的做法,直接根據塊的元素和所加的值計算元素和的增量。

 

更改後的區間加法

 1 void interval_add(LL ll,LL rr,LL v)
 2 {
 3     for(LL i=ll;i<=min(where[ll]*m,rr);i++)
 4     //這裡判斷的是where[ll]是不完全塊的情況,也就是ll在他實際塊最左端的右側,
 5     // 然後便利ll-所在塊的結尾/rr,暴力增加 
 6         a[i]+=v,sum[where[ll]]+=v;
 7         // 註意sum表示的是一個塊內的元素和,where表示的是塊的位置 
 8     if(where[ll]!=where[rr])
 9     // 註意如果是ll和rr在一個塊中的話,上面已經加過一邊,所以不用加 
10     {
11         for(LL i=(where[rr]-1)*m+1;i<=rr;i++)
12         // 這裡判斷的是rr在他實際所在塊的最右端左側的情況
13         // where[i]*m表示的是第i個塊最右端的元素
14         // where[rr]-1就是rr所在塊左邊那個塊最右端的元素
15         // 一直到rr暴力增加 
16             a[i]+=v,sum[where[rr]]+=v;
17     }    
18     for(LL i=where[ll]+1;i<=where[rr]-1;i++)
19     //這裡where[ll]和where[rr]均已暴力處理過,所以只枚舉中間的塊就可以 
20         add[i]+=v;
21 } 

 

區間查詢

 1 void interval_ask(LL ll,LL rr)
 2 {
 3     LL ans=0;
 4     for(LL i=ll;i<=min(where[ll]*m,rr);i++)
 5         ans+=a[i]+add[where[i]];
 6         // 暴力求和 ,註意where裡面要寫ll\i,因為我們始終是在ll到它的所在區間結尾的元素內迴圈 
 7         // 
 8     if(where[ll]!=where[rr])
 9         for(LL i=(where[rr]-1)*m+1;i<=rr;i++)
10         //註意這裡需要加一,因為所有的for都是<=,如果不寫+1會加兩邊 
11             ans+=a[i]+add[where[i]];
12             
13     for(LL i=where[ll]+1;i<=where[rr]-1;i++)
14         ans+=sum[i]+add[i]*m;
15         // 這裡要*區間內元素的個數 
16         
17     printf("%lld\n",ans);
18 }

 

完整代碼

 

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #include<cmath>
 5 #define LL long long 
 6 using namespace std;
 7 const LL MAXN=100001;
 8 LL n,q,m,how,l,r,v;
 9 LL a[MAXN];// 初始值
10 LL add[MAXN];// 後來每個塊上加入的值
11 LL where[MAXN];// 記錄每一個值對應第幾塊
12 LL sum[MAXN];//記錄每一塊內的元素和 
13 void interval_add(LL ll,LL rr,LL v)
14 {
15     for(LL i=ll;i<=min(where[ll]*m,rr);i++)
16     //這裡判斷的是where[ll]是不完全塊的情況,也就是ll在他實際塊最左端的右側,
17     // 然後便利ll-所在塊的結尾/rr,暴力增加 
18         a[i]+=v,sum[where[ll]]+=v;
19         // 註意sum表示的是一個塊內的元素和,where表示的是塊的位置 
20     if(where[ll]!=where[rr])
21     // 註意如果是ll和rr在一個塊中的話,上面已經加過一邊,所以不用加 
22     {
23         for(LL i=(where[rr]-1)*m+1;i<=rr;i++)
24         // 這裡判斷的是rr在他實際所在塊的最右端左側的情況
25         // where[i]*m表示的是第i個塊最右端的元素
26         // where[rr]-1就是rr所在塊左邊那個塊最右端的元素
27         // 一直到rr暴力增加 
28             a[i]+=v,sum[where[rr]]+=v;
29     }    
30     for(LL i=where[ll]+1;i<=where[rr]-1;i++)
31     //這裡where[ll]和where[rr]均已暴力處理過,所以只枚舉中間的塊就可以 
32         add[i]+=v;
33 } 
34 void interval_ask(LL ll,LL rr)
35 {
36     LL ans=0;
37     for(LL i=ll;i<=min(where[ll]*m,rr);i++)
38         ans+=a[i]+add[where[i]];
39         // 暴力求和 ,註意where裡面要寫ll,因為我們始終是在ll到它的所在區間結尾的元素內迴圈 
40         
41     if(where[ll]!=where[rr])
42         for(LL i=(where[rr]-1)*m+1;i<=rr;i++)
43         //註意這裡需要加一,因為所有的for都是<=,如果不寫+1會加兩邊 
44             ans+=a[i]+add[where[i]];
45             
46     for(LL i=where[ll]+1;i<=where[rr]-1;i++)
47         ans+=sum[i]+add[i]*m;
48         
49     printf("%lld\n",ans);
50 }
51 int main()
52 {
53     scanf("%lld",&n);
54     scanf("%lld",&q);
55     m=sqrt(n);
56     for(LL i=1;i<=n;i++)
57         scanf("%lld",&a[i]); 
58     for(LL i=1;i<=n;i++)
59         where[i]=(i-1)/m+1,sum[where[i]]+=a[i];// 這裡的i可以-1(hzwer寫的是-1)也可以不寫,不寫的話第一塊的元素個數會是m-1 
60     
61     for(LL i=1;i<=q;i++)
62     {
63         scanf("%lld",&how);
64         if(how==1)// 區間加
65         {
66             scanf("%lld%lld%lld",&l,&r,&v);
67             interval_add(l,r,v);
68         }
69         else// 單點查詢 
70         {
71             scanf("%lld%lld",&l,&r);
72             interval_ask(l,r);
73             // where保存的是這個點所屬的塊,add表示這個塊已經增加的元素
74             //a[v]是這個點開始的值,一加就是答案 
75         } 
76     }
77     return 0;
78 }

 


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

-Advertisement-
Play Games
更多相關文章
  • BarcodeLib -- 一個精簡而不失優雅的條形碼生成庫 引言 在百度進行“C# 條形碼”等類似關鍵字搜索的時候,基本上是使用 ZXing 類庫進行條形碼的生成。今天我所介紹的是另一款類庫 Barcode,一起來共同見證它的強大之處。 目錄 插曲 官方介紹 Nuget 安裝 支持的類型 簡單使用 ...
  • if( transform.position.x > -15 && transform.rotation.y == 0 ) { //小鳥X軸反方向移動速度 transform.position += new Vector3(-0.1f,0,0); } else { //小鳥Y軸旋轉,相當於人的轉身 ...
  • 在win10或者server2016上,我們安裝好IIS以後,把網站掛上去,訪問,可能會報下邊這個錯誤,這個時侯,其實我們應該首先意識到的是,錯誤並沒有告訴我們真正的原因,錯誤信息不全,所以,我們要做的不是立即找原因,而是把錯誤的詳細信息弄出來,如下: 500 - 內部伺服器錯誤。您查找的資源存在問 ...
  • 上一篇中提到驅動的鏈接方式,這篇給出完整鏈接代碼和使用實例 資料庫完整鏈接 添加數據: 更新數據: 更新,除了Update還有一個UpdateAll方法,不做過多解釋. 增,刪,改 沒什麼可說,主要說下查詢 簡單數據查詢: 靈活查詢: 可以看出只要給Find傳遞Document,然後接受數據就可以了 ...
  • 下載地址 https://github.com/samus/mongodb-csharp 官方驅動不順手,所以用了這個, 使用鏈接池的情況下,每次指定資料庫命令,都會建立一個連接,不用多長時間,連接池就會慢,設置到最大,連接池慢也是時間問題, 項目里解決方式是使用單例連接: ...
  • Java基礎三 一、關鍵字 二、標識符 2.1 定義 標識某些東西的符號:名稱:類名就是標識符的一種 26個英文字母,0-9,_和$ 2.2 註意 數字不可以開頭 不可以用關鍵字(你姓天就不要叫天安門,這是國家的名字) _和$用來連接單詞 三、註釋 非常重要 3.1 作用: 註解說明 調試程式 3. ...
  • 預設是 singleton ,單例模式,如下代碼: 獲取的 service 和 service2 都是一個對象,結果為true。 如果將 scope 設置為 prototype: 1 <bean id="userService" class="com.bjsxt.service.UserServic ...
  • Js原生Ajax和Jquery的Ajax 學習目標 案例1-非同步校驗用戶名是否存在 案例2-站內查詢 學習目標 案例1-非同步校驗用戶名是否存在 案例2-站內查詢 一、Ajax概述 1.什麼是同步,什麼是非同步 同步現象:客戶端發送請求到伺服器端,當伺服器返迴響應之前,客戶端都處於等待卡死狀態 非同步現象 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...