一攬子計劃

咕了!

liaoliao的BZOJ題表,一題題作(雖然有些題作過)。。

可是靖爺都說這些題沒有意思,沒有養分%%%

                  完成量(21/139)

BZOJ1226: [SDOI2009]學校食堂Dining

  由於每一個人的B值不超過7,因此上一個吃飯的人應該在這我的前面不超過7,後面也不超過7的位置上node

  狀壓DP,設f[i][j][k]爲前i-1我的都已經吃完飯了,第i到i+7我的的狀態爲j,最後一個吃飯的人爲第i+k我的數組

  而後就能夠轉移ide

  對於j&1==1,就是說第i我的已經吃了飯了,那麼f[i+1][j>>1][k-1]=f[i][j][k],二者的意義是同樣的spa

  對於j&1==0,那麼就枚舉應看成爲最後一個吃飯的人的位置i+p,f[i][j^(1<<p)][p]=f[i][j][k]+cal(i+k,i+p))3d

  而後由於k有可能爲負,因此總體的k+8存入數組code

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
using namespace std;
int T[1100],B[1100];
int f[1100][310][21];
int cal(int x,int y){return (x==0||y==0)?0:T[x]^T[y];}
int main()
{
    int cas;
    scanf("%d",&cas);
    while(cas--)
    {
        int n;
        scanf("%d",&n);
        for(int i=1;i<=n;i++) scanf("%d%d",&T[i],&B[i]);
        memset(f,63,sizeof(f));
        f[1][0][7]=0;
        for(int i=1;i<=n;i++)
        {
            for(int j=0;j<(1<<8);j++)
            {
                for(int k=0;k<=15;k++)
                {
                    if(f[i][j][k]==f[0][0][0]) continue;
                    if(j&1) f[i+1][j>>1][k-1]=min(f[i+1][j>>1][k-1],f[i][j][k]);
                    else
                    {
                        int mmax=1<<30;
                        for(int p=0;p<=7;p++)
                        {
                            if((j&(1<<p))==0)
                            {
                                if(i+p>mmax) break;
                                mmax=min(mmax,i+p+B[i+p]);
                                f[i][j^(1<<p)][p+8]=min(f[i][j^(1<<p)][p+8],f[i][j][k]+cal(i+k-8,i+p));
                            }
                        }
                    }
                }
            }
        }
        int ans=1<<30;
        for(int k=0;k<=8;k++) ans=min(f[n+1][0][k],ans);
        printf("%d\n",ans);
    }
    return 0;
}
BZOJ1226

BZOJ1260: [CQOI2007]塗色paint

  以前作過,仍是很容易作得出來blog

  區間DP,轉移的話排序

  f[i][j]表示i到j染成目標木板的i到j的最少塗色次數繼承

  轉移方程有兩個:隊列

  1.當st[i]==st[j]時,f[i][j]=min(min(f[i][j-1],f[i+1][j]),f[i+1][j-1]+1),由於若是左右兩端都爲一樣顏色,那麼第一次就將整塊木板塗成一樣顏色便可以作到

  2.常規找斷點,f[i][j]=min(f[i][j],f[i][k]+f[k+1][j])

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<cstdlib>
using namespace std;
int f[51][51];
char st[51];
int main()
{
    scanf("%s",st+1);int n=strlen(st+1);
    memset(f,63,sizeof(f));
    for(int i=1;i<=n;i++) f[i][i]=1;
    for(int k=2;k<=n;k++)
    {
        for(int i=1;i+k-1<=n;i++)
        {
            if(st[i]==st[i+k-1]) f[i][i+k-1]=min(min(f[i][i+k-2],f[i+1][i+k-1]),f[i+1][i+k-2]+1);
            for(int j=i;j<i+k-1;j++)
            {
                f[i][i+k-1]=min(f[i][i+k-1],f[i][j]+f[j+1][i+k-1]);
            }
        }
    }
    printf("%d\n",f[1][n]);
    return 0;
}
BZOJ1260

BZOJ1296: [SCOI2009]粉刷匠

  一開始看錯題了,還覺得能夠豎着粉刷

  先枚舉每一塊木板,而後在這塊木板上求f[i][j],表示這塊木板上前i個格子粉刷j次所能獲得的最多正確數

  而後對ans[i]表示全部木板粉刷i次所能獲得的最多正確數,把f[i][j]插入ans數組中作揹包就好了

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
#define INF 1<<30
using namespace std;
char st[51][51];
int f[51][51];//f[i][j]表示當前木板的前i個刷k次所能獲得的最多正確數
int mx[51];
int sum[51];
int ans[3100];//ans[i]表示全部木板粉刷i次最多正確數 
int main()
{
    int n,m,T;
    scanf("%d%d%d",&n,&m,&T);
    for(int i=1;i<=n;i++) scanf("%s",st[i]+1);
    memset(ans,-63,sizeof(ans));
    ans[0]=0;int s=0;
    for(int p=1;p<=n;p++)
    {
        memset(f,-63,sizeof(f));
        sum[0]=0;
        for(int i=1;i<=m;i++) sum[i]=sum[i-1]+st[p][i]-'0';
        for(int i=0;i<=m;i++) f[i][0]=0;
        for(int i=1;i<=min(m,T);i++)
        {
            mx[i]=-(1<<30);
            for(int j=i;j<=m;j++)
            {
                for(int k=i-1;k<j;k++)
                {
                    if(f[k][i-1]==f[0][1]) continue;
                    f[j][i]=max(f[j][i],f[k][i-1]+max(j-k-sum[j]+sum[k],sum[j]-sum[k]));
                }
                mx[i]=max(f[j][i],mx[i]);
            }
        }
        for(int j=T;j>=1;j--)
        {
            for(int i=1;i<=min(j,m);i++)
            {
                if(ans[j-i]!=ans[T+1]) ans[j]=max(ans[j-i]+mx[i],ans[j]),s=max(ans[j],s);
            }
        }
    }
    printf("%d\n",s);
    return 0;
}
BZOJ1296

BZOJ1560: [JSOI2009]火星藏寶圖

  結論貪心+DP

  由於每一個點的固定收益爲正,並且兩點的距離是歐幾里得距離的平方,因此能夠獲得一個結論:

  假設要從A->C,若是能A->B->C,確定A->B->C是更優的

  那咱們就將點先按x爲第一關鍵字,y爲第二關鍵字排序

  設f[i]爲從第1個點到第i個點的最大收益,顯然咱們轉移的時候只需找到前i-1個點所在每一列的最下面的點就能夠了

  這樣複雜度就是O(mn),感受卡過去了

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
using namespace std;
struct sit{int x,y,v;}p[210000];
bool cmp(sit n1,sit n2){return n1.x==n2.x?n1.y<n2.y:n1.x<n2.x;}
int f[210000],pos[1100];
int dis(sit n1,sit n2){return (n1.x-n2.x)*(n1.x-n2.x)+(n1.y-n2.y)*(n1.y-n2.y);}
int main()
{
    int n,m;
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++) scanf("%d%d%d",&p[i].x,&p[i].y,&p[i].v);
    sort(p+1,p+n+1,cmp);
    memset(f,-63,sizeof(f));
    f[1]=p[1].v;pos[1]=1;
    int x=1;
    for(int i=2;i<=n;i++)
    {
        for(int j=1;j<=p[i].y;j++)
        {
            if(pos[j]!=0)
            {
                f[i]=max(f[i],f[pos[j]]-dis(p[pos[j]],p[i]));
            }
        }
        f[i]+=p[i].v;
        pos[p[i].y]=i;
    }
    printf("%d\n",f[n]);
    return 0;
}
BZOJ1560

