欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++前綴和與差分的使用示例講解

 更新時間:2023年03月09日 08:48:41   作者:平凡的人1  
前綴和是指某序列的前n項和,可以把它理解為數(shù)學上的數(shù)列的前n項和,而差分可以看成前綴和的逆運算。合理的使用前綴和與差分,可以將某些復雜的問題簡單化。類似于數(shù)學中的求導和積分,差分可以看成前綴和的逆運算

前綴和差分是一對逆運算

1.一維前綴和

有一個長度為n的數(shù)組an:a1,a2…an;

對于前綴和:Si= a1+a2+…+ai

如何求Si,S[i] = s[i-1]+a[i]

前綴和可以快速求出原數(shù)組里面一段數(shù)的和。比如求一段區(qū)間[l,r],如果按照原來的做法,需要循環(huán)一遍,O(n),有前綴和的算法:

這個區(qū)間的數(shù)就是(Sr) - (sl-1)。同時,為了方便計算令s[0] = 0.比如計算[1,l],既s[l]-s[0] = s[l].

其實前綴和就是一個區(qū)間相減的操作,統(tǒng)一處理。前綴和其實是非常簡單的

練習題:

輸入一個長度為 nn 的整數(shù)序列。

接下來再輸入 mm 個詢問,每個詢問輸入一對 l,rl,r。

對于每個詢問,輸出原序列中從第 ll 個數(shù)到第 rr 個數(shù)的和。

輸入格式

第一行包含兩個整數(shù) nn 和 mm。

第二行包含 nn 個整數(shù),表示整數(shù)數(shù)列。

接下來 mm 行,每行包含兩個整數(shù) ll 和 rr,表示一個詢問的區(qū)間范圍。

輸出格式

共 mm 行,每行輸出一個詢問的結果。

數(shù)據(jù)范圍

1≤l≤r≤n1≤l≤r≤n,

1≤n,m≤1000001≤n,m≤100000,

−1000≤數(shù)列中元素的值≤1000

#include <iostream>
using namespace std;
const int N = 100010;
int n,m;
int a[N],S[N];
int main()
{
    scanf("%d%d",&n,&m);
    for(int i = 1;i<=n;i++) scanf("%d",&a[i]);
    for(int i = 1;i<=n;i++) S[i] = S[i-1]+a[i];
    while(m--)
    {
        int l,r;
        scanf("%d%d",&l,&r);
        printf("%d\n",S[r]-S[l-1]);
    }
    return 0;
}

2.二維前綴和

二維前綴和是在一個二維矩陣里求子矩陣的和

練習題:

輸入一個 nn 行 mm 列的整數(shù)矩陣,再輸入 qq 個詢問,每個詢問包含四個整數(shù) x1,y1,x2,y2x1,y1,x2,y2,表示一個子矩陣的左上角坐標和右下角坐標。

對于每個詢問輸出子矩陣中所有數(shù)的和。

輸入格式

第一行包含三個整數(shù) n,m,qn,m,q。

接下來 nn 行,每行包含 mm 個整數(shù),表示整數(shù)矩陣。

接下來 qq 行,每行包含四個整數(shù) x1,y1,x2,y2x1,y1,x2,y2,表示一組詢問。

輸出格式

共 qq 行,每行輸出一個詢問的結果。

數(shù)據(jù)范圍

1≤n,m≤10001≤n,m≤1000,

1≤q≤2000001≤q≤200000,

1≤x1≤x2≤n1≤x1≤x2≤n,

1≤y1≤y2≤m1≤y1≤y2≤m,

−1000≤矩陣內(nèi)元素的值≤1000\

#include <iostream>
using namespace std;
const int N = 1010;
int n,m,q;
long a[N][N],s[N][N];
int main()
{
    scanf("%d%d%d",&n,&m,&q);
    for(int i = 1;i<=n;i++)
    {
        for(int j = 1;j<=m;j++)
        {
            scanf("%d",&a[i][j]);
            //求前綴和
            s[i][j] = s[i-1][j]+s[i][j-1]-s[i-1][j-1]+a[i][j];
        }
    }
    while(q--)
    {
        int x1,y1,x2,y2;
        scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
        printf("%d\n",s[x2][y2]-s[x2][y1-1]-s[x1-1][y2]+s[x1-1][y1-1]);
    }
    return 0;
}

3.一維差分

給定a[1],a[2],…,a[n]構造差分數(shù)組b[N],使得a[i] = b[1]+b[2]+…+b[i]

b1 = a1,b2 = a2-a1,b3 = a3-a2,直到bn = an-an-1

b是a的差分,a是b的前綴和。有b數(shù)組就可以通過O(n)的時間復雜度得到a數(shù)組。

