【BZOJ-4548&3658】小奇的糖果&Jabberwocky 双向链表 + 树状数组

4548: 小奇的糖果

Time Limit: 10 Sec  Memory Limit: 256 MB
Submit:
103  Solved: 47
[Submit][Status][Discuss]

Description

有 N 个彩色糖果在平面上。小奇想在平面上取一条水平的线段,并拾起它上方或下方的所有糖果。求出最多能够拾

起多少糖果,使得获得的糖果并不包含所有的颜色。

Input

包含多组测试数据,第一行输入一个正整数 T 表示测试数据组数。

接下来 T 组测试数据,对于每组测试数据,第一行输入两个正整数 N、K,分别表示点数和颜色数。

接下来 N 行,每行描述一个点,前两个数 x, y (|x|, |y| ≤ 2^30 - 1) 描述点的位置,最后一个数 z (1 ≤ z

k) 描述点的颜色。

对于 100% 的数据,N ≤ 100000,K ≤ 100000,T ≤ 3

Output

对于每组数据在一行内输出一个非负整数 ans,表示答案

Sample Input

1
10 3
1 2 3
2 1 1
2 4
2
3 5 3
4 4 2
5 1 2
6 3 1
6 7 1
7 2 3
9 4
2

Sample Output

5

HINT

Source

By Hzwer

3658: Jabberwocky

Time Limit: 20 Sec  Memory Limit: 1024 MB
Submit:
178  Solved: 73
[Submit][Status][Discuss]

Description

平面上有n个点,每个点有k种颜色中的一个。
你可以选择一条水平的线段获得在其上方或其下方的所有点,如图所示:

请求出你最多能够得到多少点,使得获得的点并不包含所有的颜色。

Input

包含多组测试数据,第一行输入一个数T表示测试数据组数。
   
接下来T组测试数据,对于每组测试数据,第一行输入两个数n,k,分别表示点的个数和颜色数。
   
接下来n行每行描述一个点,前两个数z,y(lxl,lyl≤2^32-1)描述点的位置,最后一个数z(1≤z≤K)描述点的颜色。

Output

对于每组数据输出一行,每行一个数ans,表示答案。

Sample Input

1
10 3
1 2 3
2 1 1
2 4
2
3 5 3
4 4 2
5 1 2
6 3 1
6 7 1
7 2 3
9 4
2

Sample Output

5

HINT

N<=100000,K<=100000,T<=3

Source

Solution

思路比较好的题,本来以为可以转化成某种扫描线,但是无果

首先,按y坐标排序,我们假定这条线无限低,这时候答案相当于相邻两个同种颜色的中间的最大点数

然后这条线向上,一次性删除一行

用双向链表维护一个点的前一个和后一个同种颜色的位置

删除的话,把这个点从树状数组和链表中都删除,每次删除,统计一下这个点之前相邻的和之后相邻的答案

然后把y全都赋成相反数,再做一遍即可得到下端的情况