BZOJ1806: [Ioi2007]Miners 礦工配餐

  直接DP就行了

  設f[i][t1][t2][t3][t4]爲當前已經送了i個食物,且最近兩次給第一個礦洞送了第t1和t2種食物(t1比t2更早送),給第二個礦洞送了第t3和t4種食物(t3比t4更早送)能獲得的最大收益

  假如t1,t2,t3,t4爲0,表示當前沒有送那麼多的餐

  轉移想一想就會了

  而後這題卡空間。。不要緊,滾動數組一下就行了

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
using namespace std;
char st[110000];
int f[2][4][4][4][4];
int a[110000];
int main()
{
    int n;
    scanf("%d",&n);
    scanf("%s",st+1);
    for(int i=1;i<=n;i++)
    {
        if(st[i]=='M') a[i]=1;
        if(st[i]=='F') a[i]=2;
        if(st[i]=='B') a[i]=3;
    }
    memset(f,-1,sizeof(f));
    f[0][0][0][0][0]=0;
    int now=0,last=1;
    int ans=0;
    for(int i=1;i<=n;i++)
    {
        now^=1;last^=1;
        for(int t1=0;t1<=3;t1++)
        {
            for(int t2=0;t2<=3;t2++)
            {
                int d1=0;
                if(t1!=0&&t2==0)
                {
                    d1=1;if(t1!=a[i]) d1=2;
                }
                else if(t1==0&&t2!=0)
                {
                    d1=1;if(t2!=a[i]) d1=2;
                }
                else if(t1!=0&&t2!=0)
                {
                    if(t1==t2)
                    {
                        d1=1;if(t1!=a[i]) d1=2;
                    }
                    else
                    {
                        d1=2;if(t1!=a[i]&&t2!=a[i]) d1=3;
                    }
                }
                else d1=1;
                for(int t3=0;t3<=3;t3++)
                {
                    for(int t4=0;t4<=3;t4++)
                    {
                        int d2=0;
                        if(t3!=0&&t4==0)
                        {
                            d2=1;if(t4!=a[i]) d2=2;
                        }
                        else if(t3==0&&t4!=0)
                        {
                            d2=1;if(t4!=a[i]) d2=2;
                        }
                        else if(t3!=0&&t4!=0)
                        {
                            if(t3==t4)
                            {
                                d2=1;if(t3!=a[i]) d2=2;
                            }
                            else
                            {
                                d2=2;if(t3!=a[i]&&t4!=a[i]) d2=3;
                            }
                        }
                        else d2=1;
                        if(f[last][t1][t2][t3][t4]!=-1)
                        {
                            f[now][t2][a[i]][t3][t4]=max(f[now][t2][a[i]][t3][t4],f[last][t1][t2][t3][t4]+d1);
                            f[now][t1][t2][t4][a[i]]=max(f[now][t1][t2][t4][a[i]],f[last][t1][t2][t3][t4]+d2);
                            ans=max(ans,max(f[now][t2][a[i]][t3][t4],f[now][t1][t2][t4][a[i]]));
                        }
                    }
                }
            }
        }
    }
    printf("%d\n",ans);
    return 0;
}
BZOJ1806

BZOJ2121: 字符串游戲

  這個DP有點神了。。

  由於最後選擇刪除的多個區間,要麼是包含關係,要麼就相離關係,不可能存在區間相交的狀況

  因此設f1[l][r][i][j]爲l到r所構成的字符串刪掉若干區間後是否可以恰好構成第i個子串的前j個字符

  f2[l][r]表示可否將l到r所構成的字符串都刪掉

  轉移就要麼就日後加一個字符,要麼就找前面後面是否可以繼承就好了

  O(|L|3*|S|*|p|),居然是能過的複雜度。。

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<algorithm>
#define mes(x,y) memset(x,y,sizeof(x))
using namespace std;
bool f1[210][210][41][31],f2[210][210];
char st[210];
char cc[41][31];
int h[151],len[210];
int main()
{
    scanf("%s",st+1);
    len[0]=strlen(st+1);
    int n;scanf("%d",&n);
    for(int i=1;i<=n;i++) scanf("%s",cc[i]+1),len[i]=strlen(cc[i]+1);
    mes(f1,false);mes(f2,false);
    for(int i=1;i<=len[0];i++)
    {
        for(int j=1;j<=n;j++)
        {
            if(cc[j][1]==st[i]) f1[i][i][j][1]=true;
            if(f1[i][i][j][len[j]]==true) f2[i][i]=true;
        }
    }
    for(int k=2;k<=len[0];k++)
    {
        for(int l=1;l<=len[0]-k+1;l++)
        {
            int r=l+k-1;
            for(int i=1;i<=n;i++)
            {
                for(int j=1;j<=min(len[i],k);j++)
                {
                    if(st[r]==cc[i][j]) f1[l][r][i][j]|=f1[l][r-1][i][j-1];
                    for(int p=l;p<=r;p++)
                    {
                        if(f2[l][p]==true) f1[l][r][i][j]|=f1[p+1][r][i][j];
                        if(f2[p][r]==true) f1[l][r][i][j]|=f1[l][p-1][i][j];
                    }
                }
                if(f1[l][r][i][len[i]]==true) f2[l][r]=true;
            }
        }
    }
    mes(h,63);h[0]=0;
    for(int i=1;i<=len[0];i++)
    {
        h[i]=h[i-1]+1;
        for(int j=1;j<=i;j++) if(f2[j][i]==true) h[i]=min(h[i],h[j-1]);
    }
    printf("%d\n",h[len[0]]);
    return 0;
}
BZOJ2121

BZOJ2302: [HAOI2011]Problem c

  其實想一想就會發現與順序是沒有什麼關係的

  由於當一種方案成立,確定會使得全部編號<=i的人數會>=i,這個想一想應該就能懂

  而後就能夠寫DP方程,設f[i][j]爲有j我的的編號<=i的方案數

  轉移就組合數一下就好了,對於無解的狀況就先判斷掉

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#define INF 1<<30
#define mes(x,y) memset(x,y,sizeof(x))
#define Maxn 310
using namespace std;
typedef long long LL;
LL Mod;
LL mul(LL a,LL b){return a*b%Mod;}
LL add(LL a,LL b){return (a+b)%Mod;}
LL p_mod(LL a,LL b)
{
    LL ans=1;
    while(b!=0)
    {
        if(b%2==1) ans=mul(ans,a);
        a=mul(a,a);b>>=1;
    }
    return ans;
}
int s[Maxn];
LL f[Maxn][Maxn];//f[i][j]表示有j我的的編號<=i的方案數,顯然j>=i
LL C[Maxn][Maxn];
int main()
{
    int T;
    scanf("%d",&T);
    while(T--)
    {
        int n,m;scanf("%d%d%lld",&n,&m,&Mod);
        mes(C,0);C[0][0]=1;
        for(int i=1;i<=300;i++)
        {
            C[i][0]=1;
            for(int j=1;j<=i;j++)
            {
                C[i][j]=add(C[i-1][j-1],C[i-1][j]);
            }
        }
        mes(s,0);s[0]=n-m;
        int p,q;
        for(int i=1;i<=m;i++) scanf("%d%d",&p,&q),s[q]++;
        bool bk=false;
        for(int i=1;i<=n;i++)
        {
            s[i]+=s[i-1];
            if(s[i]<i){bk=true;break;}
        }
        if(bk==true){printf("NO\n");continue;}
        printf("YES ");
        mes(f,0);f[0][0]=1;
        for(int i=1;i<=n;i++)
        {
            for(int j=i;j<=s[i];j++)
            {
                int d=s[i]-s[i-1];
                for(int k=d;k<=j-i+1;k++)
                {
                    f[i][j]=add(f[i][j],mul(f[i-1][j-k],C[s[i]-d-j+k][k-d]));
                }
            }
        }
        printf("%lld\n",f[n][n]);
    }
    return 0;
}
BZOJ2302

