考场上以为就是转化成一个无向图然后以为无向图有什么性质可以搞出来来着。

果然应验了那句话,一个思路想太久想不出来一般是假的。

所以这种一看就需要转化的题要多尝试能往哪转化,而不是按住一个思路不动。

只要转化成求极长上升序列就能搞一个单调栈的暴力分。

神仙题一个,第二次见这么玩线段树的(\(minimax\)是第一个)

这种题果然是我这种弱鸡不配做的,考完磨了超级神犇土哥半天才搞明白。

首先第一步转化,就是把问题变成求\(p[i]\)序列里面的一个最小权极长上升子序列。

因为满足极长上升,所以一定是把所以跟他没有交集的线都选了,导致没选的都是跟他有交集的,那肯定都被消掉了。

求这个东西,可以发现如果我们设计状态\(f[i]\)表示选择\(i\)时的上升子序列,那么对于每个\(i\),他的转移点一定是随着\(i\)增单调减的。

所以大佬\(zero4338\)直接维护了一下转移点,然后直接跳转移点,并通过了本题。

但是发现如果一个数前面的东西都是单调减的,那么转移点会有很多,导致直接跳会超时。

所以这个东西,我们其实是想动态的维护一个单调栈。

因为对于每个\(i\),都需要找出若干个\(p[j]\)单调递减的\(j\)并且满足\(\forall p[j]<p[i]\)。

但是发现每个\(p[i]\)都不同,弄单调栈需要每次都重新扫一遍。

所以出现了本场主角——线段树维护单调栈。

\(\%\%\%\)学长写的\(blog\),写的很清楚,放个链接

这个东西就是通过递归定义让\(calc\)函数和我们存的一个值不断套娃得到的解。

在原来那道题(楼房修建)中,我们开了一个额外的域,\(len\)。

保存\(x\)节点的右区间中大于左区间最大值并且单调上升的长度。

这是由于题目要求,如果一个楼房对应的斜率比较高,我们不能选择不看到它,因为我们不能透视不看他看到被他挡住的。

这句话看似很废,却是可以用这个方法解题的关键。

因为左边会对右边造成影响,所以我们需要把关键信息给被造成影响的区间,也就是右区间。

正是因为这个,所以我们在把\(len\)定义上上面那样,这样才能合并区间信息。

所以,在这道题中,我们需要动态维护比一个数小的单调递减(从1开始)的序列中\(dp\)值的最小值

其实如果我们从\(i\)开始来看他的话,他就是一个单调递增的序列。

和上面那道题一样,为了设计出来那个额外域的意义,需要观察题目的性质。

其实就是如果下标大的数值大,我们必须选择他

画一个图来表示一下就是这样的。

假如下面小于\(p[i]\)是左区间和右区间的单调部分

我们需要选择

而不是

反正就是从\(i\)往\(1\)走的过程中只要比当前栈顶的\(p[i]\)大就必须入到栈中。

所以说,我们需要把状态设计成左区间的信息,因为右边会对左边造成影响。

结合题目要求,我们额外域里面存的就是

这个区间的左区间中大于右区间最大值并且满足单调递减的部分的\(dp\)值的最小值

\(calc\)为了和他套娃,需要计算

一个区间所有数都大于\(val\)并且满足单调递减的\(dp\)值的最小值

所以很好写出\(pushup\)和\(calc\)

int calc( int x , int val ){ // x的节点的区间中满足单调递减大于val的值数中的dp值的最小值
if( l [x] == r [x] ){
if( data [x].maxv > val ) return f [rl [l [x]]]; //f [rl [l [x]]]的意思就是这个位置的dp值
else return Inf ; //不满足大于val,return inf
}
if( data [x << 1 | 1].maxv > val ) return min( data [x].ans , calc( x << 1 | 1 , val ) ) ;
// 右区间最大值大于val,去递归右区间并且直接加上左区间之前计算好的贡献
else return calc( x << 1 , val ) ;
//否则直接去左区间找就行了,因为右区间最大值都不合法。
} inline void ud( int x ){ //push_up
data [x].maxv = max( data [x << 1].maxv , data [x << 1 | 1].maxv ) ;
data [x].ans = calc( x << 1 , data [x << 1 | 1].maxv ) ; //套娃从这里开始
}

然后发现还有一个问题,如果我们线段树中保存下标的是\(i\)的话,我们每次都需要满足找到的东西小于\(p[i]\)这个要求。