Code

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
int read()
{
    int x=0,f=1; char ch=getchar();
    while (ch<‘0‘ || ch>‘9‘) {if (ch==‘-‘) f=-1; ch=getchar();}
    while (ch>=‘0‘ && ch<=‘9‘) {x=x*10+ch-‘0‘; ch=getchar();}
    return x*f;
}
#define MAXN 1000010
int T,N,K,ans;
struct PointNode{int x,y,c,id;}P[MAXN];
int ls[MAXN],tp,top;
struct lkList{int last,nxt,now;}lk[MAXN];
int C[MAXN];
struct BIT
{
    int tree[MAXN];
    void Init() {memset(tree,0,sizeof(tree));}
    int lowbit(int x) {return x&(-x);}
    void Add(int pos,int D)
        {
            for (int i=pos; i<=top+1; i+=lowbit(i))
                tree[i]+=D;
        }
    int Query(int pos)
        {
            int re=0;
            for (int i=pos; i; i-=lowbit(i))
                re+=tree[i];
            return re;
        }
    int Query(int L,int R) {if (R<L) return 0; else return Query(R)-Query(L-1);}
}bit;
bool cmpY(PointNode A,PointNode B) {return A.y<B.y;}
bool cmpX(PointNode A,PointNode B) {return A.x<B.x;}
void Solve()
{
    bit.Init();
    memset(C,0,sizeof(C));
    sort(P+1,P+N+1,cmpX); lk[N+1].now=top+1;
    for (int i=1; i<=N; i++) bit.Add(P[i].x,1);
    for (int i=1; i<=N; i++)
        {
            P[i].id=i;
            lk[i].last=C[ P[i].c ]; lk[i].nxt=N+1; lk[i].now=P[i].x;
            if (C[ P[i].c ]) lk[ C[ P[i].c ] ].nxt=i;
            C[ P[i].c ]=i;
            ans=max(ans,bit.Query(P[ lk[i].last ].x+1,P[i].x-1));
        }
//  printf("Ans1=%d\n",ans);
    sort(P+1,P+N+1,cmpY);
    for (int i=1; i<=K; i++)
        ans=max(ans,bit.Query(lk[ C[i] ].now+1,top+1));
//  printf("Ans2=%d\n",ans);
    for (int t=1,i=1; i<=N; i++)
        {
            int now=P[i].id;
            while (t<=N && P[t].y==P[i].y)
                bit.Add(P[t].x,-1),t++;
            if (lk[now].nxt) lk[ lk[now].nxt ].last=lk[now].last;
            if (lk[now].last) lk[ lk[now].last ].nxt=lk[now].nxt;
            ans=max(ans,bit.Query(lk[ lk[now].last ].now+1,lk[ lk[now].nxt ].now-1));
            lk[now].nxt=lk[now].last=0;
        }
//  printf("Ans3=%d\n",ans);
}
int main()
{
    T=read();
    while (T--)
        {
            N=read(),K=read();
            ans=0;
            for (int i=1; i<=N; i++)
                ls[++tp]=P[i].x=read(),P[i].y=read(),P[i].c=read(),P[i].id=i;
            sort(ls+1,ls+tp+1);
            top=unique(ls+1,ls+tp+1)-ls-1;
            for (int i=1; i<=N; i++) P[i].x=lower_bound(ls+1,ls+top+1,P[i].x)-ls;
            //for (int i=1; i<=N; i++) printf("x=%d\n",P[i].x);
            Solve();
            for (int i=1; i<=N; i++) P[i].y=-P[i].y;
            Solve();
            printf("%d\n",ans);
        }
    return 0;
}
时间: 2024-08-10 08:52:41

【BZOJ-4548&3658】小奇的糖果&Jabberwocky 双向链表 + 树状数组的相关文章

【BZOJ4548】小奇的糖果 set(链表)+树状数组

[BZOJ4548]小奇的糖果 Description 有 N 个彩色糖果在平面上.小奇想在平面上取一条水平的线段,并拾起它上方或下方的所有糖果.求出最多能够拾起多少糖果,使得获得的糖果并不包含所有的颜色. Input 包含多组测试数据,第一行输入一个正整数 T 表示测试数据组数. 接下来 T 组测试数据,对于每组测试数据,第一行输入两个正整数 N.K,分别表示点数和颜色数. 接下来 N 行,每行描述一个点,前两个数 x, y (|x|, |y| ≤ 2^30 - 1) 描述点的位置,最后一个数

Bzoj4548 小奇的糖果(链表+树状数组)

题面 Bzoj 题解 很显然,我们只需要考虑单独取线段上方的情况,对于下方的把坐标取反再做一遍即可(因为我们只关心最终的答案) 建立树状数组维护一个横坐标区间内有多少个点,维护双向链表实现查询一个点左(右)横坐标最大(小)的与它相同的点. 首先枚举没有取到的颜色,找出所有不包含这种颜色的区间,更新答案. 接着考虑两个相同颜色的点的贡献,按照纵坐标从大到小枚举所有的点,分别在树状数组和双向链表中删除当前点,并利用这个点左右两边和它颜色相同的点之间的区间内点的个数更新答案. #include <cs