推導過程:

現(xiàn)在在a數(shù)組[L,R]中全部加上C,那就是al+C,al+1+C,…,ar+C,通過暴力的方式O(n)可以求解,那差分可以變成O(1)

在[L,R]中,如果我們在b數(shù)組bl+C,那么al也會加上C,al+1也會加上C…an+1也會加上C,因為每一次都會加上一個bl。但是我們只要al到ar加上C,那么ar后面不要加上C,那么我們直接讓br-c即可完成數(shù)組a在[L,R]范圍里全部加上C。

核心操作是將a[L~R]全部加上C等價于b[L] +=C,b[R+1]-=C

把O(n)提高到O(1)

假定a數(shù)組全是初始化為0,那b數(shù)組也是全為0,但是題目a數(shù)組并不是0,我們可以看成進行n次插入操作,第一次是在原數(shù)組a[1,1]加上a1,第二次是在原數(shù)組a[2,2]加上a2…以此類推即可,所以并不需要去想如何構造差分

題目:

輸入一個長度為 nn 的整數(shù)序列。

接下來輸入 mm 個操作,每個操作包含三個整數(shù) l,r,cl,r,c,表示將序列中 [l,r][l,r] 之間的每個數(shù)加上 cc。

請你輸出進行完所有操作后的序列。

輸入格式

第一行包含兩個整數(shù) nn 和 mm。

第二行包含 nn 個整數(shù),表示整數(shù)序列。

接下來 mm 行,每行包含三個整數(shù) l,r,cl,r,c,表示一個操作。

輸出格式

共一行,包含 nn 個整數(shù),表示最終序列。

數(shù)據(jù)范圍

1≤n,m≤1000001≤n,m≤100000,

1≤l≤r≤n1≤l≤r≤n,

−1000≤c≤1000−1000≤c≤1000,

−1000≤整數(shù)序列中元素的值≤1000

#include <iostream>
using namespace std;
const int N = 100010;
int n,m;
int a[N],b[N];
void insert(int l,int r,int c)
{
    b[l]+=c;
    b[r+1]-=c;
}
int main()
{
    cin>>n>>m;
    for(int i = 1;i<=n;i++)
    {
        cin>>a[i];
        insert(i,i,a[i]);
    }
    while(m--)
    {
        int l,r,c;
        cin>>l>>r>>c;
        insert(l,r,c);
    }
    for(int i = 1;i<=n;i++) a[i] = a[i-1]+b[i];
    for(int i = 1;i<=n;i++) printf("%d ",a[i]);
    return 0;
}

4.二維差分

二維差分也是一樣的道理

練習題:

輸入一個 nn 行 mm 列的整數(shù)矩陣,再輸入 qq 個操作,每個操作包含五個整數(shù) x1,y1,x2,y2,cx1,y1,x2,y2,c,其中 (x1,y1)(x1,y1) 和 (x2,y2)(x2,y2) 表示一個子矩陣的左上角坐標和右下角坐標。

每個操作都要將選中的子矩陣中的每個元素的值加上 cc。

請你將進行完所有操作后的矩陣輸出。

輸入格式

第一行包含整數(shù) n,m,qn,m,q。

接下來 nn 行,每行包含 mm 個整數(shù),表示整數(shù)矩陣。

接下來 qq 行,每行包含 55 個整數(shù) x1,y1,x2,y2,cx1,y1,x2,y2,c,表示一個操作。

輸出格式

共 nn 行,每行 mm 個整數(shù),表示所有操作進行完畢后的最終矩陣。

數(shù)據(jù)范圍

1≤n,m≤10001≤n,m≤1000,

1≤q≤1000001≤q≤100000,

1≤x1≤x2≤n1≤x1≤x2≤n,

1≤y1≤y2≤m1≤y1≤y2≤m,

−1000≤c≤1000−1000≤c≤1000,

−1000≤矩陣內(nèi)元素的值≤1000

