Solution -「UNR #5」「UOJ #671」诡异操作
\(\mathcal{Desciprtion}\)
Link.
给定序列 \(\{a_n\}\),支持 \(q\) 次操作:
- 给定 \(l,r,v\),\(\forall i\in[l,r],~a_i\leftarrow\lfloor\frac{a_i}{v}\rfloor\);
- 给定 \(l,r,v\),\(\forall i\in[l,r],~a_i\leftarrow a_i\otimes v\),其中 \(\otimes\) 表示二进制按位与;
- 给定 \(l,r\),求 \(\sum_{i=l}^ra_i\)。
\(n\le3\times10^5\),\(q\le2\times10^5\),值域 \(V<2^{128}\),答案对 __uint128_t 自然溢出。
\(\mathcal{Solution}\)
写个“暴力”,证明复杂度,得到正解√
不考虑 (1) 操作,想想如何支持区间与-区间求和。我们用一棵线段树来维护,对于一个长为 \(s\) 的区间,维护一个表格 \(S_{0..\log s}\),设某个 bit \(x\) 在该区间中出现了 \(c\) 次,则在 \(S\) 中每个 \(c\) 的 bit 处加上 \(x\) 的贡献。可以发现 \(\sum S_i2^i\) 就是区间和,合并类似高精度加法做到 \(\mathcal O(s)\),而区间与直接在令 \(S_i\leftarrow S_i\otimes v\) 即可解决,也是 \(\mathcal O(s)\)。不失为一种奇怪的小 trick√
取整除,自然而然暴力做。那么至多遍历 \(\omega=\log V\) 次线段树,乘上 \(\mathcal O(s)\)(\(s\) 为区间长度)的上传代价,遍历整棵树的上传代价和式 \(T(n)=2T(n/2)+\log n=\mathcal O(n)\),所以总复杂度 \(\mathcal O(\omega n)\),加上区间与和求答案,最终复杂度 \(\mathcal O(q\log^2n+\omega n)\)。
\(\mathcal{Code}\)
用 template 实现线段树√
/*~Rainybunny~*/
#include <set>
#include <cmath>
#include <cstdio>
#include <cassert>
#include <cstring>
#define rep( i, l, r ) for ( int i = l, rep##i = r; i <= rep##i; ++i )
#define per( i, r, l ) for ( int i = r, per##i = l; i >= per##i; --i )
typedef __uint128_t UI;
inline UI rxint() {
static char buf[100]; scanf( "%s", buf );
UI ret = 0;
for ( int i = 0; buf[i]; ++i ) {
ret = ret << 4 | ( buf[i] <= '9' ? buf[i] ^ '0' : buf[i] - 'a' + 10 );
}
return ret;
}
inline int rdint() {
int x = 0, f = 1, s = getchar();
for ( ; s < '0' || '9' < s; s = getchar() ) f = s == '-' ? -f : f;
for ( ; '0' <= s && s <= '9'; s = getchar() ) x = x * 10 + ( s ^ '0' );
return x * f;
}
inline void wxint( const UI x ) {
if ( x >= 16 ) wxint( x >> 4 );
putchar( ( x & 15 ) >= 10 ? 'a' + ( x & 15 ) - 10 : '0' ^ ( x & 15 ) );
}
inline int imax( const int a, const int b ) { return a < b ? b : a; }
const int MAXN = 3e5, MAXQ = 2e5;
int n, q;
template<const int H>
struct Atom {
UI val[H + 1];
inline UI& operator [] ( const int k ) { return val[k]; }
friend inline Atom<H + 1> operator + ( Atom<H>& u, Atom<H>& v ) {
static Atom<H + 1> ret;
ret[0] = u[0] ^ v[0]; UI up = u[0] & v[0];
rep ( i, 1, H ) {
ret[i] = up ^ u[i] ^ v[i];
up = ( up & u[i] ) | ( up & v[i] ) | ( u[i] & v[i] );
}
ret[H + 1] = up;
return ret;
}
inline Atom& operator &= ( const UI& x ) {
rep ( i, 0, H ) val[i] &= x;
return *this;
}
inline UI get() const {
UI ret = 0;
rep ( i, 0, H ) ret += val[i] << i;
return ret;
}
};
template<const int H>
struct SegmentTree {
Atom<H> sum; bool zero; UI tag;
SegmentTree<H - 1> lch, rch;
inline void pushup() {
sum = lch.sum + rch.sum, zero = lch.zero && rch.zero;
}
inline void pushan( const UI& v ) {
tag &= v, sum &= v;
}
inline void pushdn() {
if ( ~tag ) {
lch.pushan( tag ), rch.pushan( tag );
tag = ~UI( 0 );
}
}
inline void build( const int l, const int r ) {
int mid = l + r >> 1; tag = ~UI( 0 );
lch.build( l, mid ), rch.build( mid + 1, r );
pushup();
}
inline void secAnd( const int l, const int r,
const int ql, const int qr, const UI& v ) {
if ( zero ) return ;
if ( ql <= l && r <= qr ) return void( pushan( v ) );
int mid = l + r >> 1; pushdn();
if ( ql <= mid ) lch.secAnd( l, mid, ql, qr, v );
if ( mid < qr ) rch.secAnd( mid + 1, r, ql, qr, v );
pushup();
}
inline void secDiv( const int l, const int r,
const int ql, const int qr, const UI& v ) {
if ( zero || v == 1 ) return ;
if ( l == r ) return void( zero = !( sum[0] /= v ) );
int mid = l + r >> 1; pushdn();
if ( ql <= mid ) lch.secDiv( l, mid, ql, qr, v );
if ( mid < qr ) rch.secDiv( mid + 1, r, ql, qr, v );
pushup();
}
inline UI query( const int l, const int r, const int ql, const int qr ) {
if ( zero ) return 0;
if ( ql <= l && r <= qr ) return sum.get();
int mid = l + r >> 1; UI ret = 0; pushdn();
if ( ql <= mid ) ret += lch.query( l, mid, ql, qr );
if ( mid < qr ) ret += rch.query( mid + 1, r, ql, qr );
return ret;
}
};
template<>
struct SegmentTree<0> {
Atom<0> sum; bool zero; UI tag;
inline void pushan( const UI& v ) { tag &= v, sum &= v; }
inline void build( const int l, const int r ) {
assert( l == r );
tag = ~UI( 0 ), zero = !( sum[0] = r < n ? rxint() : 0 );
}
inline void secAnd( const int l, const int r,
const int ql, const int qr, const UI& v ) {
if ( zero ) return ;
assert( ql <= l && r <= qr ), pushan( v );
}
inline void secDiv( const int l, const int r,
const int ql, const int qr, const UI& v ) {
if ( zero || v == 1 ) return ;
zero = !( sum[0] /= v );
}
inline UI query( const int l, const int r, const int ql, const int qr ) {
if ( zero ) return 0;
return assert( ql <= l && r <= qr ), sum.get();
}
};
SegmentTree<19> sgt;
int main() {
// freopen( "machine.in", "r", stdin );
// freopen( "machine.out", "w", stdout );
n = rdint(), q = rdint();
int R = 1 << 19;
sgt.build( 0, R - 1 );
for ( int op, l, r; q--; ) {
op = rdint(), l = rdint() - 1, r = rdint() - 1;
if ( op == 1 ) sgt.secDiv( 0, R - 1, l, r, rxint() );
else if ( op == 2 ) sgt.secAnd( 0, R - 1, l, r, rxint() );
else wxint( sgt.query( 0, R - 1, l, r ) ), putchar( '\n' );
}
return 0;
}
Solution -「UNR #5」「UOJ #671」诡异操作的更多相关文章
- 「UNR#2」黎明前的巧克力
「UNR#2」黎明前的巧克力 解题思路 考虑一个子集 \(S\) 的异或和如果为 \(0\) 那么贡献为 \(2^{|S|}\) ,不难列出生产函数的式子,这里的卷积是异或卷积. \[ [x^0]\p ...
- 「UNR#1」奇怪的线段树
「UNR#1」奇怪的线段树 一道好题,感觉解法非常自然. 首先我们只需要考虑一次染色最下面被包含的那些区间,因为把无解判掉以后只要染了一个节点,它的祖先也一定被染了.然后发现一次染色最下面的那些区间一 ...
- Solution -「CTS 2019」「洛谷 P5404」氪金手游
\(\mathcal{Description}\) Link. 有 \(n\) 张卡牌,第 \(i\) 张的权值 \(w_i\in\{1,2,3\}\),且取值为 \(k\) 的概率正比于 \ ...
- Solution -「LGR-087」「洛谷 P6860」象棋与马
\(\mathcal{Description}\) Link. 在一个 \(\mathbb R^2\) 的 \((0,0)\) 处有一颗棋子,对于参数 \(a,b\),若它当前坐标为 \((x ...
- 「题解」「美团 CodeM 资格赛」跳格子
目录 「题解」「美团 CodeM 资格赛」跳格子 题目描述 考场思路 思路分析及正解代码 「题解」「美团 CodeM 资格赛」跳格子 今天真的考自闭了... \(T1\) 花了 \(2h\) 都没有搞 ...
- 【翻译】西川善司的「实验做出的游戏图形」「GUILTY GEAR Xrd -SIGN-」中实现的「纯卡通动画的实时3D图形」的秘密,后篇
http://www.4gamer.net/games/216/G021678/20140714079/ 连载第2回的本回, Arc System Works开发的格斗游戏「GUILTY G ...
- Android内存管理(4)*官方教程 含「高效内存的16条策略」 Managing Your App's Memory
Managing Your App's Memory In this document How Android Manages Memory Sharing Memory Allocating and ...
- SSH连接时出现「WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED!」解决办法
用ssh來操控github,沒想到連線時,出現「WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED!」,後面還有一大串英文,這時當然要向Google大神求助 ...
- 「Windows MFC 」「Edit Control」 控件
「Windows MFC 」「Edit Control」 控件
随机推荐
- Go语言实战-爬取校花网图片
一.目标网站分析 爬取校花网http://www.xiaohuar.com/大学校花所有图片. 经过分析,所有图片分为四个页面,http://www.xiaohuar.com/list-1-0.htm ...
- Hive建表和内外部表的使用
原文链接: https://www.toutiao.com/i6766784274965201415 一.普通建表方式 create table stu_info( id int, name stri ...
- 移动Web开发实践——解决position:fixed自适应BUG
在移动web中使用position:fixed,会踩到很多坑,在我之前的一篇文章<移动端web页面使用position:fixed问题总结>中已经总结了很多bug,但是在后续的开发中有关f ...
- 学习javaScript必知必会(6)~类、类的定义、prototype 原型、json对象
一.定义类:使用的是funciton,因为在js中没有定义类的class语句,只有function. ■ 举例: //定义一个Person类(通过类的无参构造函数定义类) function Perso ...
- 【刷题-PAT】A1119 Pre- and Post-order Traversals (30 分)
1119 Pre- and Post-order Traversals (30 分) Suppose that all the keys in a binary tree are distinct p ...
- 【记录一个问题】thanos receiver的日志中出现错误:conflict
完整的错误如下: level=debug ts=2021-08-16T09:07:43.412451Z caller=handler.go:355 component=receive componen ...
- 关于C++11共享数据带来的死锁问题的提出与解决
举个例子,如果有一份资源,假如为list<int>资源,假设有两个线程要对该资源进行压入弹出操作,如果不进行锁的话,那么如果两个线程同时操作,那么必然乱套,得到的结果肯定不是我们想要的结果 ...
- golang中的匿名函数三种用法
package main import ( "fmt" "strconv" ) func main() { // 匿名函数的使用:方式1 f1 := func( ...
- linux文件创建删除与基础命令使用
目录 一:linux介绍 二:系统介绍 三:系统命令语法格式 四:系统目录结构 五:系统路径的类型 六:系统运行命令 七:查看系统命令帮助man手册 八:系统快捷方式 九:常用系统命令 十:进行目录创 ...
- Shell 脚本进阶,经典用法及其案例
一.条件选择.判断 1.条件选择if (1)用法格式 if 判断条件 1 ; then 条件为真的分支代码 elif 判断条件 2 ; then 条件为真的分支代码 elif 判断条件 3 ; the ...