HDU-6356

题意:有m次操作,每次操作通过给定的随机函数生成 l , r , v,使得在 到 区间内,所有的a【i】变为max(a[i] , v).

  最后输出n个a【i】* i的异或和。

 

思路:线段树操作,每次维护区间的最小值,如果当前的v小于区间的最小值,直接return,lazy标记维护区间未加的最大值,必要时pushdown就ok;

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <string>
#include <vector>
#include <map>
#include <set>
#include <queue>
#include <list>
#include <cstdlib>
#include <iterator>
#include <cmath>
#include <iomanip>
#include <bitset>
#include <cctype>
using namespace std;
//#pragma comment(linker, "/STACK:102400000,102400000") //c++
#define lson (l , mid , rt << 1)
#define rson (mid + 1 , r , rt << 1 | 1)
#define debug(x) cerr << #x << " = " << x << "\n";
#define pb push_back
#define pq priority_queue typedef long long ll;
typedef unsigned long long ull; typedef pair<ll ,ll > pll;
typedef pair<int ,int > pii; //priority_queue<int> q;//这是一个大根堆q
//priority_queue<int,vector<int>,greater<int> >q;//这是一个小根堆q
#define fi first
#define se second
// #define endl '\n' #define OKC ios::sync_with_stdio(false);cin.tie(0)
#define FT(A,B,C) for(int A=B;A <= C;++A) //用来压行
#define REP(i , j , k) for(int i = j ; i < k ; ++i)
//priority_queue<int ,vector<int>, greater<int> >que; const ll mos = 0x7FFFFFFF; //
const ll nmos = 0x80000000; //-2147483648
const int inf = 0x3f3f3f3f;
const ll inff = 0x3f3f3f3f3f3f3f3f; // template<typename T>
inline T read(T&x){
x=;int f=;char ch=getchar();
while (ch<''||ch>'') f|=(ch=='-'),ch=getchar();
while (ch>=''&&ch<='') x=x*+ch-'',ch=getchar();
return x=f?-x:x;
}
// #define _DEBUG; //*//
#ifdef _DEBUG
freopen("input", "r", stdin);
// freopen("output.txt", "w", stdout);
#endif
/*-----------------------show time----------------------*/
unsigned X,Y,Z,W;
unsigned RNG61(){
X = X ^ (X<<);
X = X ^ (X>>);
X = X ^ (X<<);
X = X ^ (X>>);
W = X ^ (Y ^ Z);
X = Y;
Y = Z;
Z = W;
return Z;
} const ll MOD = (<<);
const int maxn = 1e5+;
int n,m; ll sum[maxn*],lazy[maxn*];
void build(int l,int r,int rt){
sum[rt] = ;
lazy[rt] = ;
if(l==r){
return;
}
int mid = (l + r)/;
build(l,mid,rt<<);
build(mid+,r,rt<<|);
}
void pushup(int rt){
sum[rt] = min(sum[rt<<] , sum[rt<<|]);
}
void pushdown(int rt){
if(lazy[rt]){
lazy[rt<<] = max(lazy[rt],lazy[rt<<]);
lazy[rt<<|] = max(lazy[rt],lazy[rt<<|]);
sum[rt<<] = max(sum[rt<<] , lazy[rt]);
sum[rt<<|] = max(sum[rt<<|] , lazy[rt]);
lazy[rt] = ;
}
}
void update(int l, int r, int rt,int L,int R,ll val){
if(sum[rt] > val)return;
if(l>=L && r <=R){ sum[rt] = max(sum[rt],val); lazy[rt] = max(lazy[rt],val);
return;
}
pushdown(rt);
int mid = (l+r)/;
if(mid >= L)update(l,mid,rt<<,L,R,val);
if(mid < R)update(mid+,r, rt<<|, L,R,val);
pushup(rt);
}
ll ans = 0ll;
void g_ans(int l,int r,int rt,int L,int R){
if(l==r){
// debug(sum[rt]);
ans ^= (1ll*l*sum[rt]);
return;
}
pushdown(rt);
int mid = (l+r)/;
g_ans(l,mid,rt<<,L,R);
g_ans(mid+,r, rt<<|, L,R);
}
int main(){
int t; scanf("%d", &t);
while(t--){ scanf("%d%d", &n, &m);
scanf("%u%u%u", &X, &Y, &Z);
build(,n,);
// debug(X);
for(int i=; i<=m; i++){
ll s = RNG61();
ll t = RNG61();
int tmp1 = s%n+;
int tmp2 = t%n+; int le = min(tmp1 , tmp2);
int ri = max(tmp1, tmp2);
ll v = RNG61() % MOD;
update(,n,,le,ri,v);
}
ans = 0ll;
g_ans(,n,,,n);
printf("%lld\n", ans);
} return ;
}

