牛客小白月赛16 小石的妹子 二分 or 线段树
这个题目我AC之后看了一下别人的题解,基本上都是线段树,不过二分也可以。
这个题目很自然就肯定要对其中一个进行排序,排完序之后再处理另外一边,另一边记得离散化。
怎么处理呢,你仔细想想,找找规律就可以发现,其实我们就是在找递增子序列。
第一次找到的就是重要程度为1 的妹子,然后删除这些元素,继续第二次找,第二次找到的就是重要程度为二的妹子。
所以到每一个点,我们就只要根据它的大小来判断它应该放的位置(尽量靠前,并且小于这个数),然后更新这个位置,再返回这个位置,它所在的位置就是它的重要程度。
emmm 其实就是用一个数组,数组的每一个位置 i 存的就是到目前位置重要程度为 i 的最大值。
具体看代码吧。
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <queue>
#include <vector>
#include <algorithm>
#include <string>
#include <iostream>
#include <map>
#define inf 0x3f3f3f3f
#define inf64 0x3f3f3f3f3f3f3f3f
using namespace std;
const int maxn = 2e5 + ;
typedef long long ll;
int num[maxn];
map<ll, ll>mp;
struct node
{
int l, r;
int sum;
}tree[*maxn];
struct edge
{
ll a, b, id, num, ans;
}ex[maxn]; bool cmp(edge a,edge b)
{
return a.a > b.a;
} bool cmp1(edge a,edge b)
{
return a.b < b.b;
} bool cmp2(edge a,edge b)
{
return a.id < b.id;
} int tot = ;
int ok(ll x)
{
if (x > num[]) return ;
if (x < num[tot])
{
tot++;
return tot;
}
int l = , r = tot, ans = ;
int mid = (l + r) >> ;
while (l <= r) {
int mid = (l + r) / ;
if (num[mid]<x) ans = mid, r = mid - ;
else l = mid + ;
}
return ans;
} int main()
{
int n;
scanf("%d", &n);
for (int i = ; i <= n; i++) {
scanf("%lld%lld", &ex[i].a, &ex[i].b), ex[i].id = i;
}
sort(ex + , ex + + n, cmp1);
for (int i = ; i <= n; i++) mp[ex[i].b] = i;
sort(ex + , ex + + n, cmp);
num[] = mp[ex[].b];
ex[].ans = ;
tot = ;
for(int i=;i<=n;i++)
{
int f = ok(mp[ex[i].b]);
// printf("%lld ex[%d]=%lld f=%d\n",mp[ex[i].b], i, ex[i].b, f);
num[f] = mp[ex[i].b];
ex[i].ans = f;
}
sort(ex + , ex + + n, cmp2);
for (int i = ; i <= n; i++) printf("%lld\n", ex[i].ans);
return ;
}
二分
网上的线段树的方法我感觉和逆序对有点像,就是首先还是把 b 离散化,然后对 a 进行排序,
然后从 1 ~ n 遍历,如果对于每一个 b 首先判断 b ~ n 有没有值,其实就是有没有比 b 大的再前面放过了,有的话就去最大值,重要程度就是 最大值+1 (这个是因为存进去的就是最大值)
没有那么重要程度就是 1 ,然后再更新这个点 b 把 b 的重要程度放到线段树的 b 这个位置以便后面的查询。
这个很好写的。
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <queue>
#include <vector>
#include <algorithm>
#include <string>
#include <iostream>
#include <map>
#define inf 0x3f3f3f3f
#define inf64 0x3f3f3f3f3f3f3f3f
using namespace std;
const int maxn = 2e5 + ;
typedef long long ll;
map<ll, ll>mp;
struct node
{
int l, r;
int num;
}tree[*maxn];
struct edge
{
ll a, b, id, num, ans;
}ex[maxn]; bool cmp(edge a,edge b)
{
return a.a > b.a;
} bool cmp1(edge a,edge b)
{
return a.b < b.b;
} bool cmp2(edge a,edge b)
{
return a.id < b.id;
} void build(int id, int l, int r) {
tree[id].l = l;
tree[id].r = r;
if (l == r) {
tree[id].num = ;
return;
}
int mid = (l + r) >> ;
build(id << , l, mid);
build(id << | , mid + , r);
} int query(int id, int x, int y) {
int l = tree[id].l;
int r = tree[id].r;
if (x <= l && y >= r) {
// printf("id=%d sum=%d \n", id,tree[id].sum);
return tree[id].num;
}
int ans = ;
int mid = (l + r) >> ;
if (x <= mid) ans = max(ans, query(id << , x, y));
if (y > mid) ans = max(ans, query(id << | , x, y));
// printf("id=%d ans=%d l=%d r=%d x=%d y=%d \n", id, ans, l, r,x,y);
return ans;
} void push_up(int id) {
tree[id].num = max(tree[id << ].num , tree[id << | ].num);
} void update(int id, int x,int val) {
int l = tree[id].l;
int r = tree[id].r;
if (l == r) {
tree[id].num = val;
return;
}
int mid = (l + r) >> ;
if (x <= mid) update(id << , x,val);
else update(id << | , x,val);
push_up(id);
} int main()
{
int n;
scanf("%d", &n);
for (int i = ; i <= n; i++) {
scanf("%lld%lld", &ex[i].a, &ex[i].b), ex[i].id = i;
}
sort(ex + , ex + + n, cmp1);
for (int i = ; i <= n; i++) mp[ex[i].b] = i;
sort(ex + , ex + + n, cmp);
build(, , n);
for(int i=;i<=n;i++)
{
int f = query(, mp[ex[i].b], n) + ;
ex[i].ans = f;
update(, mp[ex[i].b],f);
}
sort(ex + , ex + + n, cmp2);
for (int i = ; i <= n; i++) printf("%lld\n", ex[i].ans);
return ;
}
逆序对 线段树
牛客小白月赛16 小石的妹子 二分 or 线段树的更多相关文章
- 牛客小白月赛16 F 小石的妹子 (线段树)
链接:https://ac.nowcoder.com/acm/contest/949/F来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言52428 ...
- 牛客小白月赛16 A 小石的签到题 ( 博弈)
链接:https://ac.nowcoder.com/acm/contest/949/A来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言52428 ...
- 牛客小白月赛16 E 小雨的矩阵 ( 暴搜)
链接:https://ac.nowcoder.com/acm/contest/949/E来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言52428 ...
- 牛客小白月赛16 H 小阳的贝壳 (差分+线段树)
链接:https://ac.nowcoder.com/acm/contest/949/H来源:牛客网 题目描述 小阳手中一共有 n 个贝壳,每个贝壳都有颜色,且初始第 i 个贝壳的颜色为 colico ...
- 牛客小白月赛16 D 小阳买水果 (思维题)
链接:https://ac.nowcoder.com/acm/contest/949/D来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言52428 ...
- 牛客小白月赛16 H小阳的贝壳 (线段树+差分数组)
链接:https://ac.nowcoder.com/acm/contest/949/H来源:牛客网 题目描述 小阳手中一共有 n 个贝壳,每个贝壳都有颜色,且初始第 i 个贝壳的颜色为 colico ...
- 树的最长链-POJ 1985 树的直径(最长链)+牛客小白月赛6-桃花
求树直径的方法在此转载一下大佬们的分析: 可以随便选择一个点开始进行bfs或者dfs,从而找到离该点最远的那个点(可以证明,离树上任意一点最远的点一定是树的某条直径的两端点之一:树的直径:树上的最长简 ...
- 牛客网 牛客小白月赛5 I.区间 (interval)-线段树 or 差分数组?
牛客小白月赛5 I.区间 (interval) 休闲的时候写的,但是写的心情有点挫,都是完全版线段树,我的一个队友直接就水过去了,为啥我的就超内存呢??? 试了一晚上,找出来了,多初始化了add标记数 ...
- 牛客小白月赛8 - E - 诡异数字 数位DP
牛客小白月赛8 - E - 诡异数字 题意: 求区间中,满足限制条件的数字的个数. 限制条件就是某些数字不能连续出现几次. 思路: 比较裸的数位DP, DP数组开一个dp[len][x][cnt] 表 ...
随机推荐
- 微信小程序与H5数据传递
这的场景是 小程序webview 加载 H5应用 需求点: 1. 小程序的登录code 需要与H5应用的sessionId建立绑定关系 2.H5内发起微信小程序支付,支付参数传递到小程序,支付结果传递 ...
- AJ学IOS 之微博项目实战(6)导航控制器NavigationController 的滑动回退功能实现
AJ分享,必须精品 一:效果 第二篇里面写了怎样自定义navigation实现自定义的导航控制器左右按钮样式,但是当我们自己实现后,系统自带的向右边滑动来实现回退的功能就不能用了. 这里主要实现滑动回 ...
- [算法总结]康托展开Cantor Expansion
目录 一.关于康托展开 1.什么是康托展开 2.康托展开实现原理 二.具体实施 1.模板 一.关于康托展开 1.什么是康托展开 求出给定一个由1n个整数组成的任意排列在1n的全排列中的位置. 解决这样 ...
- 小小小小小flag
2020:300道题 小小小小小flag 150红题 100道橙题 50道黄题 努力变强!加油 我的主页: 主页https://www.luogu.com.cn/user/306734 谢谢大家,目前 ...
- ajax ★ ★ ★ ★ ★
ajax 1 定义: 是创建交互式应用的网页交互技术 2 特点:无刷新.异步 3 中介数据类型: 1) XML - 可扩展的标记语言 ...
- Java 基础讲解
Hello,老同学们,又见面啦,新同学们,你们好哦! 在看完本人的<数据结构与算法>专栏的博文的老同学,恭喜你们在学习本专栏时,你们将会发现好多知识点都讲解过,都易于理解,那么,没看过的同 ...
- CentOS 使用中问题记录
⚠️使用yum提示Error: rpmdb open failed的解决方案 清除原rpmdb文件,这一步可能不用操作,直接进行第2步 # rm -f /var/lib/rpm/__db.* 重建rp ...
- UnicodeDecodeError: 'gbk' codec can't decode byte 0x8a in position 2: illegal multibyte sequence
pycharm报错UnicodeDecodeError: 'gbk' codec can't decode byte 0x8a in position 2: illegal multibyte seq ...
- Java 多线程 -- volatile 山寨版的synchronized
在 多线程中,每个线程会把数据从主内存中拷贝到自己的工作内存中,当线程完成计算后,再把工作内存的数据更新到主内存中,或者当主内存主数据有更新是,线程会去主内存取最新数据.但是,当线程特别忙时,就不会去 ...
- python信息收集(一)
在渗透测试初期,需要进行大量的信息收集.一般情况下,信息收集可以分为两大类----被动信息收集和主动信息收集. 其中,被动信息收集主要是通过各种公开的渠道来获取目标系统的信息,例如:站 ...