藍橋杯練習 操作格仔

2021-10-01 07:08:08 字數 2823 閱讀 2635

問題描述

有n個格仔,從左到右放成一排,編號為1-n。

共有m次操作,有3種操作型別:

1.修改乙個格仔的權值,

2.求連續一段格仔權值和,

3.求連續一段格仔的最大值。

對於每個2、3操作輸出你所求出的結果。

輸入格式

第一行2個整數n,m。

接下來一行n個整數表示n個格仔的初始權值。

接下來m行,每行3個整數p,x,y,p表示操作型別,p=1時表示修改格仔x的權值為y,p=2時表示求區間[x,y]內格仔權值和,p=3時表示求區間[x,y]內格仔最大的權值。

輸出格式

有若干行,行數等於p=2或3的操作總數。

每行1個整數,對應了每個p=2或3操作的結果。

樣例輸入

4 31 2 3 4

2 1 3

1 4 3

3 1 4

樣例輸出63

資料規模與約定

對於20%的資料n <= 100,m <= 200。

對於50%的資料n <= 5000,m <= 5000。

對於100%的資料1 <= n <= 100000,m <= 100000,0 <= 格仔權值 <= 10000。

#include

using

namespace std;

const

int n =

100000

;struct node

line[

3*n]

;int num[n]

;int first,second;

//儲存p=2和p=3時的解

//建立線段樹

void

create

(int k,

int x,

int y)

//k是當前結點的編號,x是左邊界,y是右邊界 初始時傳的引數是1 1 n

int mid =

(x+y)/2

;//如果不是葉子結點,繼續向下建立樹

//對一棵樹從上到下從左到右編號,則編號為k的結點,其左孩子編號為k*2,右孩子編號為k*2+1

create

(k*2

,x,mid)

;//左孩子的左邊界是父親的左邊界,右邊界是是父親左右邊界的中點

create

(k*2+1

,mid+

1,y)

;//右孩子的左邊界是父親左右邊界的中點,右邊界是父親的右邊界

//遞迴返回之後,左右孩子的maxn和sum值才能確定,所以本節點maxn和sum的更新在遞迴返回之後才進行

line[k]

.maxn =

max(line[k*2]

.maxn,line[k*2+

1].maxn)

;

line[k]

.sum = line[k*2]

.sum+line[k*2+

1].sum;

}//單點修改

void

update

(int k,

int a,

int b)

//k是當前處理的樹節點編號,a是要定位的格仔編號,b是要修改的值

//如果不是我們要定位的結點

int mid =

(line[k]

.l+line[k]

.r)/2;

//判斷我們要定位的結點在當前處理的左孩子裡還是右孩子裡

if(a <= mid)

update

(k*2

,a,b)

;//左孩子裡

else

update

(k*2+1

,a,b)

;//在右孩子裡

//對葉子結點進行修改後,其查詢路徑上的結點值都要更新,因此在遞迴返回後要重新計算一遍maxn和sum

line[k]

.sum = line[k*2]

.sum+line[k*2+

1].sum;

line[k]

.maxn =

max(line[k*2]

.maxn,line[k*2+

1].maxn);}

//區間查詢,p=2和p=3用的都是這個函式,因為他們都是區間查詢,只不過查詢的資料不同

void

query

(int k,

int x,

int y)

//k是當前處理的樹節點編號,x是要查詢的區間左端點,y是要查詢的區間右端點

int mid =

(line[k]

.l+line[k]

.r)/2;

//只要能進入當前處理的結點,說明要查詢的區間肯定是位於當前結點區間裡面的,只不過其左右端點與當前區間中點的位置關係不明確,但與當前區間端點的關係肯定是明確的

if(y <= mid)

//要查詢的區間完全處於當前結點區間中點的左側 ,注意這裡的等號是必須的,如果y=mid時不進這裡,下面mid+1就會超過y

query

(k*2

,x,y)

;else

if(x>mid)

//要查詢的區間完全處於當前結點區間中點的右側

query

(k*2+1

,x,y)

;else

//要查詢的區間部分位於當前區間中點的內部,部分位於外部

}int

main()

else

if(a==2)

else

}return0;

}

藍橋杯 演算法練習 操作格仔(線段樹)

問題描述 有n個格仔,從左到右放成一排,編號為1 n。共有m次操作,有3種操作型別 1.修改乙個格仔的權值,2.求連續一段格仔權值和,3.求連續一段格仔的最大值。對於每個2 3操作輸出你所求出的結果。輸入格式 第一行2個整數n,m。接下來一行n個整數表示n個格仔的初始權值。接下來m行,每行3個整數p...

藍橋杯 操作格仔 線段樹

題目 有n個格仔,從左到右放成一排,編號為1 n。共有m次操作,有3種操作型別 1.修改乙個格仔的權值,2.求連續一段格仔權值和,3.求連續一段格仔的最大值。對於每個2 3操作輸出你所求出的結果。輸入格式 第一行2個整數n,m。接下來一行n個整數表示n個格仔的初始權值。接下來m行,每行3個整數p,x...

藍橋杯 操作格仔 線段樹

剛學習了線段樹,解決區間問題確實是不錯的利器,線段樹實際上就是一棵平衡二叉樹,對於任何操作都能在o long2n 的時間內完成,相比對普通陣列o n 的時間複雜度,有不錯的效率,下面以藍橋網上乙個題操練一下吧。問題描述 有n個格仔,從左到右放成一排,編號為1 n。共有m次操作,有3種操作型別 1.修...