BZOJ2466: [中山市選2009]樹

  這個樹形DP仍是很好想的

  設f[x][i]表示第x個點爲i狀態時所須要的最少操做數

  對於狀態0表示沒按按鈕且沒亮,1表示沒按按鈕且亮了,2表示按了按鈕且沒亮,3表示按了按鈕且亮了

  而後在保證子節點必定亮的基礎上轉移就行了

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cstdio>
#include<algorithm>
#define mes(x,y) memset(x,y,sizeof(x))
#define Maxn 110
#define INF 1<<30
using namespace std;
typedef long long LL;
struct node{int x,y,next;}a[Maxn*2];int len,last[Maxn];
void ins(int x,int y){a[++len]=(node){x,y,last[x]};last[x]=len;}
int f[Maxn][5],n;
void dfs(int x,int fa)
{
    int f1[4];f[x][0]=0;f[x][3]=1;f[x][1]=f[x][2]=n+1;
    for(int k=last[x];k;k=a[k].next)
    {
        int y=a[k].y;
        if(y==fa) continue;
        dfs(y,x);
        f1[0]=f[x][0];f1[1]=f[x][1];f1[2]=f[x][2];f1[3]=f[x][3];
        f[x][0]=min(f1[0]+f[y][1],f1[1]+f[y][3]);
        f[x][1]=min(f1[1]+f[y][1],f1[0]+f[y][3]);
        f[x][2]=min(f1[2]+f[y][0],f1[3]+f[y][2]);
        f[x][3]=min(f1[3]+f[y][0],f1[2]+f[y][2]);
    }
}
int main()
{
    while(scanf("%d",&n)!=EOF)
    {
        if(n==0) break;
        int x,y;
        len=0;mes(last,0);
        for(int i=1;i<n;i++) scanf("%d%d",&x,&y),ins(x,y),ins(y,x);
        dfs(1,0);
        printf("%d\n",min(f[1][1],f[1][3]));
    }
    return 0;
}
BZOJ2466

BZOJ1787: [Ahoi2008]Meet 緊急集合

  作過

  轉化爲求三個點的LCA

  發現三個點兩兩之間的LCA中一定有兩個相同,並且其中不一樣的那個就是三個點的LCA

  而後直接作就行了

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<cstdlib>
using namespace std;
struct node
{
    int x,y,next;
}a[1100000];int len,last[510000];
int f[510000][25],dep[510000];
void ins(int x,int y)
{
    len++;
    a[len].x=x;a[len].y=y;
    a[len].next=last[x];last[x]=len;
}
int bin[25];
void dfs(int x,int fa)
{
    dep[x]=dep[fa]+1;
    f[x][0]=fa;
    for(int i=1;bin[i]<=dep[x];i++) f[x][i]=f[f[x][i-1]][i-1];
    for(int k=last[x];k;k=a[k].next)
    {
        int y=a[k].y;
        if(y==fa) continue;
        dfs(y,x);
    }
}
int LCA(int x, int y)
{
    if(dep[x]<dep[y]) swap(x,y);
    for(int i=20;i>=0;i--) if(dep[x]-dep[y]>=bin[i]) x=f[x][i];
    if(x==y) return x;
    for(int i=20;i>=0;i--) if(dep[x]>=(1<<i)&&f[x][i]!=f[y][i]) x=f[x][i],y=f[y][i];
    return f[x][0];
}
int n,m;
int dis(int x,int y)
{
    int lca=LCA(x,y);
    return dep[x]+dep[y]-2*dep[lca];
}
int main()
{
    scanf("%d%d",&n,&m);
    bin[0]=1;
    for(int i=1;i<=20;i++) bin[i]=bin[i-1]<<1;
    len=0;memset(last,0,sizeof(last));
    for(int i=1;i<n;i++)
    {
        int x,y;
        scanf("%d%d",&x,&y);
        ins(x,y);ins(y,x);
    }
    dfs(1,0);
    for(int i=1;i<=m;i++)
    {
        int x,y,z;
        scanf("%d%d%d",&x,&y,&z);
        int l1=LCA(x,y),l2=LCA(x,z),l3=LCA(y,z);
        if(l1==l2) printf("%d %d\n",l3,dis(x,l3)+dis(y,l3)+dis(z,l3));
        else if(l1==l3) printf("%d %d\n",l2,dis(x,l2)+dis(y,l2)+dis(z,l2));
        else if(l2==l3) printf("%d %d\n",l1,dis(x,l1)+dis(y,l1)+dis(z,l1));
    }
    return 0;
}
BZOJ1787

BZOJ1202: [HNOI2005]狡猾的商人

  帶權並查集就行了,判斷是否合法

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
using namespace std;
int fa[110],v[110];
int findfa(int x)
{
    if(fa[x]==x) return x;
    int f=fa[x];
    fa[x]=findfa(fa[x]);
    v[x]+=v[f];
    return fa[x];
}
int main()
{
    int T;
    scanf("%d",&T);
    while(T--)
    {
        memset(v,0,sizeof(v));
        int n,m;
        scanf("%d%d",&n,&m);
        for(int i=0;i<=n;i++) fa[i]=i;
        bool bk=true;
        for(int i=1;i<=m;i++)
        {
            int x,y,k;
            scanf("%d%d%d",&x,&y,&k);
            if(bk==false) continue;
            int fx=findfa(x-1),fy=findfa(y);
            if(fx==fy)
            {
                if(v[y]-v[x-1]!=k)
                {
                    bk=false;continue;
                }
            }
            else
            {
                fa[fy]=fx;
                v[fy]=k-v[y]+v[x-1];
            }
        }
        if(bk==false) printf("false\n");
        else printf("true\n");
    }
    return 0;
}
BZOJ1202

BZOJ2006: [NOI2010]超級鋼琴

  實際上就是求前k大子段和

  暴力作就是枚舉每一個左端點而後枚舉右端點

  然而咱們能夠預處理前綴和,用RMQ就可以獲得每一個點爲左端點時的最優右端點

  能夠發現以當前點爲左端點的次優勢必定不是右端點

  那麼咱們就用優先隊列一個個取,設四元組(p,x,y,d)爲當前以p點爲左端點,在x到y中選擇最優右端點,獲得的最大子段和爲d

  每次取出隊頂的時候,將隊頂分爲(p,x,j-1,d1),(p,j+1,y,d2)j爲以p點爲左端點,在x到y中的最優右端點

  而d1,d2就不用說了,就這樣取k次就行了

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
#include<queue>
#define mes(x,y) memset(x,y,sizeof(x))
#define Maxn 510000
using namespace std;
typedef long long LL;
struct node
{
    int p,l,r;LL d;
    friend bool operator < (node n1,node n2){return n1.d<n2.d;}
};
priority_queue<node> q;
LL a[Maxn],f[Maxn][21];
int Log[Maxn],p[Maxn][21];
LL mx(int l,int r)
{
    int t=Log[r-l+1];
    return max(f[l][t],f[r-(1<<t)+1][t]);
}
int P(int l,int r)
{
    int t=Log[r-l+1];
    if(f[l][t]>f[r-(1<<t)+1][t]) return p[l][t];
    else return p[r-(1<<t)+1][t];
}
int main()
{
    int n,k,L,R;
    scanf("%d%d%d%d",&n,&k,&L,&R);
    for(int i=1;i<=n;i++) scanf("%lld",&a[i]),a[i]+=a[i-1],f[i][0]=a[i],p[i][0]=i;
    Log[0]=-1;for(int i=1;i<=n;i++) Log[i]=Log[i/2]+1;
    for(int i=1;i<=20;i++)
    {
        for(int j=1;j<=n-(1<<i)+1;j++)
        {
            if(f[j][i-1]>f[j+(1<<(i-1))][i-1]) f[j][i]=f[j][i-1],p[j][i]=p[j][i-1];
            else f[j][i]=f[j+(1<<(i-1))][i-1],p[j][i]=p[j+(1<<(i-1))][i-1];
        }
    }
    for(int i=1;i<=n-L+1;i++)
    {
        int x=i+L-1,y=min(i+R-1,n);
        q.push((node){i,x,y,mx(x,y)-a[i-1]});
    }
    LL ans=0;
    for(int i=1;i<=k;i++)
    {
        node tno=q.top();q.pop();
        int p=P(tno.l,tno.r);
        if(tno.l<=p-1) q.push((node){tno.p,tno.l,p-1,mx(tno.l,p-1)-a[tno.p-1]});
        if(p+1<=tno.r) q.push((node){tno.p,p+1,tno.r,mx(p+1,tno.r)-a[tno.p-1]});
        ans+=tno.d;
    }
    printf("%lld\n",ans);
    return 0;
}
BZOJ2006

