1
希望对你有帮助.|

zouyua

园龄:1年10个月粉丝:3关注:3

E. Level Up

E. Level Up

  • 题意
    玩家初始等级为 1, 有 n 只怪物,每个怪物有一个等级 ai, 如果怪物等级高于你,则你们会战斗,战斗后经验加1,否则怪物会逃跑,你不会获得经验,每 k 点经验就会升级。给你 q 个询问,给个询问给出 ix, 问你当 k=x 时,会不会发生战斗(即问你你的等级会不会小于等于此时的ai
  • 1n,q2105,1ai2105,1i,xn

听说根号分治的 sqrt 会被 hack,这里讲一个nlognlogn的做法

可以根据q,问的是到ik=x时的lv会不会小于等于ai, 因为lv与k是反比例关系,k越小,lv就会越高,对于每个位置处理出如果能发生战斗所需要的最大的k,记为bi, 如过bi<=x, 则可以说明,比能发生战斗所需的k还大,则到这个位置的lv会比ai还小,则会发生战斗(因为是反比例函数,你求得是最大上界,小于k的都会比ai 大,这里要仔细想一下

因为这样k具有单调性,比当前界限大的话,lv会变小,则以后战斗的怪只会增加不会减少,我们可以二分一个最大的mid, 满足在这个位置的战斗的怪所能到达的等级小于等于ai, mid - 1位置的战斗的怪所能到达的等级都大于ai, 因为是反函数。

现在我们需要一个数据结构来维护,在一段k的范围内加 1,这里记为k能战斗的数量(经验值),询问某个位置的数量(经验值),来当作二分得check函数计算等级, 也就是区间修改和单点查询,这里可以用线段树维护(树状数组也是可以得)。最后二分出的位置记录下来,当查询的判断,修改时修改的是大于等于答案的位置,因为大于等于的位置k越大,lv越小,战斗的次数越多。多想一下边界问题

#include<bits/stdc++.h>

using namespace std;
using ull = unsigned long long;
using ll = long long;
using PII = pair<ll,ll>;
using PIII = pair<ll, pair<ll,ll>>;
#define endl "\n"
#define pb push_back
#define IOS ios::sync_with_stdio(false);cin.tie(0)
#define lowbit(x) (x) & (-x)
#define point(x) setiosflags(ios::fixed)<<setprecision(x)
const int N=1e5+10;
const int INF=0x3f3f3f3f;
const int mod=1e9+7;
struct Info {//一定要初始化
	int val;

	int add = 0;
	Info (int x) {
		val = x;
	}
	Info () {
		val = 0;
	}
};
Info merge(const Info& a, const Info& b) {
	Info c;

	c.val = b.val + c.val;
	
	return c;
}
struct segtree {
	#define ls (u << 1)
	#define rs (u << 1 | 1)
	int n;
	segtree(int n) {init(n);};
	vector<Info> info;
	vector<int> a;
	void init(int n) {
		this->n = n;
		info.resize(n << 2);
		a.resize(n << 1);
	}
	void push_up(int u) {
		info[u] = merge(info[ls], info[rs]);
	}
	void build(int u, int l, int r)
    {
        if(l == r)
        {
            info[u] = Info();//填值
            return ;
        }
        int mid = l + r >> 1;
		build(u << 1, l, mid);
		build(u << 1 | 1, mid + 1, r);
		push_up(u);
    }
	void settag(int u, int k) {//处理数据
		info[u].val += k;
		info[u].add += k;
 	}
	void push_down(int u)
	{
		if(info[u].add)
        {
            settag(ls, info[u].add);
            settag(rs, info[u].add);
            info[u].add = 0;
        }
	}
	void update(int u, int l, int r, int pos, int k) {
		if(l == r) {
			info[u] = Info(k);
            return;
		}
        push_down(u);
		int mid = l + r >> 1;
		if(pos <= mid) update(ls, l, mid, pos, k);
		else update(rs, mid + 1, r, pos, k);
		push_up(u);

	};
	void update(int u, int l, int r, int x, int y, int k) {
		if(x <= l && r <= y) {
			settag(u, k);
            return;
		}
        push_down(u);
		int mid = l + r >> 1;
		if(x <= mid) update(ls, l, mid, x, y, k);
		if(mid < y) update(rs, mid + 1, r, x, y, k);
		push_up(u);
	}
	void update(int pos, int v) {
		update(1, 1, n, pos, v);
	}
	void update(int x, int y, int k) {
		update(1, 1, n, x, y, k);
	}
	Info query(int u, int l, int r, int x, int y) {
        if (x <= l && r <= y) return info[u];
        push_down(u);
        int mid = l + r >> 1;
        if (y <= mid) return query(ls, l, mid, x, y);
        else if (mid < x) return query(rs, mid + 1, r, x, y);
        else return merge(query(ls, l, mid, x, y), query(rs, mid + 1, r, x, y));
    }
    Info query(int l, int r) {
        return query(1, 1, n, l, r);
    }

};
void solve(){
	int n, q;
	cin >> n >> q;
	vector<int> a(n + 1), b(n + 1);
	for(int i = 1; i <= n; i ++) cin >> a[i];
	segtree tr(n);//维护区间加,直接lazy标记就行
	for(int i = 1; i <= n; i ++) {
		int l = 1, r = n;
		while(l < r) {
			int mid = l + r >> 1;
			if(tr.query(1, 1, n, mid, mid).val / mid + 1 <= a[i]) r = mid;//判断等级,单点查询
			else l = mid + 1;
		}
		b[i] = l;
		tr.update(1, 1, n, l, n, 1);
	}
	while(q --) {
		int x, y; cin >> x >> y;
		if(b[x] <= y) cout << "YES" << endl;
		else cout << "NO" << endl;
	}
	
}
int main()
{
	int T=1;
	//cin>>T;
	while(T--){
		solve();
	}
	return 0;
}

本文作者:zouyua

本文链接:https://www.cnblogs.com/ZouYua/p/18336107

版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。

posted @   zouyua  阅读(25)  评论(0编辑  收藏  举报
点击右上角即可分享
微信分享提示
评论
收藏
关注
推荐
深色
回顶
收起