BZOJ 1828: [Usaco2010 Mar]balloc 农场分配

Description

Input

第1行:两个用空格隔开的整数:N和M * 第2行到N+1行:第i+1行表示一个整数C_i * 第N+2到N+M+1行: 第i+N+1行表示2个整数 A_i和B_i

Output

* 第一行: 一个整数表示最多能够被满足的要求数

题解:

将请求按右端点排序,然后依次添加,用线段树判断是否能添加,不能的放弃。统计个数即可。

代码:

#include<cstdio>
#include<cstring>
#include<algorithm>
//by zrt
//problem:
using namespace std;
int minn[100005*4];
int mark[100005*4];
int c[100005];
void pd(int o){
    if(mark[o]){
        minn[o<<1]-=mark[o];
        minn[o<<1|1]-=mark[o];
        mark[o<<1]+=mark[o];
        mark[o<<1|1]+=mark[o];
        mark[o]=0;
    }
}
void insert(int o,int l,int r,int L,int R){
    if(l==L&&r==R){
        mark[o]+=1;
        minn[o]-=1;
        return;
    }else{
        pd(o);
        int m=(L+R)>>1;
        if(r<=m) insert(o<<1,l,r,L,m);
        else if(l>m) insert(o<<1|1,l,r,m+1,R);
        else insert(o<<1,l,m,L,m),insert(o<<1|1,m+1,r,m+1,R);
        minn[o]=min(minn[o<<1],minn[o<<1|1]);
    }
}
void build(int o,int l,int r){
    if(l==r){
        minn[o]=c[l];
    }else{
        int m=(l+r)>>1;
        build(o<<1,l,m);
        build(o<<1|1,m+1,r);
        minn[o]=min(minn[o<<1],minn[o<<1|1]);
    }
}
bool ok;
void ask(int o,int l,int r,int L,int R){
    if(l==L&&r==R)  ok&=(minn[o]>0);
    else{
        pd(o);
        int m=(L+R)>>1;
        if(r<=m) ask(o<<1,l,r,L,m);
        else if(l>m) ask(o<<1|1,l,r,m+1,R);
        else ask(o<<1,l,m,L,m),ask(o<<1|1,m+1,r,m+1,R);
    }
}
struct node{
    int a,b;
}q[100005];
bool cmp(node a,node b){
    return a.b<b.b;
}
int main(){
    #ifdef LOCAL
    freopen("in.txt","r",stdin);
    freopen("out.txt","w",stdout);
    #endif
    int n,m;
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++){
        scanf("%d",&c[i]);
    }
    for(int i=0;i<m;i++){
        scanf("%d%d",&q[i].a,&q[i].b);
    }
    sort(q,q+m,cmp);
    int ans=0;
    build(1,1,n);
    for(int i=0;i<m;i++){
        ok=1;ask(1,q[i].a,q[i].b,1,n);
        if(ok){
            ans++;
            insert(1,q[i].a,q[i].b,1,n);
        }
    }
    printf("%d\n",ans);
    return 0;
}
时间: 2024-10-24 15:52:25

BZOJ 1828: [Usaco2010 Mar]balloc 农场分配的相关文章

BZOJ1828 [Usaco2010 Mar]balloc 农场分配

直接贪心,我们把线段按照右端点从小到大排序,然后一个个尝试插入即可... 来证明贪心的正确性: 不妨设贪心得到的答案集合为$S$,最优解的答案集合为$T$ 若$S$不是最优解,那么$S \not= T$,不妨设按照右端点排序后,第一个不同的位置为$i$ 则$S_i \not= T_i$,分情况讨论: (1)$S_i$的左端点在$T_i$的右端点后,由于贪心的步骤这是不可能的 (2)$S_i$的右端点在$T_i$的右端点之前: (2.1)$S_i$的右端点在$T_i$的左端点之前,即$S_i$.$

BZOJ_1828_[Usaco2010 Mar]balloc 农场分配_线段树

