HDOJ 1754 HDU 1754 I Hate It ACM 1754 IN HDU
Posted on 2010-09-15 16:08 MiYu 閱讀(456) 評(píng)論(0) 編輯 收藏 引用 所屬分類: ACM ( 數(shù)據(jù)結(jié)構(gòu) ) 、ACM ( 線段樹(shù) )MiYu原創(chuàng), 轉(zhuǎn)帖請(qǐng)注明 : 轉(zhuǎn)載自 ______________白白の屋
題目地址:
http://acm.hdu.edu.cn/showproblem.php?pid=1754
題目描述:
I Hate It
Time Limit: 9000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 6306 Accepted Submission(s): 2267
Problem Description
很多學(xué)校流行一種比較的習(xí)慣。老師們很喜歡詢問(wèn),從某某到某某當(dāng)中,分?jǐn)?shù)最高的是多少。
這讓很多學(xué)生很反感。
不管你喜不喜歡,現(xiàn)在需要你做的是,就是按照老師的要求,寫(xiě)一個(gè)程序,模擬老師的詢問(wèn)。當(dāng)然,老師有時(shí)候需要更新某位同學(xué)的成績(jī)。
這讓很多學(xué)生很反感。
不管你喜不喜歡,現(xiàn)在需要你做的是,就是按照老師的要求,寫(xiě)一個(gè)程序,模擬老師的詢問(wèn)。當(dāng)然,老師有時(shí)候需要更新某位同學(xué)的成績(jī)。
Input
本題目包含多組測(cè)試,請(qǐng)?zhí)幚淼轿募Y(jié)束。
在每個(gè)測(cè)試的第一行,有兩個(gè)正整數(shù) N 和 M ( 0<N<=200000,0<M<5000 ),分別代表學(xué)生的數(shù)目和操作的數(shù)目。
學(xué)生ID編號(hào)分別從1編到N。
第二行包含N個(gè)整數(shù),代表這N個(gè)學(xué)生的初始成績(jī),其中第i個(gè)數(shù)代表ID為i的學(xué)生的成績(jī)。
接下來(lái)有M行。每一行有一個(gè)字符 C (只取'Q'或'U') ,和兩個(gè)正整數(shù)A,B。
當(dāng)C為'Q'的時(shí)候,表示這是一條詢問(wèn)操作,它詢問(wèn)ID從A到B(包括A,B)的學(xué)生當(dāng)中,成績(jī)最高的是多少。
當(dāng)C為'U'的時(shí)候,表示這是一條更新操作,要求把ID為A的學(xué)生的成績(jī)更改為B。
在每個(gè)測(cè)試的第一行,有兩個(gè)正整數(shù) N 和 M ( 0<N<=200000,0<M<5000 ),分別代表學(xué)生的數(shù)目和操作的數(shù)目。
學(xué)生ID編號(hào)分別從1編到N。
第二行包含N個(gè)整數(shù),代表這N個(gè)學(xué)生的初始成績(jī),其中第i個(gè)數(shù)代表ID為i的學(xué)生的成績(jī)。
接下來(lái)有M行。每一行有一個(gè)字符 C (只取'Q'或'U') ,和兩個(gè)正整數(shù)A,B。
當(dāng)C為'Q'的時(shí)候,表示這是一條詢問(wèn)操作,它詢問(wèn)ID從A到B(包括A,B)的學(xué)生當(dāng)中,成績(jī)最高的是多少。
當(dāng)C為'U'的時(shí)候,表示這是一條更新操作,要求把ID為A的學(xué)生的成績(jī)更改為B。
Output
對(duì)于每一次詢問(wèn)操作,在一行里面輸出最高成績(jī)。
Sample Input
5 6 1 2 3 4 5 Q 1 5 U 3 6 Q 3 4 Q 4 5 U 2 9 Q 1 5
Sample Output
5 6 5 9HintHuge input,the C function scanf() will work better than cin
感覺(jué)好久沒(méi)有A題了 , 最近一直沒(méi)有狀態(tài), 豆豆也轉(zhuǎn)行了, 郁悶....... 因?yàn)榇蛩?專精 數(shù)據(jù)結(jié)構(gòu)方面,
所以這幾天一直都在復(fù)習(xí) 數(shù)據(jù)結(jié)構(gòu), 再一次學(xué)習(xí)了 線段樹(shù), 以前只會(huì)用它來(lái) 更新點(diǎn) 求和 , 現(xiàn)在終于水了一
個(gè) RMQ 的裸題了, HAPPY 一下....
對(duì)于 RMQ 的題目, 看PPT 上面的 DP 我直接0rz了...........表示DP只會(huì)做水題.... 這方面還是交給
YCH 吧. 不過(guò)看了 shǎ崽 大神 博客的 線段樹(shù)專輯后, 發(fā)現(xiàn) 用線段樹(shù)處理 這類問(wèn)題 非常方便, 修改查詢
都是 O (logN)的 , 稍稍優(yōu)化了下輸入, 234MS AC ........
代碼如下 :