BZOJ2957: 樓房重建

  對於一個樓房被看到,能夠看做它與(0,0)的連線的斜率比前面全部的斜率都大

  那麼咱們能夠把每次操做都當成單點修改,而後求整段區間從第一個樓房開始斜率遞增所獲得的樓房數

  設mx爲每一個區間中最大的斜率,c爲每一個區間從左端點開始能看到的樓房數

  顯然咱們須要在修改的時候維護線段樹

  對於一段區間,顯然左子區間的c值的貢獻必定所有在整段區間的c值中,因此咱們只要對右子區間進行處理

  若是當前左子區間的最大值>=右子區間的最大值,那麼右子區間對整段區間是沒有貢獻的

  否則則在右子區間中找出以第一個大於左子區間的最大值的樓房,而後貢獻爲從這個樓房日後獲得的樓房數(包括這個樓房)

  就這樣,注意一下精度就能夠了

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<cstdlib>
#define eps 1e-10
using namespace std;
struct trnode
{
    int l,r,lc,rc,c;
    double mx;
}tr[210000];int trlen;
void bt(int l,int r)
{
    int now=++trlen;
    tr[now].l=l;tr[now].r=r;
    tr[now].lc=tr[now].rc=-1;
    tr[now].c=0;tr[now].mx=0.0;
    if(l<r)
    {
        int mid=(l+r)/2;
        tr[now].lc=trlen+1;bt(l,mid);
        tr[now].rc=trlen+1;bt(mid+1,r);
    }
}
int ans;
void findd(int now,double d)
{
    if(tr[now].l==tr[now].r){ans++;return ;}
    int lc=tr[now].lc,rc=tr[now].rc;
    if(tr[lc].mx<=d) findd(rc,d);
    else ans+=tr[now].c-tr[lc].c,findd(lc,d);
}
void follow(int now)
{
    int lc=tr[now].lc,rc=tr[now].rc;
    tr[now].c=tr[lc].c;tr[now].mx=tr[lc].mx;
    if(tr[rc].mx-tr[lc].mx>eps)
    {
        tr[now].mx=tr[rc].mx;
        ans=0;findd(rc,tr[lc].mx);
        tr[now].c+=ans;
    }
}
void change(int now,int x,double d)
{
    if(tr[now].l==tr[now].r)
    {
        tr[now].mx=d;
        tr[now].c=1;
        return ;
    }
    int lc=tr[now].lc,rc=tr[now].rc,mid=(tr[now].l+tr[now].r)/2;
    if(x<=mid) change(lc,x,d);
    else change(rc,x,d);
    follow(now);
}
int main()
{
    int n,m;
    scanf("%d%d",&n,&m);
    trlen=0;bt(1,n);
    for(int i=1;i<=m;i++)
    {
        int x,y;
        scanf("%d%d",&x,&y);
        change(1,x,double(y)/double(x));
        printf("%d\n",tr[1].c);
    }
    return 0;
}
BZOJ2957

BZOJ3289: Mato的文件管理

  莫隊,考慮加數減數時,樹狀數組求逆序對數對答案的影響便可

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
using namespace std;
typedef long long LL;
struct LS
{
    int d,id;
}s[51000];
struct qn
{
    int l,r,id;
    LL d;
}q[51000];
int bk[51000];
bool cmp(qn n1,qn n2)
{
    return bk[n1.l]==bk[n2.l]?n1.r<n2.r:bk[n1.l]<bk[n2.l];
}
bool cmpd(qn n1,qn n2)
{
    return n1.id<n2.id;
}
bool lsd(LS n1,LS n2)
{
    return n1.d<n2.d;
}
bool lsid(LS n1,LS n2)
{
    return n1.id<n2.id;
}
LL a[51000];
int lowbit(int x){return x&-x;}
int n;
LL getsum(int x)
{
    LL ans=0;
    while(x!=0)
    {
        ans+=a[x];
        x-=lowbit(x);
    }
    return ans;
}
void change(int x,LL c)
{
    while(x<=n)
    {
        a[x]+=c;
        x+=lowbit(x);
    }
}
LL t[51000];
int main()
{
    scanf("%d",&n);
    int block=int(sqrt(n));
    for(int i=1;i<=n;i++)
    {
        scanf("%d",&s[i].d);
        s[i].id=i;
        bk[i]=(i-1)/block+1;
    }
    sort(s+1,s+n+1,lsd);
    for(int i=1;i<=n;i++) s[i].d=i;
    sort(s+1,s+n+1,lsid);
    int m;
    scanf("%d",&m);
    for(int i=1;i<=m;i++)
    {
        scanf("%d%d",&q[i].l,&q[i].r);
        q[i].id=i;
    }
    sort(q+1,q+m+1,cmp);
    int l=1,r=0;
    LL ans=0;
    memset(a,0,sizeof(a));
    for(int i=1;i<=m;i++)
    {
        while(l<q[i].l)
        {
            if(l>r){l++;continue;}
            ans-=(r-l+1)-getsum(s[l].d);
            change(1,-1);
            if(s[l].d!=n) change(s[l].d+1,1);
            l++;
        }
        while(l>q[i].l)
        {
            if(l>r+1){l--;continue;}
            ans+=(r-l+1)-getsum(s[l-1].d);
            l--;
            change(1,1);
            if(s[l].d!=n) change(s[l].d+1,-1);
        }
        while(r<q[i].r)
        {
            if(l>r+1){r++;continue;}
            ans+=getsum(s[r+1].d);
            r++;
            change(1,1);
            if(s[r].d!=n) change(s[r].d+1,-1);
        }
        while(r>q[i].r)
        {
            if(l>r){r--;continue;}
            ans-=getsum(s[r].d)-1;
            change(1,-1);
            if(s[r].d!=n) change(s[r].d+1,1);
            r--;
        }
        q[i].d=ans;
    }
    sort(q+1,q+m+1,cmpd);
    for(int i=1;i<=m;i++) printf("%lld\n",q[i].d);
    return 0;
}
BZOJ3289