BZOJ 2738 矩阵乘法(整体二分+二维树状数组)

[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2738 [题目大意] 给出一个方格图,询问要求求出矩阵内第k小的元素 [题解] 我们对答案的大小进行整体二分,用二维树状数组维护二维区间和, 将超过数量的分治到左区间,不满足的分治到右区间即可. [代码] #include <cstdio> #include <algorithm> #include <cstring> using namespace std;

[CSP-S模拟测试]:小P的单调数列(树状数组+DP)

题目描述 小$P$最近喜欢上了单调数列,他觉得单调的数列具有非常多优美的性质.经过小$P$复杂的数学推导,他计算出了一个单调增数列的艺术价值等于该数列中所有书的总和.并且以这个为基础,小$P$还可以求出任意一个数列的艺术价值,它等于将这个数列顺次划分若干个极长单调区间(相邻两个单调区间的单调性必须不相同)后,每个单调区间中元素总和的平均值.比如对于数列$3\ 7\ 9\ 2\ 4\ 5$,它将被划分为$[3\ 7\ 9]\ [2]\ [4\ 5]$,其艺术价值为$\frac{19+2+9}{3}

求序列A中每个数的左边比它小的数的个数(树状数组)

给定一个有N个正整数的序列A(N<=10^5,A[i]<=10^5),对序列中的每一个数,求出序列中它左边比它小的数的个数. 思路:树状数组的经典应用(裸题) 1 #include <iostream> 2 #include <algorithm> 3 #include <cstring> 4 5 using namespace std ; 6 7 const int N = 100010 ; 8 9 int c[N] ; 10 11 int lowbit(

[BZOJ 1103][POI 2007]大都市(DFS求拓扑序+树状数组)

题目链接:http://www.lydsy.com:808/JudgeOnline/problem.php?id=1103 题目大意:给你一个树,刚开始所有树边边权均为1,不断地将其中的某些边边权改为0,其间问你某个点到根节点之间路径上的边权和. 此题和POJ的Apple Tree很相近... 首先DFS生成整棵树的拓扑序,DFS时每个结点i进入的时间l[i]和离开的时间r[i],然后对每次更改操作,维护树状数组即可. #include <iostream> #include <stdi

bzoj 3594: [Scoi2014]方伯伯的玉米田 dp树状数组优化

3594: [Scoi2014]方伯伯的玉米田 Time Limit: 60 Sec  Memory Limit: 128 MBSubmit: 314  Solved: 132[Submit][Status] Description 方伯伯在自己的农田边散步,他突然发现田里的一排玉米非常的不美.这排玉米一共有N株,它们的高度参差不齐.方伯伯认为单调不下降序列很美,所以他决定先把一些玉米拔高,再把破坏美感的玉米拔除掉,使得剩下的玉米的高度构成一个单调不下降序列.方伯伯可以选择一个区间,把这个区间的

[BZOJ 3211]花神游历各国(并查集+树状数组)

Description Solution 树状数组单点修改区间查询 我们知道一个数n最多修改loglogn次就会变为1 并查集维护每个数右边第一个不为1的位置 #include<cstdio> #include<iostream> #include<cstdlib> #include<cstring> #include<cmath> #define MAXN 100005 using namespace std; typedef long lon

BZOJ 2244 SDOI2011 拦截导弹 CDQ分治/二维树状数组

题目大意:给定一个序列,每个元素是一个二元组,等概率选择一LIS,求LIS长度以及每个元素被选中的概率 第一问CDQ分治裸上 第二问用每个元素所在的LIS个数/总LIS个数就是答案 每个元素所在的LIS自己必选,然后统计前面的方案数和后面的方案数 以前面的方案数为例,令f[x]为以x结尾的LIS长度,那么有DP方程: g[i]=Σg[j] (f[j]+1=f[i],j<i,a[j].x<a[i].x,a[j].y<a[i].y) 将所有元素按f值排序,分层DP,每层DP是一个三维偏序,上