/*
Coded By : MiYu
Link : http://www.cnblogs.com/MiYu || http://www.shnenglu.com/MiYu
Author By : MiYu
Test : 1
Program : 1754
*/
//#pragma warning( disable:4789 )
#include <iostream>
#include <algorithm>
#include <string>
#include <set>
#include <map>
#include <utility>
#include <queue>
#include <stack>
#include <list>
#include <vector>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
using namespace std;
inline int max ( int a, int b ){
return a > b ? a : b;
}
typedef struct seg_Tree {
int left, right;
int mid() { return (left+right)>>1; }
int max;
}S;
S seg[605000];
int key[200010];
int creat ( int left, int right, int root = 1 ){
seg[root].left = left;
seg[root].right = right;
if ( left == right )
return seg[root].max = key[left];
int mid = seg[root].mid();
return seg[root].max = max ( creat ( left, mid, root << 1 ),creat ( mid + 1, right, ( root << 1 ) + 1 ) );
}
void modify ( int val, int pos, int r = 1 ){
if ( seg[r].left == seg[r].right ){
seg[r].max = val;
return;
}
int mid = seg[r].mid();
if ( pos <= mid ){
modify ( val, pos, r << 1 );
} else {
modify ( val, pos, ( r << 1 ) + 1 );
}
seg[r].max = max ( seg[r<<1].max, seg[ (r<<1) + 1 ].max );
}
int quy ( int left, int right, int r = 1 ){
if ( seg[r].left == left && seg[r].right == right ){
return seg[r].max;
}
int mid = seg[r].mid();
if ( right <= mid ){
return quy ( left, right, r << 1 );
} else if ( left > mid ) {
return quy ( left, right, (r << 1) + 1 );
} else {
return max ( quy ( left, mid, r << 1 ), quy ( mid + 1, right, (r << 1) + 1 ) );
}
}
inline bool scan_d(int &num)
{
char in;bool IsN=false;
in=getchar();
if(in==EOF) return false;
while(in!='-'&&(in<'0'||in>'9')) in=getchar();
if(in=='-'){ IsN=true;num=0;}
else num=in-'0';
while(in=getchar(),in>='0'&&in<='9'){
num*=10,num+=in-'0';
}
if(IsN) num=-num;
return true;
}
int main ()
{
int N, M, x, y;
while ( scan_d(N) && scan_d(M) ){
for ( int i = 1; i <= N; ++ i ){
scan_d( key[i] );
}
creat ( 1, N );
while ( M -- ){
char ask[5];
scanf ( "%s", ask );
scan_d(x);
scan_d(y);
switch ( ask[0] ){
case 'Q': printf ( "%d\n", quy ( x,y ) );
break;
case 'U': modify ( y, x );
}
}
}
return 0;
}
/*
5 6
1 2 3 4 5
Q 1 5
U 3 6
Q 3 4
Q 4 5
U 2 9
Q 1 5
*/
Coded By : MiYu
Link : http://www.cnblogs.com/MiYu || http://www.shnenglu.com/MiYu
Author By : MiYu
Test : 1
Program : 1754
*/
//#pragma warning( disable:4789 )
#include <iostream>
#include <algorithm>
#include <string>
#include <set>
#include <map>
#include <utility>
#include <queue>
#include <stack>
#include <list>
#include <vector>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
using namespace std;
inline int max ( int a, int b ){
return a > b ? a : b;
}
typedef struct seg_Tree {
int left, right;
int mid() { return (left+right)>>1; }
int max;
}S;
S seg[605000];
int key[200010];
int creat ( int left, int right, int root = 1 ){
seg[root].left = left;
seg[root].right = right;
if ( left == right )
return seg[root].max = key[left];
int mid = seg[root].mid();
return seg[root].max = max ( creat ( left, mid, root << 1 ),creat ( mid + 1, right, ( root << 1 ) + 1 ) );
}
void modify ( int val, int pos, int r = 1 ){
if ( seg[r].left == seg[r].right ){
seg[r].max = val;
return;
}
int mid = seg[r].mid();
if ( pos <= mid ){
modify ( val, pos, r << 1 );
} else {
modify ( val, pos, ( r << 1 ) + 1 );
}
seg[r].max = max ( seg[r<<1].max, seg[ (r<<1) + 1 ].max );
}
int quy ( int left, int right, int r = 1 ){
if ( seg[r].left == left && seg[r].right == right ){
return seg[r].max;
}
int mid = seg[r].mid();
if ( right <= mid ){
return quy ( left, right, r << 1 );
} else if ( left > mid ) {
return quy ( left, right, (r << 1) + 1 );
} else {
return max ( quy ( left, mid, r << 1 ), quy ( mid + 1, right, (r << 1) + 1 ) );
}
}
inline bool scan_d(int &num)
{
char in;bool IsN=false;
in=getchar();
if(in==EOF) return false;
while(in!='-'&&(in<'0'||in>'9')) in=getchar();
if(in=='-'){ IsN=true;num=0;}
else num=in-'0';
while(in=getchar(),in>='0'&&in<='9'){
num*=10,num+=in-'0';
}
if(IsN) num=-num;
return true;
}
int main ()
{
int N, M, x, y;
while ( scan_d(N) && scan_d(M) ){
for ( int i = 1; i <= N; ++ i ){
scan_d( key[i] );
}
creat ( 1, N );
while ( M -- ){
char ask[5];
scanf ( "%s", ask );
scan_d(x);
scan_d(y);
switch ( ask[0] ){
case 'Q': printf ( "%d\n", quy ( x,y ) );
break;
case 'U': modify ( y, x );
}
}
}
return 0;
}
/*
5 6
1 2 3 4 5
Q 1 5
U 3 6
Q 3 4
Q 4 5
U 2 9
Q 1 5
*/