BZOJ3242: [Noi2013]快餐店

  顯然是一棵基環樹,並且對於一個最優快餐店的位置而言,它確定有一條環上的邊不會走

  那麼就枚舉刪邊,而後在環上操做,求出至關於當前刪邊後的樹的直徑,求出最小的

  而後再和原圖上原來的子樹的直徑比較就好了

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<algorithm>
#include<queue>
#define mes(x,y) memset(x,y,sizeof(x))
#define Maxn 110000
#define INF 1LL<<62
using namespace std;
struct node{int x,y,next;double d;}a[Maxn*2];int len,last[Maxn];
void ins(int x,int y,double d){a[++len]=(node){x,y,last[x],d};last[x]=len;}
int dfn[Maxn],id,fa[Maxn];
int e[Maxn],cnt,p[Maxn];
bool in[Maxn];
void ph(int x)
{
    dfn[x]=++id;
    for(int k=last[x];k;k=a[k].next)
    {
        int y=a[k].y;
        if(y==a[fa[x]].x) continue;
        if(dfn[y]!=0)
        {
            if(dfn[x]>dfn[y]) continue;
            e[++cnt]=((k-1)^1)+1;p[cnt]=x;in[x]=true;
            while(y!=x) e[++cnt]=fa[y],in[y]=true,p[cnt]=y,y=a[fa[y]].x;
        }
        else fa[y]=k,ph(y);
    }
}
double f1[Maxn],f2[Maxn],MX;//f1直徑,f2鏈 
void dfs(int x,int fa)
{
    for(int k=last[x];k;k=a[k].next)
    {
        int y=a[k].y;
        if(y==fa||in[y]==true) continue;
        dfs(y,x);
        f1[x]=max(f1[x],f2[x]+f2[y]+a[k].d);
        MX=max(MX,f1[x]);
        f2[x]=max(f2[x],f2[y]+a[k].d);
    }
}
double s[Maxn];
double L[Maxn],l[Maxn],R[Maxn],r[Maxn];
double ml[Maxn],mr[Maxn];
int to[Maxn];
int main()
{
    int n;
    scanf("%d",&n);
    len=0;mes(last,0);
    for(int i=1;i<=n;i++)
    {
        int x,y;double d;
        scanf("%d%d%lf",&x,&y,&d);
        ins(x,y,d);ins(y,x,d);
    }
    mes(in,false);
    cnt=id=0;ph(1);
    mes(f1,0);mes(f2,0);
    for(int i=1;i<=cnt;i++) dfs(p[i],0);
    reverse(e+1,e+cnt+1);
    to[1]=a[e[1]].x;s[1]=a[e[1]].d;
    for(int i=2;i<=cnt;i++) to[i]=a[e[i-1]].y,s[i]=a[e[i]].d;
    to[cnt+1]=to[1];
//    int head=1,tail=1;list[1]=1;
//    for(int i=2;i<=cnt*2+1;i++)
//    {
//        while(head<=tail&&i-list[head]>=cnt) head++;
//        int x=list[head];
//        if(i>=cnt) R[i]=f1[to[i]]+f1[to[x]]+s[i]-s[x];
//        while(head<=tail&&(f1[to[i]]-s[i])>=(f1[to[list[tail]]]-s[list[tail]])) tail--;
//        list[++tail]=i;
//    }
    L[0]=l[0]=0;
    double sum=0,mx=0;
//    for(int i=1;i<=cnt+1;i++) printf("%d ",to[i]);printf("\n");
//    for(int i=1;i<=cnt+1;i++) printf("%.1lf ",f2[to[i]]);printf("\n");
//    printf("L\n");
    for(int i=1;i<=cnt+1;i++)
    {
        sum+=s[i-1];
        L[i]=max(L[i-1],mx+f2[to[i]]+sum);
        l[i]=max(l[i-1],sum+f2[to[i]]);
        mx=max(mx,f2[to[i]]-sum);
//        printf("%d: L:%.1lf l:%.1lf ml:%.1lf\n",i,L[i],l[i],ml[i]);
    }
    R[cnt+1]=r[cnt+1]=mx=0;sum=0;
//    printf("R\n");
    for(int i=cnt;i>=1;i--)
    {
        sum+=s[i];
        R[i]=max(R[i+1],mx+f2[to[i]]+sum);
        r[i]=max(r[i+1],sum+f2[to[i]]);
        mx=max(mx,f2[to[i]]-sum);
//        printf("%d: R:%.1lf r:%.1lf mr:%.1lf\n",i,R[i],r[i],mr[i]);
    }
    double ans=INF;
    for(int i=1;i<=cnt;i++) ans=min(ans,max(max(L[i],R[i+1]),l[i]+r[i+1]));
    printf("%.1lf\n",max(ans,MX)/2.0);
    return 0;
}
BZOJ3242

BZOJ4170: 極光

  將位置i以及權值d,看成是座標(x,y),那麼每次詢問就是求與當前座標的曼哈頓距離<=k的點數

  顯然很差求,那麼就轉換一下座標系,將座標轉換爲(x+y,x-y),將曼哈頓距離轉換爲切比雪夫距離

  轉換爲矩陣以後就是CDQ分治裸題了

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<algorithm>
#define mes(x,y) memset(x,y,sizeof(x))
#define Maxn 110000
using namespace std;
struct ask
{
    int tp,x,y,t,f;/*type px py anst 1/-1*/
    friend bool operator < (ask n1,ask n2){return n1.x==n2.x?n1.tp<n2.tp:n1.x<n2.x;}
}Q[Maxn*4],tmp[Maxn*4];//1爲修改,2爲詢問
int a[Maxn*8];
int lowbit(int x){return x&-x;}
void change(int x,int d)
{
    while(x<=Maxn*4)
    {
        a[x]+=d;
        x+=lowbit(x);
    }
}
int getsum(int x)
{
    int ans=0;
    while(x!=0)
    {
        ans+=a[x];
        x-=lowbit(x);
    }
    return ans;
}
void clear(int x)
{
    while(x!=0)
    {
        if(a[x]==0) break;
        a[x]=0;x+=lowbit(x);
    }
}
int ans[Maxn];
void CDQ(int l,int r)
{
    if(l==r) return ;
    int mid=(l+r)/2;
    CDQ(l,mid);CDQ(mid+1,r);
    int p=l,q=mid+1,t=l;
    while(p<=mid&&q<=r)
    {
        if(Q[p]<Q[q])
        {
            if(Q[p].tp==1) change(Q[p].y,1);
            tmp[t++]=Q[p++];
        }
        else
        {
            if(Q[q].tp==2) ans[Q[q].t]+=Q[q].f*getsum(Q[q].y);
            tmp[t++]=Q[q++];
        }
    }
    while(p<=mid) tmp[t++]=Q[p++];
    while(q<=r)
    {
        if(Q[q].tp==2) ans[Q[q].t]+=Q[q].f*getsum(Q[q].y);
        tmp[t++]=Q[q++];
    }
    for(int i=l;i<=r;i++)
    {
        if(i<=mid) clear(Q[i].y);
        Q[i]=tmp[i];
    }
}
int nx[Maxn],ny[Maxn];
int main()
{
    int n,q;
    scanf("%d%d",&n,&q);
    for(int i=1;i<=n;i++)
    {
        int d;scanf("%d",&d);
        Q[i]=(ask){1,i+d+Maxn,i-d+Maxn,0,0};
        nx[i]=Q[i].x;ny[i]=Q[i].y;
    }
    int cnt=0,nn=n;
    for(int i=1;i<=q;i++)
    {
        char st[7];int x,d;
        scanf("%s%d%d",st+1,&x,&d);
        if(st[1]=='M')
        {
            Q[++n]=(ask){1,x+d+Maxn,x-d+Maxn,0,0};
            nx[x]=Q[n].x;ny[x]=Q[n].y;
        }
        else
        {
            ++cnt;
            Q[++n]=(ask){2,min(Maxn*4,nx[x]+d),min(Maxn*4,ny[x]+d),cnt,1};
            Q[++n]=(ask){2,max(nx[x]-d-1,1),min(Maxn*4,ny[x]+d),cnt,-1};
            Q[++n]=(ask){2,min(Maxn*4,nx[x]+d),max(ny[x]-d-1,1),cnt,-1};
            Q[++n]=(ask){2,max(nx[x]-d-1,1),max(ny[x]-d-1,1),cnt,1};
        }
    }
    mes(a,0);
    CDQ(1,n);
    //for(int i=1;i<=n;i++) printf("%d %d %d %d\n",tmp[i].tp,tmp[i].x,tmp[i].y,tmp[i].nn);printf("\n");
    for(int i=1;i<=cnt;i++) printf("%d\n",ans[i]);
    return 0;
}
BZOJ4170