线段树

由于询问只有一次,就是最后的输出,所以可以用ST表,这道题算是一个逆向的构造ST表,推出dp【0】【i】的结果;

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <string>
#include <vector>
#include <map>
#include <set>
#include <queue>
#include <list>
#include <cstdlib>
#include <iterator>
#include <cmath>
#include <iomanip>
#include <bitset>
#include <cctype>
#include <iostream>
using namespace std;
//#pragma comment(linker, "/STACK:102400000,102400000") //c++
#define lson (l , mid , rt << 1)
#define rson (mid + 1 , r , rt << 1 | 1)
#define debug(x) cerr << #x << " = " << x << "\n";
#define pb push_back
#define pq priority_queue typedef long long ll;
typedef unsigned long long ull; typedef pair<ll ,ll > pll;
typedef pair<int ,int > pii; //priority_queue<int> q;//这是一个大根堆q
//priority_queue<int,vector<int>,greater<int> >q;//这是一个小根堆q
#define fi first
#define se second
// #define endl '\n' #define OKC ios::sync_with_stdio(false);cin.tie(0)
#define FT(A,B,C) for(int A=B;A <= C;++A) //用来压行
#define REP(i , j , k) for(int i = j ; i < k ; ++i)
//priority_queue<int ,vector<int>, greater<int> >que; const ll mos = 0x7FFFFFFF; //
const ll nmos = 0x80000000; //-2147483648
const int inf = 0x3f3f3f3f;
const ll inff = 0x3f3f3f3f3f3f3f3f; // template<typename T>
inline T read(T&x){
x=;int f=;char ch=getchar();
while (ch<''||ch>'') f|=(ch=='-'),ch=getchar();
while (ch>=''&&ch<='') x=x*+ch-'',ch=getchar();
return x=f?-x:x;
}
// #define _DEBUG; //*//
#ifdef _DEBUG
freopen("input", "r", stdin);
// freopen("output.txt", "w", stdout);
#endif
/*-----------------------show time----------------------*/
unsigned X,Y,Z,W;
unsigned RNG61(){
X = X ^ (X<<);
X = X ^ (X>>);
X = X ^ (X<<);
X = X ^ (X>>);
W = X ^ (Y ^ Z);
X = Y;
Y = Z;
Z = W;
return Z;
} const ll MOD = (<<);
const int maxn = 1e5+;
int n,m; ll dp[][maxn];
int Log[maxn];
void update(int le,int ri,ll val){
int k = Log[ri-le+];
dp[k][le] = max(dp[k][le], val);
dp[k][ri-(<<k)+] = max(val,dp[k][ri-(<<k)+]);
}
void solve(){
cin>>n>>m>>X>>Y>>Z;
for(int j=; j<;j++){
for(int i=; i<=n; i++){
dp[j][i] = ;
}
} for(int i=; i<=m; i++){
int t1 = RNG61()%n + ;
int t2 = RNG61()%n + ;
int le = min(t1, t2);
int ri = max(t1, t2);
ll val = RNG61() % MOD; update(le,ri,val);
// cout<<val<<endl;
} for(int j=; j; j--){
for(int i=; i+(<<(j-)) <=n; i++){
dp[j-][i] = max(dp[j][i] , dp[j-][i]);
dp[j-][i+(<<(j-))] = max(dp[j-][i+(<<(j-))] ,dp[j][i]);
}
} ll ans = ;
for(int i=; i<=n; i++){
ans = ans ^ (i * dp[][i]);
// cout<<dp[0][i]<<" ";
}
// cout<<endl;
cout<<ans<<endl;
} int main(){
OKC;
Log[] = ;
for(int i=; i<maxn; i++){
Log[i] = Log[i>>] + ;
}
int t; cin>>t; while(t--){
solve();
}
return ;
}

ST表