Description Input 第1行:两个用空格隔开的整数:N和M * 第2行到N+1行:第i+1行表示一个整数C_i * 第N+2到N+M+1行: 第i+N+1行表示2个整数 A_i和B_i Output * 第一行: 一个整数表示最多能够被满足的要求数 Sample Input 5 4 1 3 2 1 3 1 3 2 5 2 3 4 5 Sample Output 3 分析:把每头牛按右端点升序排序,然后能插就插,我们需要维护一下这段区间剩余空间的最小值,如果最小值大于0说明能放进去.

【树形DP/搜索】BZOJ 1827: [Usaco2010 Mar]gather 奶牛大集会

1827: [Usaco2010 Mar]gather 奶牛大集会 Time Limit: 1 Sec  Memory Limit: 64 MBSubmit: 793  Solved: 354[Submit][Status][Discuss] Description Bessie正在计划一年一度的奶牛大集会,来自全国各地的奶牛将来参加这一次集会.当然,她会选择最方便的地点来举办这次集会.每个奶牛居住在 N(1<=N<=100,000) 个农场中的一个,这些农场由N-1条道路连接,并且从任意一个

BZOJ 1827: [Usaco2010 Mar]gather 奶牛大集会( dp + dfs )

选取任意一个点为root , size[ x ] 表示以 x 为根的子树的奶牛数 , dp一次计算出size[ ] && 选 root 为集会地点的不方便程度 . 考虑集会地点由 x 点向它的子节点 son 转移 , 那么以 son 为集会地点比以 x 为集会地点要多 dist( x , son ) * ( tot - size[ x ] ) - dist( x , son ) * size[ x ] = dist( x , son ) * ( tot - 2 * size[ x ] )

BZOJ 1827: [Usaco2010 Mar]gather 奶牛大集会

Description Bessie正在计划一年一度的奶牛大集会,来自全国各地的奶牛将来参加这一次集会.当然,她会选择最方便的地点来举办这次集会.每个奶牛居住在 N(1<=N<=100,000) 个农场中的一个,这些农场由N-1条道路连接,并且从任意一个农场都能够到达另外一个农场.道路i连接农场A_i和B_i(1 <= A_i <=N; 1 <= B_i <= N),长度为L_i(1 <= L_i <= 1,000).集会可以在N个农场中的任意一个举行.另外

BZOJ 1827 [Usaco2010 Mar]gather 奶牛大集会(树形DP)

[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1827 [题目大意] 给出一棵有点权和边权的树, 请确定一个点,使得每个点到这个点的距离乘上该点乘积的总和最小. [题解] 定1为根,我们先计算当这个点为1的时候的值,同时记录每个子树的size 之后我们再做一遍dfs,计算出每个点作为中心时候的答案 我们发现当一个点从父节点往子节点移动的时候 对于答案的变化是ans+=(size[1]-2*size[x])*len(fx->x) 所以

【BZOJ 1827】 [Usaco2010 Mar]gather 奶牛大集会

1827: [Usaco2010 Mar]gather 奶牛大集会 Time Limit: 1 Sec  Memory Limit: 64 MB Submit: 722  Solved: 314 [Submit][Status][Discuss] Description Bessie正在计划一年一度的奶牛大集会,来自全国各地的奶牛将来参加这一次集会.当然,她会选择最方便的地点来举办这次集会.每个奶牛居住在 N(1<=N<=100,000) 个农场中的一个,这些农场由N-1条道路连接,并且从任意

DP经典 BZOJ 1584: [Usaco2009 Mar]Cleaning Up 打扫卫生

BZOJ 1584: [Usaco2009 Mar]Cleaning Up 打扫卫生 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 419  Solved: 278 Description 有N头奶牛,每头那牛都有一个标号Pi,1 <= Pi <= M <= N <= 40000.现在Farmer John要把这些奶牛分成若干段,定义每段的不河蟹度为:若这段里有k个不同的数,那不河蟹度为k*k.那总的不河蟹度就是所有段的不河蟹度的总和

【BZOJ】1827: [Usaco2010 Mar]gather 奶牛大集会

[算法]树型DP [题解] 两遍DFS,第一次得到所有节点子树的路径和,第二次给出除了该子树外其它部分的路径和,时时计算答案. long long!!! #include<cstdio> #include<cstring> #include<algorithm> #include<cctype> #define ll long long using namespace std; const int maxn=100010; struct edge{int v