BZOJ3781: 小B的詢問

  莫隊便可

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
using namespace std;
typedef long long LL;
int a[51000];
struct qn
{
    int l,r,id;
    LL d;
}q[51000];
int bk[51000];
bool cmp(qn n1,qn n2)
{
    return bk[n1.l]==bk[n2.l]?n1.r<n2.r:bk[n1.l]<bk[n2.l];
}
bool cmpd(qn n1,qn n2)
{
    return n1.id<n2.id;
}
LL sum[51000],ans;
void update(int x,int ad)
{
    ans-=sum[a[x]]*sum[a[x]];
    sum[a[x]]+=ad;
    ans+=sum[a[x]]*sum[a[x]];
}
int main()
{
    int n,m,k;
    scanf("%d%d%d",&n,&m,&k);
    int block=int(sqrt(n));
    for(int i=1;i<=n;i++)
    {
        scanf("%d",&a[i]);
        bk[i]=(i-1)/block+1;
    }
    for(int i=1;i<=m;i++)
    {
        scanf("%d%d",&q[i].l,&q[i].r);
        q[i].id=i;
    }
    sort(q+1,q+m+1,cmp);
    int l=1,r=0;
    ans=0;
    memset(sum,0,sizeof(sum));
    for(int i=1;i<=m;i++)
    {
        while(l<q[i].l){update(l,-1);l++;}
        while(l>q[i].l){update(l-1,1);l--;}
        while(r<q[i].r){update(r+1,1);r++;}
        while(r>q[i].r){update(r,-1);r--;}
        q[i].d=ans;
    }
    sort(q+1,q+m+1,cmpd);
    for(int i=1;i<=m;i++) printf("%lld\n",q[i].d);
    return 0;
}
BZOJ3781

BZOJ3809: Gty的二逼妹子序列

  莫隊便可

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
using namespace std;
struct node
{
    int l,r,a,b,id,d;
}q[1100000];
int s[110000];
int bl[1100],br[1100],belong[110000];
bool cmp1(node n1,node n2)
{
    if(belong[n1.l]<belong[n2.l]) return true;
    if(belong[n1.l]>belong[n2.l]) return false;
    if(belong[n1.l]==belong[n2.l])
    {
        if(n1.r<n2.r) return true;
        if(n1.r>n2.r) return false;
    }
    return false;
}
bool cmp2(node n1,node n2)
{
    return n1.id<n2.id;
}
int d[1100],sum[110000];
void add(int x)
{
    sum[x]++;
    if(sum[x]==1) d[belong[x]]++;
}
void del(int x)
{
    sum[x]--;
    if(sum[x]==0) d[belong[x]]--;
}
int solve(int x,int y)
{
    int bx=belong[x],by=belong[y];
    int ans=0;
    if(bx==by)
    {
        for(int i=x;i<=y;i++)
        {
            if(sum[i]>0) ans++;
        }
        return ans;
    }
    for(int i=bx+1;i<=by-1;i++) ans+=d[i];
    for(int i=x;i<=br[bx];i++) if(sum[i]>0) ans++;
    for(int i=bl[by];i<=y;i++) if(sum[i]>0) ans++;
    return ans;
}
int main()
{
    int n,m;
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++) scanf("%d",&s[i]);
    int block=int(sqrt(n));
    for(int i=1;i<=n;i++)
    {
        int t=(i-1)/block+1;
        belong[i]=t;
        if(bl[t]==0) br[t-1]=i-1,bl[t]=i;
    }
    br[belong[n]]=n;
    for(int i=1;i<=m;i++)
    {
        scanf("%d%d%d%d",&q[i].l,&q[i].r,&q[i].a,&q[i].b);
        q[i].id=i;
    }
    sort(q+1,q+m+1,cmp1);
    memset(sum,0,sizeof(sum));
    memset(d,0,sizeof(d));
    int l=1,r=0;
    for(int i=1;i<=m;i++)
    {
        while(l>q[i].l){l--;add(s[l]);}
        while(l<q[i].l){del(s[l]);l++;}
        while(r>q[i].r){del(s[r]);r--;}
        while(r<q[i].r){r++;add(s[r]);}
        q[i].d=solve(q[i].a,q[i].b);
    }
    sort(q+1,q+m+1,cmp2);
    for(int i=1;i<=m;i++) printf("%d\n",q[i].d);
    return 0;
}
BZOJ3809

