程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> C++入門知識 >> ZOJ 3790 Consecutive Blocks

ZOJ 3790 Consecutive Blocks

編輯:C++入門知識

ZOJ 3790 Consecutive Blocks


題目鏈接~~>

做題感悟:這題可以直接二分,也可以分段二分和枚舉差不多。

解題思路:

存每個數的時候同時存每個數的下標,然後排個序先按從小到大排,再按下標從小到大排,這樣相同的數都在一塊,因為最多可以選 k 個,那麼我們就按 K 個來,這樣依次枚舉每個數的最大范圍(在同一個數的區間內),分段二分就可以了。感覺數據有點水,寫完竟然 1 A .

代碼:

#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std  ;
#define INT long long int
#define L(x)  (x * 2)
#define R(x)  (x * 2 + 1)
const int INF = 0x3f3f3f3f ;
const double esp = 0.0000000001 ;
const double PI = acos(-1.0) ;
const int mod = 1000000007 ;
const int MY = (1<<5) + 5 ;
const int MX = (1<<20) + 5 ;
const int S = 20 ;
int n ,m ;
int sum[MX] ,g[MX] ;
struct node
{
    int c ,id ;
}T[MX] ;
bool cmp(node a ,node b)
{
    if(a.c == b.c)  return a.id < b.id ;
    else   return a.c < b.c ;
}
int main()
{
    while(~scanf("%d%d" ,&n ,&m))
    {
        for(int i = 1 ;i <= n ; ++i)
        {
            scanf("%d" ,&T[i].c) ;
            T[i].id = i ;       // 位置
        }
        sort(T ,T+n+1 ,cmp) ;
        memset(sum ,0 ,sizeof(sum)) ;
        g[0] = 0 ; g[1] = T[1].c ;
        for(int i = 2 ;i <= n ; ++i)
        {
            if(T[i].c == T[i-1].c)
                 sum[i] = sum[i-1] + T[i].id - T[i-1].id -1 ;
            g[i] = T[i].c ;
        }
        int i = 1 ,ans = 1 ,Le ,Rt ;
        while(i < n)
        {
            Le = i ;
            Rt = upper_bound(g ,g+n+1 ,T[i].c) - g ; // 尋找同一組的邊界
            for(int j = Le ;j < Rt ; ++j)
            {
                int mx = upper_bound(sum+Le ,sum+Rt ,sum[j]+m) - sum ;
                ans = max(ans ,mx - j) ;
            }
            i = Rt ;
        }
        cout<


  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved