【UOJ #46】 【清华集训2014】玄学

题目描述

巨酱有 n 副耳机,他把它们摆成了一列,并且由 1 到n依次编号。每个耳机有一个玄学值,反映了各自的一些不可名状的独特性能。玄学值都是 0 到 m-1 间的整数。在外界的作用下(包括但不限于换线、上放、更换电源为核电、让kAc叔叔给它们讲故事),这些耳机的玄学值会发生改变。特别地,巨酱观察发现,每种作用 o 对应了两个整数 ao与 bo,在这种作用之后,玄学值原本为 x 的耳机,其玄学值恰会变成 (aox+bo)modm。

巨酱对他手头耳机的表现并不满意,遗憾的是,最近他并不有钱,无法任性,不能赶紧买买买以满足自己。手头紧张的他准备拟定一个相对经济的方案,通过各种作用来改善他手头玩具的性能。具体地说,为了尽快完成方案的制订,巨酱希望自己能高效地完成以下工作:

巨酱想到了一种操作,能让耳机的玄学值由 x 变为 (ax+b)modm,并且他计划对编号为 i 到 j 的耳机执行这种操作。
巨酱想知道如果将(并且仅将)自己的第 i 个到第 j 个计划按顺序付诸行动,编号为 k 的耳机的玄学值将会变成多少。
出于著名算法竞赛选手的矜持,巨酱表示自己才不需要你的帮助。但是如果巨酱真的厌倦了自己的玩具,它们就会被50包邮出给主席。为了不让后者白白捡到便宜,你考虑再三还是决定出手。

题解

二进制分组的思想。

用线段树维护时间的操作序列,每次操作一个一个往线段树里面插,等到一个线段被插满的时候用归并来维护区间的信息。查询的时候如果一个线段没有被插满就递归下去。定位到一个区间的时候在区间里面归并出来的信息二分。

代码

 1 #include <cstdio>
 2
 3 #define maxn 100010
 4 #define maxm 600010
 5 #define R register
 6 int x[maxn], tnum;
 7 struct Seg {
 8     int l, r, a, b;
 9 } p[maxn * 200];
10 int lef[maxm << 2], rig[maxm << 2], pcnt, ta, tb, ql, qr, n, m, k, ans;
11 void update(R int o, R int l, R int r)
12 {
13     lef[o] = pcnt + 1;
14     for (R int i = lef[o << 1], j = lef[o << 1 | 1], head = 1; i <= rig[o << 1] || j <= rig[o << 1 | 1]; )
15         if (p[i].r <= p[j].r)
16         {
17             p[++pcnt] = (Seg) {head, p[i].r, 1ll * p[i].a * p[j].a % m, (1ll * p[j].a * p[i].b + p[j].b) % m};
18             head = p[i].r + 1;
19             p[i].r == p[j].r ? ++j : 0; ++i;
20         }
21         else
22         {
23             p[++pcnt] = (Seg) {head, p[j].r, 1ll * p[i].a * p[j].a % m, (1ll * p[j].a * p[i].b + p[j].b) % m};
24             head = p[j].r + 1; ++j;
25         }
26     rig[o] = pcnt;
27 }
28 int find(R int o, R int t, R int &s)
29 {
30     R int l = lef[o], r = rig[o];
31     while (l < r)
32     {
33         R int mid = l + r >> 1;
34         if (t <= p[mid].r) r = mid;
35         else l = mid + 1;
36     }
37 //    printf("%d %d t %d s %d %d %d\n", p[l].l, p[l].r, t, s, p[l].a, p[l].b);
38     s = (1ll * s * p[l].a + p[l].b) % m;
39 }
40 void modify(R int o, R int l, R int r, R int t)
41 {
42     if (l == r)
43     {
44         lef[o] = pcnt + 1;
45         ql > 1 ? p[++pcnt] = (Seg) {1, ql - 1, 1, 0}, 1: 0;
46         p[++pcnt] = (Seg) {ql, qr, ta, tb};
47         qr < n ? p[++pcnt] = (Seg) {qr + 1, n, 1, 0}, 1: 0;
48         rig[o] = pcnt;
49         return ;
50     }
51     R int mid = l + r >> 1;
52     if (t <= mid) modify(o << 1, l, mid, t);
53     else modify(o << 1 | 1, mid + 1, r, t);
54
55     if (t == r) update(o, l, r);
56 }
57 void query(R int o, R int l, R int r)
58 {
59     if (ql <= l && r <= qr)
60     {
61         find(o, k, ans);
62         return ;
63     }
64     R int mid = l + r >> 1;
65     if (ql <= mid) query(o << 1, l, mid);
66     if (mid < qr) query(o << 1 | 1, mid + 1, r);
67 }
68 int main()
69 {
70     R int type; scanf("%d%d%d", &type, &n, &m);
71     for (R int i = 1; i <= n; ++i) scanf("%d", &x[i]);
72     R int Q; scanf("%d", &Q);
73     for (R int QQ = 1; QQ <= Q; ++QQ)
74     {
75         R int opt, l, r; scanf("%d%d%d", &opt, &l, &r);
76         type & 1 ? l ^= ans, r ^= ans : 0;
77         if (opt == 1)
78         {
79             scanf("%d%d", &ta, &tb); ++tnum; ql = l; qr = r;
80             modify(1, 1, Q, tnum);
81         }
82         else
83         {
84             scanf("%d", &k); type & 1 ? k ^= ans : 0; ql = l; qr = r;
85             ans = x[k];
86             query(1, 1, Q);
87             printf("%d\n", ans);
88         }
89     }
90     return 0;
91 }
时间: 2024-08-29 10:16:20

【UOJ #46】 【清华集训2014】玄学的相关文章

uoj #46[清华集训2014]玄学

uoj 因为询问是关于一段连续区间内的操作的,所以对操作构建线段树,这里每个点维护若干个不交的区间,每个区间\((l,r,a,b)\)表示区间\([l,r]\)内的数要变成\(ax+b\) 每次把新操作加入线段树中下一个叶子,然后如果某个节点里所有操作都加进去了,就条到父亲,把两个儿子的信息合并到父亲上.这里合并就是把两个区间集合合并成一个,例如两个区间\([a,c]\)和\([b,d](a\le b\le c\le d)\)会合并成\([a,b),[b,c),[c,d]\).合并出来的区间如果

线性基 uoj 36 清华集训2014 玛里苟斯

http://uoj.ac/problem/36 感觉挺可做的但是不会.. \(k=1\)按位扫就好了 出现\(1\)概率就是\(0.5\) \(k=2\)要特殊处理 设xor后的数是\(b_1 b_2 \dots b_n\) 计算平方贡献就好了\(\sum_{i,j}b_i b_j 2^{i+j}*p\) \(p\)存在一位\(i\),\(j\)不同的话是1/4 否则就是1/2 \(k\geq3\) 首先要知道随机选数的话 如果维护的集合可以通过xor算子得到\(x\) 那么\(x\)与答案无

[UOJ46][清华集训2014]玄学

uoj description 给出\(n\)个变换,第\(i\)个变换是将区间中\(l_i,r_i\)的数\(x\)变成\((a_ix+b_i)\mod m\). 每次会新增一个变换,或者查询询问如果进行编号\([s,t]\)的操作,第\(k\)个数会变成多少. \(n\le10^5,q\le6\times10^5\) sol 二进制分组. 按顺序把变化插入线段树,如果线段树的某个满了就向上归并两个儿子的变换. \(a_j(a_ix+b_i)+b_j=a_ia_jx+a_jb_i+b_j\)就

bzoj 3816&amp;&amp;uoj #41. [清华集训2014]矩阵变换

