hdu2158 最短区间版大家来找碴
生活随笔
收集整理的這篇文章主要介紹了
hdu2158 最短区间版大家来找碴
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
題意:
? ? ? ? ? ? ? ? ?最短區間版大家來找碴
Problem Description
給定一個序列,有N個整數,數值范圍為[0,N)。有M個詢問,每次詢問給定Q個整數,可能出現重復值。要求找出一個最短區間,該區間要包含這Q個整數數值。
你能找的出來嗎?
Input
第一行有兩個整數N,M。(N<100000, M<1000)接著一行有N個整數。再有M個詢問,每個詢問的第一行有一個整數Q(Q<100),第二行跟著Q個整數。當N,M同時為0時,輸入結束。
?
Output
請輸出最短區間的長度。保證有解。
1 2 2 3 1
3
1 2 3
3
1 1 3
0 0
2
Hint
第二個查詢,找到的區間是[4,5]
思路:
? ? ? ? ? ? ? ? ?最短區間版大家來找碴
Problem Description
給定一個序列,有N個整數,數值范圍為[0,N)。有M個詢問,每次詢問給定Q個整數,可能出現重復值。要求找出一個最短區間,該區間要包含這Q個整數數值。
你能找的出來嗎?
Input
第一行有兩個整數N,M。(N<100000, M<1000)接著一行有N個整數。再有M個詢問,每個詢問的第一行有一個整數Q(Q<100),第二行跟著Q個整數。當N,M同時為0時,輸入結束。
?
Output
請輸出最短區間的長度。保證有解。
?
Sample Input
5 21 2 2 3 1
3
1 2 3
3
1 1 3
0 0
Sample Output
32
Hint
第二個查詢,找到的區間是[4,5]
思路:
? ? ? 這個題目做的有點糾結,我的時間復雜度沒次詢問都是O(N)的,那么一次測試的時間復雜度就是O(N*M)這樣是1e了,這樣估計就TLE了,但是卻AC了,雖然理論上測試數據是隨機數據,但是感覺還是有點勉強。我的思路是先找到一個最基本的L,R然后L不停的往后擠,然后維護R來保證當前的區間的正確性,同時更新最小值(這次的代碼寫的有點挫-_-).
#include<stdio.h> #include<string.h>#define N 110000 int num[N] ,markc[N] ,markq[N];int main () {int n ,m ,q ,a ,i;while(~scanf("%d %d" ,&n ,&m) && n + m){for(i = 1 ;i <= n ;i ++)scanf("%d" ,&num[i]);while(m--){scanf("%d" ,&q);memset(markc ,0 ,sizeof(markc));memset(markq ,0 ,sizeof(markq));int ss = 0;for(i = 1 ;i <= q ;i ++){scanf("%d" ,&a);if(!markq[a]) ss ++;markq[a] = 1;}int L ,R ,nowsum ,Ans;L = 1 ,nowsum = 0 ,Ans = n;//找到LR for(i = 1 ;i <= n ;i ++){if(markq[num[i]]) {if(!markc[num[i]]) nowsum ++;markc[num[i]] ++;if(nowsum == ss) {R = i;break;}}}Ans = R - L + 1; for(i = L ;i <= n ;i ++){if(markq[num[i]]) if(!(--markc[num[i]])){int ok = 0;for(int j = R + 1 ;j <= n ;j ++){if(markq[num[j]]){markc[num[j]] ++;if(num[j] == num[i]){ok = 1;R = j;break;}}}if(!ok) break;}if(Ans > R - i) Ans = R - i;}printf("%d\n" ,Ans);}}return 0; }
總結
以上是生活随笔為你收集整理的hdu2158 最短区间版大家来找碴的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: hdu2102 水搜索
- 下一篇: hdu2155 小黑的镇魂曲(dp)