很难实现。

所以把下标设置成\(p[i]\),所以只用让\(i\)满足单调递减就行了,又因为我们是从\(1\)到\(i\)加进来的,所以大于\(i\)的线段树中肯定没有。

这样就解决了\(p[i]\)的限制,也是一步很神仙的转化(本弱想不到系列)。

然后\(ask\)的时候直接找从\(1\)到\(p[i]\)就行了,但是由于我们需要把线段树上的区间拼接起来,所以需要先去右边,再搜左边。

记录一下右边搜到的最大值,层层加限制就行了。

然后就可以\(logn\)计算出来\(f[i]\)了,最后直接找一下看哪个点可以做极长上升序列的结尾点,并统计他对\(ans\)的贡献就行了。

code
#include <cstdio>
#include <algorithm>
#include <cstring>
#define R register int
#define int long
#define printf Ruusupuu = printf
#define scanf Ruusupuu = scanf
#define freopen rsp_5u = freopen int Ruusupuu ;
FILE * rsp_5u ; using namespace std ;
typedef long long L ;
typedef long double D ;
typedef pair<int , int> PI ;
const int N = 2e5 + 10 ;
const int Inf = 0x3f3f3f3f ; inline void of(){ freopen( "in.in" , "r" , stdin ) , freopen( "out.out" , "w" , stdout ) ; }
inline void cf(){ fclose( stdin ) , fclose( stdout ) ; }
inline int read(){
int w = 0 ; bool fg = 0 ; char ch = getchar() ;
while( ch < '0' || ch > '9' ) fg |= ( ch == '-' ) , ch = getchar() ;
while( ch >= '0' && ch <= '9' ) w = ( w << 1 ) + ( w << 3 ) + ( ch - '0' ) , ch = getchar() ;
return fg ? -w : w ;
} bool fg [N] ;
int n , m , rl [N] ;
int p [N] , w [N] , f [N] ;
int l [N << 2] , r [N << 2] , lx , rx , pos , dlt ;
struct T{ int maxv , ans ; } data [N << 2] ;
//data [x].ans 保存,x的左边区间 大于 x的右边区间的最大值的 并且单调递减 的部分内 dp值的最小值 int calc( int x , int val ){ // x的节点的区间中满足单调递减大于val的值数中的dp值的最小值
if( l [x] == r [x] ){
if( data [x].maxv > val ) return f [rl [l [x]]];
else return Inf ;
}
if( data [x << 1 | 1].maxv > val ) return min( data [x].ans , calc( x << 1 | 1 , val ) ) ;
else return calc( x << 1 , val ) ;
} inline void ud( int x ){
data [x].maxv = max( data [x << 1].maxv , data [x << 1 | 1].maxv ) ;
data [x].ans = calc( x << 1 , data [x << 1 | 1].maxv ) ;
} void build( int x , int ll , int rr ){
l [x] = ll , r [x] = rr ;
data [x].ans = Inf , data [x].maxv = -Inf ;
if( ll == rr ) return ;
int mid = ( ll + rr ) >> 1 ;
build( x << 1 , ll , mid ) ;
build( x << 1 | 1 , mid + 1 , rr ) ;
ud( x ) ;
} int nowmax ; int ask( int x ){
if( l [x] >= lx && r [x] <= rx ){
int t = calc( x , nowmax ) ;
nowmax = max( nowmax , data [x].maxv ) ; //更新nowmax,为了满足单调性,接下来搜到的单调序列里面的数不能小于nowmax
return t ;
}
int mid = ( l [x] + r [x] ) >> 1 , ans = 0x3f3f3f3f ;
if( rx > mid ) ans = min( ans , ask( x << 1 | 1 ) ) ;
if( lx <= mid ) ans = min( ans , ask( x << 1 ) ) ;
return ans ;
} void cge( int x ){
if( l [x] == r [x] ){ data [x].maxv = rl [l [x]] ; return ; }
int mid = ( l [x] + r [x] ) >> 1 ;
if( pos <= mid ) cge( x << 1 ) ;
else cge( x << 1 | 1 ) ;
ud( x ) ;
} inline int Ask( int x ){
lx = 1 , rx = p [x] , nowmax = 0 ;
int t = ask( 1 ) ;
return ( ( t == Inf ) ? 0 : t ) ;
} void sc(){
n = read() ;
for( R i = 1 ; i <= n ; i ++ ) p [i] = read() , rl [p [i]] = i ;
for( R i = 1 ; i <= n ; i ++ ) w [i] = read() ;
memset( f , 0x3f , sizeof( f ) ) ;
build( 1 , 1 , n ) ;
} void work(){
int maxn = 0 , ansn = 0x3f3f3f3f ;
for( R i = n ; i >= 1 ; i -- )
if( p [i] > maxn ) maxn = p [i] , fg [i] = 1 ; for( R i = 1 ; i <= n ; i ++ ){
f [i] = Ask( i ) + w [i] , pos = p [i] , dlt = f [i] , cge( 1 ) ;
if( fg [i] ) ansn = min( ansn , f [i] ) ;
}
printf( "%ld\n" , ansn ) ;
} signed main(){
// of() ;
sc() ;
work() ;
// cf() ;
return 0 ;
}