BZOJ3133: [Baltic2013]ballmachine

  仍是挺裸的

  能夠發現實際上一開始將全部球扔下去以後獲得的序列是固定的,每次放球的時候,只要求出序列中最靠前的位置放球就好了

  並且能夠一個一個球放,由於輸入合法,刪的球數最多就是放的球數

  對於刪球,實際上求的就是當前位置向上的球數

  直接用兩個線段樹,再樹剖一下就好了

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
#include<vector>
#define mes(x,y) memset(x,y,sizeof(x))
#define Maxn 110000
using namespace std;
vector<int> a[Maxn];
void ins(int x,int y){a[x].push_back(y);}
int pt[Maxn],id,pre[Maxn],to[Maxn];
bool cmp(int x,int y){return pt[x]<pt[y];}
struct node{int l,r,lc,rc,c;}tr[Maxn*2],p[Maxn*2];int trlen,plen;
void bt(int t,int l,int r)
{
    if(t==0)
    {
        int now=++trlen;
        tr[now]=(node){l,r,-1,-1,0};
        if(l<r)
        {
            int mid=(l+r)/2;
            tr[now].lc=trlen+1;bt(0,l,mid);
            tr[now].rc=trlen+1;bt(0,mid+1,r);
        }
    }
    else
    {
        int now=++plen;
        p[now]=(node){l,r,-1,-1,0};
        if(l<r)
        {
            int mid=(l+r)/2;
            p[now].lc=plen+1;bt(1,l,mid);
            p[now].rc=plen+1;bt(1,mid+1,r);
        }
    }
}
int getspace(int now)
{
    if(tr[now].l==tr[now].r) return tr[now].l;
    int lc=tr[now].lc,rc=tr[now].rc;
    if(tr[lc].r-tr[lc].l+1>tr[lc].c) return getspace(lc);
    else return getspace(rc);
}
void inout(int now,int x,int d)
{
    if(tr[now].l==tr[now].r){tr[now].c=d;return ;}
    int lc=tr[now].lc,rc=tr[now].rc,mid=(tr[now].l+tr[now].r)/2;
    if(x<=mid) inout(lc,x,d);
    else inout(rc,x,d);
    tr[now].c=tr[lc].c+tr[rc].c;
}
int tot[Maxn],son[Maxn],fa[Maxn],dep[Maxn];
void dfs1(int x)//pre_tree_node
{
    son[x]=0;tot[x]=1;
    for(int i=0;i<a[x].size();i++)
    {
        int y=a[x][i];
        fa[y]=x;dep[y]=dep[x]+1;
        dfs1(y);
        tot[x]+=tot[y];
        if(tot[y]>tot[son[x]]) son[x]=y;
        pt[x]=min(pt[x],pt[y]);
    }
}
void dfs2(int x)
{
    sort(a[x].begin(),a[x].end(),cmp);
    for(int i=0;i<a[x].size();i++)
    {
        int y=a[x][i];
        dfs2(y);
    }
    pre[++id]=x;to[x]=id;
}
int top[Maxn],ys[Maxn],z,bk[Maxn];
void pre_tree_edge(int x,int tp)
{
    ys[x]=++z;top[x]=tp;bk[z]=x;
    if(son[x]!=0) pre_tree_edge(son[x],tp);
    for(int i=0;i<a[x].size();i++)
    {
        int y=a[x][i];
        if(y!=son[x]) pre_tree_edge(y,y);
    }
}
void change(int now,int x,int d)
{
    if(p[now].l==p[now].r){p[now].c=d;return ;}
    int lc=p[now].lc,rc=p[now].rc,mid=(p[now].l+p[now].r)/2;
    if(x<=mid) change(lc,x,d);
    else change(rc,x,d);
    p[now].c=p[lc].c+p[rc].c;
}
int getsum(int now,int l,int r)
{
    if(p[now].l==l&&p[now].r==r) return p[now].c;
    int lc=p[now].lc,rc=p[now].rc,mid=(p[now].l+p[now].r)/2;
    if(r<=mid) return getsum(lc,l,r);
    else if(l>mid) return getsum(rc,l,r);
    else return getsum(lc,l,mid)+getsum(rc,mid+1,r);
}
int sit,rt;
int getsit(int x)
{
    int tx=top[x],ans=0,pp;
    while(x!=0)
    {
        int d=getsum(1,ys[tx],ys[x]);
        ans+=d;
        if(d==ys[x]-ys[tx]+1) pp=tx,x=fa[tx],tx=top[x];
        else
        {
            if(d!=0) sit=bk[ys[x]-d+1];
            else sit=pp;
            return ans-1;
        }
    }
    sit=rt;
    return ans-1;
}
int main()
{
    int n,Q;
    scanf("%d%d",&n,&Q);
    for(int i=1;i<=n;i++)
    {
        int f;
        scanf("%d",&f);
        if(f==0) rt=i;
        else ins(f,i);
    }
    for(int i=1;i<=n;i++) pt[i]=i;
    fa[rt]=dep[rt]=0;dfs1(rt);
    id=0;dfs2(rt);
    z=0;pre_tree_edge(rt,rt);
    trlen=0;bt(0,1,n);
    plen=0;bt(1,1,z);
    while(Q--)
    {
        int op,x,sp;
        scanf("%d%d",&op,&x);
        if(op==1)
        {
            for(int i=1;i<=x;i++)
            {
                sp=getspace(1);
                inout(1,sp,1);
                change(1,ys[pre[sp]],1);
                if(i==x) printf("%d\n",pre[sp]);
            }
        }
        else
        {
            printf("%d\n",getsit(x));
            inout(1,to[sit],0);
            change(1,ys[sit],0);
        }
    }
    return 0;
}
BZOJ3133