稳定婚姻问题: 有n个男生,n个女生,所有女生在每个男生眼里有个排名,反之一样. 将男生和女生两两配对,保证不会出现婚姻不稳定的问题. 即A-1,B-2 而A更喜欢2,2更喜欢A. 算法流程: 每次男生向自己未追求过的排名最高女生求婚. 然后每个有追求者的女生在自己现男友和追求者中选择一个最喜欢的接受,然后拒绝其他人. 算法一定可以结束. 因为如果最后有一个男生单身,那他一定把所有女生都追求过一遍,说明没有女生单身,产生矛盾. #include<iostream> #include<cs

UOJ.41.[清华集训2014]矩阵变换(稳定婚姻)

题目链接 稳定婚姻问题:有n个男生n个女生,每个男/女生对每个女/男生有一个不同的喜爱程度.给每个人选择配偶. 若不存在 x,y未匹配,且x喜欢y胜过喜欢x当前的配偶,y喜欢x也胜过y当前的配偶 的完备匹配,则称这是一个稳定匹配. 稳定匹配一定存在,且存在一个\(O(n^2)\)的算法: 任选一个未匹配的男生x,按x的喜爱程度从大到小枚举每个女生,若当前女生没有配偶或喜欢x胜过喜欢当前配偶,则与x匹配.直到所有男生都匹配. 这一题我们用行表示男生,n个数表示女生.喜爱程度为:行更喜欢靠前的数,数

uoj 41 【清华集训2014】矩阵变换 婚姻稳定问题

[清华集训2014]矩阵变换 Time Limit: 20 Sec  Memory Limit: 256 MB 题目连接 http://uoj.ac/problem/41 Description 给出一个 N 行 M 列的矩阵A, 保证满足以下性质: M>N.    矩阵中每个数都是 [0,N] 中的自然数.    每行中, [1,N] 中每个自然数都恰好出现一次.这意味着每行中 0 恰好出现 M−N 次.    每列中,[1,N] 中每个自然数至多出现一次. 现在我们要在每行中选取一个非零数,

AC日记——【清华集训2014】奇数国 uoj 38

#38. [清华集训2014]奇数国 思路: 题目中的number与product不想冲: 即为number与product互素: 所以,求phi(product)即可: 除一个数等同于在模的意义下乘以一个数的逆元: 代码: #include <cstdio> #include <cstring> #include <iostream> #include <algorithm> using namespace std; #define maxn 100005

清华集训2014 做题记录

清华集训2014做题记录 已完成 [清华集训2014]玛里苟斯 [清华集训2014]主旋律 [清华集训2014]奇数国 [清华集训2014]矩阵变换 [清华集训2014]sum [清华集训2014]虫逢 [清华集训2014]玄学 [清华集训2014]文学 未完成 [清华集训2014]卡常数 [清华集训2014]简单回路 [清华集训2014]Router [清华集训2014] Breaking Bomber 写一题要膜一题题解,膜完题解膜代码,膜完代码膜指导,膜了好几天了还有四个题没做. [清华集

「清华集训2014」主旋律

「清华集训2014」主旋律 这个题好难难啊,我想了一个小时连50分都不会,只能去摸周指导了. 我们试图直接爆算集合 \(S\) 的非强连通导出子图数量,考虑将这个导出子图的所有强连通分量缩点后,一定是一个点数 \(\geq 2\) 的 \(\text{DAG}\) .即缩完以后至少要有一个入度为 \(0\) 的点,这个条件充分性显然,必要性考虑如果不存在入度为 \(0\) 的点,那么一定存在一个环,说明并没有将所有强连通分量缩掉.然后我们枚举入度为 \(0\) 的点集 \(T\) ,记 \(F(

「清华集训2014」矩阵变换

「清华集训2014」矩阵变换 解题思路 问题转化为找一个行与选的数字的完美匹配,记 \(pos[i][j]\) 为数字 \(i\) 在第 \(j\) 行的出现位置,要求不存在匹配边 \((a,b),(c,d)\) 使得 \(pos[b][a] < pos[b][c]\ \& \ pos[d][c] > pos[b][c]\) . 观察发现,这个式子相当于让一条边的权值看做 \(pos[x][y]\) ,当存在一个行与一个数没有匹配但是强行让它们匹配后数的权值会变大,行的权值会变小,这个