题解—God Knows的更多相关文章

  1. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

  2. noip2016十连测题解

    以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...

  3. BZOJ-2561-最小生成树 题解(最小割)

    2561: 最小生成树(题解) Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1628  Solved: 786 传送门:http://www.lyd ...

  4. Codeforces Round #353 (Div. 2) ABCDE 题解 python

    Problems     # Name     A Infinite Sequence standard input/output 1 s, 256 MB    x3509 B Restoring P ...

  5. 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解

    题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...

  6. 2016ACM青岛区域赛题解

    A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Jav ...

  7. poj1399 hoj1037 Direct Visibility 题解 (宽搜)

    http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...

  8. 网络流n题 题解

    学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...

  9. CF100965C题解..

    求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...

  10. JSOI2016R3 瞎BB题解

    题意请看absi大爷的blog http://absi2011.is-programmer.com/posts/200920.html http://absi2011.is-programmer.co ...

随机推荐

  1. Java+Selenium3.3.1环境搭建

    一.背景和目的 selenium从2.0开始,加入了webdriver,实际上,我们说的selenium自动化测试,大部分情况都是在使用webdriver的API.现在去Selenium官网,发现最新 ...

  2. 构造函数 析构函数的区别与联系 C#

    构造函数 __construct:是在对象被创建是自动调用的方法,用来完成初始化操作 构造函数有以下特点:1.构造函数的名字必须与类名相同:2.构造函数可以有任意类型的参数,但不能具有返回类型:3.定 ...

  3. Requests方法 -- session方法

    import requests#禁用安全请求警告from requests.packages.urllib3.exceptions import InsecureRequestWarningreque ...

  4. ecshop二次开发笔记--订单表结构ecs_order_info说明

    -- 表的结构 `ecs_order_info`  CREATE TABLE IF NOT EXISTS `ecs_order_info` (  `order_id` mediumint(8) uns ...

  5. 使用Angular CDK实现一个Service弹出Toast组件

    在Angular中,官方团队在开发Material组件库的同时,顺手做了一套Component dev kit,也就是在Angular世界中大名鼎鼎的CDK,这套工具包提供了非常多的前端开发的通用功能 ...

  6. Git的使用(六)

    前言 版本管理工具总结: 开发团队项目,对项目的版本进行管理. 使用过的版本管理工具: TFS.SVN与Git. TFS:管理项目,通过visual Studio管理源码,拉取分支,提交代码等.也可以 ...

  7. 第一篇 -- 下载并安装IDEA

    此篇讲的是安装IDEA企业版,社区版是免费的,就不多说了. 参考链接:https://www.exception.site/essay/how-to-free-use-intellij-idea-20 ...

  8. 构建前端第6篇之---内嵌css样式 <el-button style="width:100%"> 登录 </el-button>

    张艳涛写于2021-1-20日 What: 如何让button的长度和input长度一致呢 最先想到的是给这个button加一个class ="buttonclass",然后在vu ...

  9. Fast Run:提高 MegEngine 模型推理性能的神奇功能

    作者:王博文 | 旷视 MegEngine 架构师 一.背景 对于深度学习框架来说,网络的训练/推理时间是用户非常看中的.在实际生产条件下,用户设计的 NN 网络是千差万别,即使是同一类数学计算,参数 ...

  10. Java中Arrays数组的定义与使用

    初始化 Java中数组是固定长度,数组变量是个对象. NullPointerException 空指针异常. ArrayIndexOutOfBoundsException 索引值越界. 数组三种初始化 ...