BZOJ4662: Snow

  有個數組開小了,成功拍了兩個下午,真開心。。

  由於區間互不包含,並且左端點遞增,因此右端點也是遞增的

  那麼就先離散化,將全部都分紅一段一段的

  而後線段樹維護每一個人的清理區域,由於每一小段影響的人的編號也是一個區間

  而後每次找到最小的人就暴力掃一遍這我的清理區間的全部小段,逐一修改

  用並查集來加快掃,保證每一個小段只會被掃一次就好了

  一直覺得本身處理每一小段的影響時有細節沒作好才致使WA,結果是由於有一個數組沒開好(怪不得小數據拍不出來

#include<cstdio>
#include<cstdlib>
#include<cmath>
#include<algorithm>
#include<cstring>
#define Maxn 310000
#define mes(x,y) memset(x,y,sizeof(x))
#define INF 1LL<<62
using namespace std;
typedef long long LL;
struct LS{int x,z,id;}A[Maxn*4];
bool cmp1(LS n1,LS n2){return n1.x<n2.x;}
bool cmp2(LS n1,LS n2){return n1.id<n2.id;}
int to[Maxn*2];
struct node{int l,r,lc,rc,p;LL lz,c;}tr[Maxn*4];int trlen;
int lx(int x){return A[2*x-1].x;}
int rx(int x){return A[2*x].x;}
int lz(int x){return A[2*x-1].z;}
int rz(int x){return A[2*x].z;}
void bt(int l,int r)
{
    int now=++trlen;
    tr[now]=(node){l,r,-1,-1,0,0,INF};
    if(l==r) tr[now].c=rx(l)-lx(l),tr[now].p=l;
    else
    {
        int mid=(l+r)/2;
        tr[now].lc=trlen+1;bt(l,mid);
        tr[now].rc=trlen+1;bt(mid+1,r);
        int lc=tr[now].lc,rc=tr[now].rc;
        tr[now].c=min(tr[lc].c,tr[rc].c);
        tr[now].p=tr[lc].c<=tr[rc].c?tr[lc].p:tr[rc].p;
    }
}
void update(int now)
{
    int lc=tr[now].lc,rc=tr[now].rc;
    if(lc!=0)
    {
        tr[lc].c+=tr[now].lz;
        tr[lc].lz+=tr[now].lz;
    }
    if(rc!=0)
    {
        tr[rc].c+=tr[now].lz;
        tr[rc].lz+=tr[now].lz;
    }
    tr[now].lz=0;
}
void change(int now,int l,int r,LL c)
{
    if(tr[now].l==l&&tr[now].r==r)
    {
        tr[now].c+=c;
        tr[now].lz+=c;
        return ;
    }
    int lc=tr[now].lc,rc=tr[now].rc,mid=(tr[now].l+tr[now].r)/2;
    if(tr[now].lz!=0) update(now);
    if(r<=mid) change(lc,l,r,c);
    else if(l>mid) change(rc,l,r,c);
    else change(lc,l,mid,c),change(rc,mid+1,r,c);
    tr[now].c=min(tr[lc].c,tr[rc].c);
    tr[now].p=tr[lc].c<=tr[rc].c?tr[lc].p:tr[rc].p;
}
int fa[Maxn*4];
int findfa(int x)
{
    if(fa[x]!=x) fa[x]=findfa(fa[x]);
    return fa[x];
}
int L[Maxn*4],R[Maxn*4];
int main()
{
//    freopen("4662.in","r",stdin);
//    freopen("4662.out","w",stdout);
    int t,n;
    scanf("%d%d",&t,&n);
    for(int i=1;i<=n;i++) scanf("%d%d",&A[2*i-1].x,&A[2*i].x),A[2*i-1].id=2*i-1,A[2*i].id=2*i;
    sort(A+1,A+2*n+1,cmp1);
    int z=1;A[1].z=1;to[1]=A[1].x;
    for(int i=2;i<=2*n;i++)
    {
        if(A[i].x!=A[i-1].x) z++;
        A[i].z=z;to[z]=A[i].x;
    }
    sort(A+1,A+2*n+1,cmp2);
    trlen=0;bt(1,n);
    int p=1;
    for(int i=2;i<=z;i++)
    {
        if(lz(p)<=i-1&&i<=rz(p)) L[i]=p;
        else
        {
            while(p<=n&&(!(lz(p)<=i-1&&i<=rz(p))))
            {
                p++;
                if(lz(p)>=i) break;
            }
            if(!(lz(p)<=i-1&&i<=rz(p))) L[i]=z+1;
            else L[i]=p;
        }
    }
    p=n;
    for(int i=z;i>=2;i--)
    {
        if(lz(p)<=i-1&&i<=rz(p)) R[i]=p;
        else
        {
            while(p>=1&&(!(lz(p)<=i-1&&i<=rz(p))))
            {
                p--;
                if(rz(p)<=i-1) break;
            }
            if(!(lz(p)<=i-1&&i<=rz(p))) R[i]=0;
            else R[i]=p;
        }
    }
    for(int i=1;i<=z+1;i++) fa[i]=i;
    for(int i=1;i<=n;i++)
    {
        int sit=tr[1].p;printf("%d\n",sit);
//        if(tr[1].c<0)
//        {
//            printf("WA\n");
//            return 0;
//        }
        int x=findfa(lz(sit)+1);
        while(x<=rz(sit))
        {
            if(L[x]<=R[x]) change(1,L[x],R[x],-to[x]+to[x-1]);
            int y=findfa(x+1);fa[x]=y;
            x=y;
        }
        change(1,sit,sit,INF);
    }
    return 0;
}
BZOJ4662

BZOJ1237: [SCOI2008]配對

  貪心+DP便可

  先將兩個數組排序

  隨便畫畫圖就大概能夠知道,配對的兩個數相對位置不會>2

  因此設f[i]爲配i對數的最小值,而後從i-2,i-1,i繼承就好了

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#define Maxn 110000
#define mes(x,y) memset(x,y,sizeof(x))
using namespace std;
typedef long long LL;
int A[Maxn],B[Maxn];
LL f[Maxn];
int main()
{
    int n;
    scanf("%d",&n);
    for(int i=1;i<=n;i++) scanf("%d%d",&A[i],&B[i]);
    sort(A+1,A+n+1);sort(B+1,B+n+1);
    if(n==1&&A[1]==B[1]){printf("-1\n");return 0;}
    memset(f,63,sizeof(f));f[0]=0;
    for(int i=1;i<=n;i++)
    {
        if(A[i]!=B[i]) f[i]=min(f[i],f[i-1]+abs(A[i]-B[i]));
        if(i>=2&&A[i-1]!=B[i]&&A[i]!=B[i-1]) f[i]=min(f[i],f[i-2]+abs(A[i-1]-B[i])+abs(A[i]-B[i-1]));
        if(i>=3)
        {
            //i-2-->i-1 i-1-->i i-->i-2
            if(A[i-2]!=B[i-1]&&A[i-1]!=B[i]&&A[i]!=B[i-2]) f[i]=min(f[i],f[i-3]+abs(A[i-2]-B[i-1])+abs(A[i-1]-B[i])+abs(A[i]-B[i-2]));
            //i-1-->i-2 i-->i-1 i-2-->i
            if(A[i-1]!=B[i-2]&&A[i]!=B[i-1]&&A[i-2]!=B[i]) f[i]=min(f[i],f[i-3]+abs(A[i-1]-B[i-2])+abs(A[i]-B[i-1])+abs(A[i-2]-B[i]));
            //i-2-->i i-1-->i-1 i-->i-2
            if(A[i-2]!=B[i]&&A[i-1]!=B[i-1]&&A[i]!=B[i-2]) f[i]=min(f[i],f[i-3]+abs(A[i-2]-B[i])+abs(A[i-1]-B[i-1])+abs(A[i]-B[i-2]));
        }
    }
    printf("%lld\n",f[n]);
    return 0;
}
BZOJ1237

BZOJ1221: [HNOI2001] 軟件開發

  看題就很像費用流

  這題關鍵在於毛巾可以重複用

  現將天天拆成兩個點,一個入點,一個出點

  如下(x,y,c,d)表示x連向y,流量爲c,費用爲d

  i表示入點,i+n表示出點

  (st,i,n[i],0) (i+n,ed,n[i],0)至關於先給毛巾,而後再處理費用的問題

  (i+n,i+a+1,inf,fa) (i+n,i+b+1,inf,fb)將出點連向可以A或B消毒後的那一天的入點

  (i,i+1,inf,0)沒用完的毛巾能夠留到下一天

  (st,i+n,INF,f)能夠直接在這一天買毛巾

  這樣就能保證天天有n[i]條毛巾用,並且能循環利用了

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
#include<queue>
#define mes(x,y) memset(x,y,sizeof(x))
#define Maxn 1100
#define INF 1<<30
using namespace std;
struct node{int x,y,c,d,next,other;}a[Maxn*200];int last[Maxn*2],len;
void ins(int x,int y,int c,int d)
{
    int k1=++len,k2=++len;
    a[k1].x=x;a[k1].y=y;a[k1].d=d;a[k1].c=c;
    a[k1].next=last[x];last[x]=k1;
    a[k2].x=y;a[k2].y=x;a[k2].d=-d;a[k2].c=0;
    a[k2].next=last[y];last[y]=k2;
    a[k1].other=k2;
    a[k2].other=k1;
}
int p[Maxn],st,ed;
int d[Maxn*2],pos[Maxn*2];
bool v[Maxn*2];
queue<int> q;
bool spfa()
{
    memset(v,false,sizeof(v));v[st]=true;
    memset(d,63,sizeof(d));d[st]=0;
    q.push(st);
    bool bk=false;
    while(q.empty()==0)
    {
        int x=q.front();q.pop();
        for(int k=last[x];k;k=a[k].next)
        {
            int y=a[k].y;
            if(a[k].c>0&&d[y]>d[x]+a[k].d)
            {
                d[y]=d[x]+a[k].d;
                pos[y]=k;
                if(v[y]==false)
                {
                    v[y]=true;
                    if(y==ed) bk=true;
                    else q.push(y);
                }
            }
        }
        v[x]=false;
    }
    return bk;
}
int Flow()
{
    int ans=0;
    while(spfa())
    {
        int mmin=INF;
        for(int x=ed;x!=st;x=a[pos[x]].x)
        {
            mmin=min(a[pos[x]].c,mmin);
        }
        ans+=d[ed]*mmin;
        for(int x=ed;x!=st;x=a[pos[x]].x)
        {
            a[pos[x]].c-=mmin;
            a[a[pos[x]].other].c+=mmin;
        }
    }
    return ans;
}
int main()
{
    int n,A,B,f,fa,fb;
    scanf("%d%d%d%d%d%d",&n,&A,&B,&f,&fa,&fb);
    for(int i=1;i<=n;i++) scanf("%d",&p[i]);
    len=0;mes(last,0);
    st=2*n+1;ed=2*n+2;
    for(int i=1;i<=n;i++) ins(st,i,p[i],0),ins(st,i+n,INF,f);
    for(int i=1;i<=n-A-1;i++) ins(i,i+n+A+1,INF,fa);
    for(int i=1;i<=n-B-1;i++) ins(i,i+n+B+1,INF,fb);
    for(int i=1;i<n;i++) ins(i,i+1,INF,0);
    for(int i=1;i<=n;i++) ins(i+n,ed,p[i],0);
//  printf("%d\n",len/2);
//  for(int i=1;i<=len;i+=2) printf("%d %d %d %d\n",a[i].x,a[i].y,a[i].c,a[i].d);
    printf("%d\n",Flow());
    return 0;
}
BZOJ1221

BZOJ2698: 染色

相關文章
相關標籤/搜索