#include <iostream>
using namespace std;
const int N = 1010;
int n,m,q;
int a[N][N],b[N][N];
void Insert(int x1,int y1,int x2,int y2,int c)
{
    b[x1][y1]+=c;
    b[x2+1][y1]-=c;
    b[x1][y2+1]-=c;
    b[x2+1][y2+1]+=c;
}
int main()
{
    scanf("%d%d%d",&n,&m,&q);
    for(int i = 1;i<=n;i++)
    {
        for(int j = 1;j<=m;j++)
        {
            scanf("%d",&a[i][j]);
        }
    }
    for(int i = 1;i<=n;i++)
    {
        for(int j = 1;j<=m;j++)
        {
            Insert(i,j,i,j,a[i][j]);
        }
    }
    while(q--)
    {
        int x1,y1,x2,y2,c;
        scanf("%d%d%d%d%d",&x1,&y1,&x2,&y2,&c);
        Insert(x1,y1,x2,y2,c);
    }
    for(int i = 1;i<=n;i++)
    {
        for(int j = 1;j<=m;j++)
        {
            b[i][j] += b[i-1][j]+b[i][j-1]-b[i-1][j-1];
        }
    }
     for(int i = 1;i<=n;i++)
    {
        for(int j = 1;j<=m;j++)
        {
            printf("%d ",b[i][j]);
        }
        puts("");
    }
    return 0;
}

到此這篇關于C++前綴和與差分的使用示例講解的文章就介紹到這了,更多相關C++前綴和與差分內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • C和C++中的基本數(shù)據(jù)類型的大小及表示范圍詳解

    C和C++中的基本數(shù)據(jù)類型的大小及表示范圍詳解

    這篇文章主要介紹了C和C++中的基本數(shù)據(jù)類型的大小及表示范圍詳解,基本數(shù)據(jù)類型有int、long、long long、float、double、char、string,正文有詳細介紹,歡迎參考
    2018-01-01
  • 基于Qt實現(xiàn)可拖動自定義控件

    基于Qt實現(xiàn)可拖動自定義控件

    這篇文章主要為大家詳細介紹了如何基于Qt實現(xiàn)可拖動自定義控件,文中的示例代碼講解詳細,具有一定的借鑒價值,感興趣的小伙伴可以了解一下
    2023-04-04
  • Qt數(shù)據(jù)庫應用之實現(xiàn)數(shù)據(jù)分組導出

    Qt數(shù)據(jù)庫應用之實現(xiàn)數(shù)據(jù)分組導出

    這篇文章主要為大家詳細介紹了如何利用Qt實現(xiàn)數(shù)據(jù)庫數(shù)據(jù)分組導出,文中的示例代碼講解詳細,對我們學習或工作有一定參考價值,需要的可以了解一下
    2022-06-06
  • C語言編程數(shù)據(jù)結構的棧和隊列

    C語言編程數(shù)據(jù)結構的棧和隊列

    本篇文章是C語言編程篇,主要為大家介紹C語言編程中的數(shù)據(jù)結構,詳細的講解了數(shù)據(jù)結構的棧和隊列有需要的朋友可以借鑒參考下,希望可以有所幫助
    2021-09-09
  • C++實現(xiàn)十進制數(shù)轉(zhuǎn)為其它進制數(shù)

    C++實現(xiàn)十進制數(shù)轉(zhuǎn)為其它進制數(shù)

    這篇文章主要為大家詳細介紹了C++實現(xiàn)十進制數(shù)轉(zhuǎn)為其它進制數(shù),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-04-04
  • C語言進階棧幀示例詳解教程

    C語言進階棧幀示例詳解教程

    這篇文章主要為大家介紹了C語言進階棧幀的示例詳解教程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-02-02
  • 利用C++ R3層斷鏈實現(xiàn)模塊隱藏功能

    利用C++ R3層斷鏈實現(xiàn)模塊隱藏功能

    在R3層的模塊隱藏,我們需要做的就是將其該鏈表斷鏈,將某一模塊從這個雙向鏈表中摘除,這樣再調(diào)用傳統(tǒng)的API時就會搜索不到。本文重點給大家介紹利用C++ R3層斷鏈實現(xiàn)模塊隱藏功能,感興趣的朋友一起看看吧
    2019-10-10
  • c語言實現(xiàn)的帶通配符匹配算法

    c語言實現(xiàn)的帶通配符匹配算法

    這篇文章主要介紹了c語言實現(xiàn)的帶通配符匹配算法,需要的朋友可以參考下
    2015-03-03
  • C++對象內(nèi)存分布詳解(包括字節(jié)對齊和虛函數(shù)表)

    C++對象內(nèi)存分布詳解(包括字節(jié)對齊和虛函數(shù)表)

    下面小編就為大家?guī)硪黄狢++對象內(nèi)存分布詳解(包括字節(jié)對齊和虛函數(shù)表)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-12-12
  • C++類模板與函數(shù)模板基礎詳細講解

    C++類模板與函數(shù)模板基礎詳細講解

    C++語言的模板技術包括函數(shù)模板和類模板,模板技術是一種代碼重用技術,函數(shù)和類是C++語言中兩種主要的重用代碼形式,這篇文章主要介紹了C++函數(shù)模板和類模板,需要的朋友可以參考下
    2022-08-08

最新評論