HDU 5360 Hiking(线段树)

来源:互联网 发布:数据库原理试题 编辑:程序博客网 时间:2024/05/22 17:19

题意:

n个人接受邀请的条件是已经接受邀请的人数区间在l[i] , r[i]
问怎样设置邀请顺序能使得接受邀请的人数最多

解析:

先对区间从小按照右边界从小到大排序,如果右边界相同,再按照左边界从小到大排序。因为右边界越小优先级越高,左边界同理。
如果用优先队列,是算出选择当前人选择哪个区间是最优的。
由于本人不会两个条件的优先队列,所以只能换了一种写法来写。
于是我想到了线段树,逆向思维,利用线段树来维护这个区间选择哪个人是最优的。
选择完这个人之后,将这个人置为无穷大并维护线段树。
并记录下当前人,选择的是哪个区间。

my code

#include <cstdio>#include <cstring>#include <algorithm>#define ls o<<1#define rs o<<1|1#define lson ls, L, M#define rson rs, M+1, Rusing namespace std;const int INF = 0x3f3f3f3f;const int N = 1e5 + 10;int n;struct Node {    int l, r, id;} node[N];int minv[N<<2];inline void pushUp(int o) {    minv[o] = min(minv[ls], minv[rs]);}void build(int o, int L, int R) {    if(L == R) {        minv[o] = L;        return ;    }    int M = (L + R)/2;    build(lson);    build(rson);    pushUp(o);}int query(int o, int L, int R, int ql, int qr) {    if(ql <= L && R <= qr) return minv[o];    int M = (L + R)/2, ret = INF;    if(ql <= M) ret = min(ret, query(lson, ql, qr));    if(qr > M) ret = min(ret, query(rson, ql, qr));    return ret;}void modify(int o, int L, int R, int pos) {    if(L == R) {        minv[o] = INF;        return ;    }    int M = (L + R)/2;    if(pos <= M) modify(lson, pos);    else modify(rson, pos);    pushUp(o);}int ans[N], res[N];bool vis[N];bool cmp(Node a, Node b) {    if(a.r != b.r)        return a.r < b.r;    return a.l < b.l;}int main() {    int T;    scanf("%d", &T);    while(T--) {        scanf("%d", &n);        for(int i = 1; i <= n; i++) {            scanf("%d", &node[i].l);            node[i].id = i;        }        for(int i = 1; i <= n; i++)            scanf("%d", &node[i].r);        sort(node+1, node+n+1, cmp);        memset(ans, -1, sizeof(ans));        memset(vis, false, sizeof(vis));        build(1, 0, n);        for(int i = 1; i <= n; i++) {            int ret = query(1, 0, n, node[i].l, node[i].r);            if(ret != INF) {                ans[ret] = node[i].id;                modify(1, 0, n, ret);            }        }        int tot = 0;        for(int i = 0; i < n; i++) {            if(ans[i] != -1) {                res[tot++] = ans[i];                vis[ans[i]] = true;            }else break;        }        printf("%d\n", tot);        for(int i = 1; i <= n; i++) {            if(!vis[i]) res[tot++] = i;        }        for(int i = 0; i < tot; i++) {            printf("%d", res[i]);            if(i == tot-1) puts("");            else putchar(' ');        }    }    return 0;}
1 0
原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 君子兰发的小苗怎么办 蔷薇光长枝条不开花怎么办 牡丹发芽又干了怎么办 擦皮炎平后皮肤变黑怎么办 误喝发霉的咖啡渣怎么办 狗吃了速溶咖啡怎么办 咖啡机放豆的地方进水怎么办 干吃咖啡粉上瘾怎么办 去良友花艺住宿怎么办 充气娃娃放了气怎么办 煮杜鹃根没有锅怎么办 淘宝店卖鲜花被买家拒收货怎么办 执业医师电子注册忘记密码怎么办 怀孕吃了油炸的怎么办 百合长得太高怎么办 百合的杆没了怎么办 百合花长得太细怎么办 沙漠玫瑰的花苞打不开怎么办 鲜切花 较小的花苞怎么办 大棚玫瑰苗水大涝的不长怎么办 鲜花买回来蔫了怎么办 喝玫瑰醋上火了怎么办 插在花泥上的花怎么办 插的花蔫了怎么办 紫睡莲的茎软了怎么办 家养的荷花烂叶怎么办 家养的荷花叶老是枯萎怎么办 新买的绣球蔫了怎么办 绣球花被太阳晒阉了怎么办 羊肉香精放多了怎么办 被飞机防腐剂弄到皮肤怎么办 狗吃了脱氧保鲜剂呕吐怎么办 小孩误吃试纸了保鲜剂怎么办 狗狗把保鲜剂吃了怎么办 小孩吃了防潮珠怎么办 狗吃了防潮剂怎么办 洋桔梗有点烂根怎么办 变色球花枯萎了怎么办 桔梗花叶子蔫了怎么办 洋桔梗头垂下来怎么办 洋桔梗花容易折断怎么办