HDU-6356 Glad You Came 线段树 ST表的更多相关文章

  1. HDU 6356.Glad You Came-线段树(区间更新+剪枝) (2018 Multi-University Training Contest 5 1007)

    6356.Glad You Came 题意就是给你一个随机生成函数,然后从随机函数里确定查询的左右区间以及要更新的val值.然后最后求一下异或和就可以了. 线段树,区间最大值和最小值维护一下,因为数据 ...

  2. 51Nod.1766.树上最远点对(树的直径 RMQ 线段树/ST表)

    题目链接 \(Description\) 给定一棵树.每次询问给定\(a\sim b,c\sim d\)两个下标区间,从这两个区间中各取一个点,使得这两个点距离最远.输出最远距离. \(n,q\leq ...

  3. luoguP5108 仰望半月的夜空 [官方?]题解 后缀数组 / 后缀树 / 后缀自动机 + 线段树 / st表 + 二分

    仰望半月的夜空 题解 可以的话,支持一下原作吧... 这道题数据很弱..... 因此各种乱搞估计都是能过的.... 算法一 暴力长度然后判断判断,复杂度\(O(n^3)\) 期望得分15分 算法二 通 ...

  4. CF803G-Periodic RMQ Problem【离散化,线段树,ST表】

    正题 题目链接:https://www.luogu.com.cn/problem/CF803G 题目大意 一个长度为\(n\)的序列\(a\)复制\(k\)份连接,要求支持 区间赋值 区间查询最小值 ...

  5. 线段树模板(HDU 6356 Glad You Came)

    题目: HDU 6356 http://acm.hdu.edu.cn/showproblem.php?pid=6356 很裸的线段树 #include<bits/stdc++.h> #de ...

  6. HDU 3016 Man Down (线段树+dp)

    HDU 3016 Man Down (线段树+dp) Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Ja ...

  7. HDU.5692 Snacks ( DFS序 线段树维护最大值 )

    HDU.5692 Snacks ( DFS序 线段树维护最大值 ) 题意分析 给出一颗树,节点标号为0-n,每个节点有一定权值,并且规定0号为根节点.有两种操作:操作一为询问,给出一个节点x,求从0号 ...

  8. HDU.1556 Color the ball (线段树 区间更新 单点查询)

    HDU.1556 Color the ball (线段树 区间更新 单点查询) 题意分析 注意一下pushdown 和 pushup 模板类的题还真不能自己套啊,手写一遍才行 代码总览 #includ ...

  9. HDU.1166 敌兵布阵 (线段树 单点更新 区间查询)

    HDU.1166 敌兵布阵 (线段树 单点更新 区间查询) 题意分析 加深理解,重写一遍 代码总览 #include <bits/stdc++.h> #define nmax 100000 ...

随机推荐

  1. vuex的实现

    源代码可以查看我的github:  https://github.com/Jasonwang911/TryHardEveryDay/tree/master/Vuex/vuex-resouce  欢迎s ...

  2. 【JDK】JDK源码分析-LinkedList

    概述 相较于 ArrayList,LinkedList 在平时使用少一些. LinkedList 内部是一个双向链表,并且实现了 List 接口和 Deque 接口,因此它也具有 List 的操作以及 ...

  3. DesignPattern系列__06迪米特原则

    迪米特原则定义 迪米特原则,也叫最少知道原则,即一个类应该对自己依赖的类知道的越少越好,而你被依赖的类多么复杂,对我都没有关系.也就是说,对于别依赖的类来说,不管业务逻辑多么复杂,都应该尽量封装在类的 ...

  4. EnjoyingSoft之Mule ESB开发教程第六篇:Data Transform - 数据转换

    目录 1. 数据转换概念 2. 数据智能感知 - DataSense 3. 简单数据转换组件 3.1 Object to JSON 3.2 JSON to XML 3.3 JSON to Object ...

  5. C++学习想法

    今天是周一,今天做早操的时候舍友说准备买一本C++基础的书.我觉得这样的想法很好,突然想到自己最近几天因为自己私人原因事情很忙,蛋这不能成为我不学C++的理由.所以我在这规划了我这一周的学习进程.首先 ...

  6. koa2+vue实现登陆以及是否登陆控制

    这里我们先说说登陆以及登陆状态控制需要的插件jsonwebtoken,jsonwebtoken就可以实现token的生成与反向解密出用户数据.安装步骤: npm install jsonwebtoke ...

  7. Spring boot实战项目整合阿里云RocketMQ (非开源版)消息队列实现发送普通消息,延时消息 --附代码

    一.为什么选择RocketMQ消息队列? 首先RocketMQ是阿里巴巴自研出来的,也已开源.其性能和稳定性从双11就能看出来,借用阿里的一句官方介绍:历年双 11 购物狂欢节零点千万级 TPS.万亿 ...

  8. Ubuntu下安装php7.1的gd,mysql,pdo_mysql扩展库

    执行以下命令 # apt-get install php7.1-gd # apt-get install php7.0-mysql 重新启动 php7.1-fpm(因为我是安装的 Nginx 和 ph ...

  9. 【数学+思维】ZZULIOJ 1531: 小L的区间求和

    题目链接 题目描述 在给定的一个整数序列中,小L希望找到一个连续的区间,这个区间的和能够被k整除,请你帮小L算一下满足条件的最长的区间长度是多少. 输入 第一行输入两个整数n.k.(1 <= n ...

  10. 快速了解Python并发编程的工程实现(上)

    关于我 一个有思想的程序猿,终身学习实践者,目前在一个创业团队任team lead,技术栈涉及Android.Python.Java和Go,这个也是我们团队的主要技术栈. Github:https:/ ...