2017 多校4 Security Check
2017 多校4 Security Check
题意:
有\(A_i\)和\(B_i\)两个长度为\(n\)的队列过安检,当\(|A_i-B_j|>K\)的时候, \(A_i和B_j\)是可以同时过安检的,过安检必须按照队列的顺序,问你两个队列过完安检最少花费的时间
\(1<=n<=6e4\),
\(1<=A_i,B_i<=n\)
题解:
设\(f_{i,j}\)
表示仅考虑\(a[1..i]\)与\(b[1..j]\)时,最少需要多少时间。
若\(|a_i-b_j|>k\)
,则\(f_{i,j}=f_{i-1,j-1}+1\)否则\(f_{i,j}=\min(f_{i-1,j},f_{i,j-1})+1\)
注意到后者只有\(O(nk)\)个,可以暴力DP,前者可以通过二分找到最大的\(t\),满足\(i,j\)往前\(t\)个均不冲突,然后再从某个后者状态转移过来。
时间复杂度\(O(nk\log n)\)。
这个\(dp\)就跟求最长公共子序列的dp一样,很容易想到\(O(n^{2})\)的做法,优化就不容易想到了,想到了也不知道如何实现
对于\(|A_i-B_j|<=K\)的时候 可以暴力\(dp\),
否则在\(i\)的左边找一个最接近\(i\)的\(t\) 满足\(|A_t-B_{j-i+t}|<=K\)
那么在\(t<p<=i\) 这一段 都是满足\(|A_p-B_{j-i+p}|>K\)他们同时过安检显然是最优的,直接算即可,这样下次有跳跃到了\(<=K\)的情况,所以只需要\(做dp\) 计算\(<=K\)的情况
如何找到这个\(t\)呢,开始我直接写了一发二分下标,WA,没仔细思考,这样二分下标不具有单调性。去网上看了看别人的写法,是把所有\(i-j\)差值为\(x\)而且\(|A_i-B_j|<=K\)的\(i\)存起来,然后去二分这个\(i\),好巧的做法,因为\(i,j\)一直往左走,走到\(t\),差值都是相同的,这样是可行的。
#include<bits/stdc++.h>
#define LL long long
#define P pair<int,int>
using namespace std;
const int N = 6e4 + 10;
int read(){
    int x = 0;
    char c = getchar();
    while(c < '0' || c > '9') c = getchar();
    while(c >= '0' && c <= '9') x = x * 10 + c - 48, c = getchar();
    return x;
}
int n,k;
int a[N],b[N],pos[N];
vector<int> v[2 * N];///下标差值为x且绝对值小于等于k的 a_i的位置
int dp[N][22];
int Find(int i,int j){///i左边找到第一个x 使得abs(a[x] - b[j - i + x])<=k
   int x = upper_bound(v[i - j + n].begin(),v[i - j + n].end(),i) - v[i - j + n].begin() - 1;
   if(x < 0) return 0;
   return v[i-j+n][x];
}
int dfs(int i,int j){
    if(!i || !j) return i + j;
    if(abs(a[i] - b[j]) <= k){
        int &res = dp[i][a[i] - b[j] + k];
        if(res != -1) return res;
        return res = min(dfs(i-1,j),dfs(i,j-1)) + 1;
    }
    int x = Find(i,j);
    return dfs(x,max(0,j - i + x)) + i - x;
}
int main(){
    int T;
    T = read();
    while(T--){
        n = read(),k = read();
        for(int i = 1;i <= n;i++) a[i] = read();
        for(int i = 1;i <= n;i++) b[i] = read(),pos[b[i]] = i;
        for(int i = 0;i <= 2 * n;i++) v[i].clear();
        for(int i = 1;i <= n;i++){
            for(int j = max(1,a[i] - k);j <= min(a[i] + k,n);j++){
                v[i - pos[j] + n].push_back(i);
            }
        }
        memset(dp, -1, sizeof(dp));
        printf("%d\n",dfs(n,n));
    }
    return 0;
}
2017 多校4 Security Check的更多相关文章
- HDU 6076 - Security Check   |  2017 Multi-University Training Contest 4
		/* HDU 6076 - Security Check [ DP,二分 ] | 2017 Multi-University Training Contest 4 题意: 给出两个检票序列 A[N], ... 
- hdu6097[二分+解析几何] 2017多校6
		/*hdu6097[二分+解析几何] 2017多校6*/ #include <bits/stdc++.h> using namespace std; ; struct node{ doub ... 
- hdu6076 Security Check 分类dp 思维
		/** 题目:hdu6076 Security Check 链接:http://acm.hdu.edu.cn/showproblem.php?pid=6076 题意:有两个队列在排队,每一次警察可以检 ... 
- 2017 多校5 hdu 6093  Rikka with Number
		2017 多校5 Rikka with Number(数学 + 数位dp) 题意: 统计\([L,R]\)内 有多少数字 满足在某个\(d(d>=2)\)进制下是\(d\)的全排列的 \(1 & ... 
- 2017 多校5 Rikka with String
		2017 多校5 Rikka with String(ac自动机+dp) 题意: Yuta has \(n\) \(01\) strings \(s_i\), and he wants to know ... 
- 2017 多校4 Wavel Sequence
		2017 多校4 Wavel Sequence 题意: Formally, he defines a sequence \(a_1,a_2,...,a_n\) as ''wavel'' if and ... 
- 2017 多校3 hdu 6061 RXD and functions
		2017 多校3 hdu 6061 RXD and functions(FFT) 题意: 给一个函数\(f(x)=\sum_{i=0}^{n}c_i \cdot x^{i}\) 求\(g(x) = f ... 
- 2017 多校2 hdu 6053 TrickGCD
		2017 多校2 hdu 6053 TrickGCD 题目: You are given an array \(A\) , and Zhu wants to know there are how ma ... 
- 2017 多校1 I Curse Myself
		2017 多校2 I Curse Myself(第k小生成树) 题目: 给一张带权无向连通图,该图的任意一条边最多只会经过一个简单环,定义\(V(k)为第k小生成树的权值和\),求出\(\sum_{k ... 
随机推荐
- 使用工具Android Studio实现一个简单的Android版的新闻APP
			目的: 这是我学完Android课程后所写的一个小的.简单版的新闻APP 技术概要: 用到了SQLite数据库,用它来存储每篇新闻下的评论 新闻的来源是新浪新闻,我通过使用Fiddler来对新浪新闻A ... 
- poj_3641_Pseudoprime numbers
			Fermat's theorem states that for any prime number p and for any integer a > 1, ap = a (mod p). Th ... 
- Java的“Goto”与标签
			goto在Java中是一个保留字,但在语言中并没有用到它:Java没有goto.但是,Java也能完成一些类似于跳转的操作,主要是依靠:标签. 为什么要使用标签 在迭代语句中,我们可以使用break和 ... 
- 简单php实现同一时间内一个账户只允许在一个终端登陆
			在账户表的基础上,我新建了一个账户account_session表,用来记录登录账户的account_id和最新一次登录成功用户的session_id,然后首先要修改登录方法:每次登录成功后,要将登录 ... 
- python学习之列表和元组
			配置环境:python 3.6 python编辑器:pycharm,代码如下: #!/usr/bin/python # -*- coding: UTF-8 -*- # list:是一种有序的集合,可以 ... 
- 硬盘安装Windows Server 2008(解决系统盘符变成D盘)
			硬盘安装Windows 2008系统方法 操作系统最好用的无疑是server 2003,但是现在Server 2003支持的软件越来越少,很多是故意不支持Server 2003了, 像php5.5以上 ... 
- Ball CodeForces - 12D
			传送门 N ladies attend the ball in the King's palace. Every lady can be described with three values: be ... 
- [Uva11178]Morley's Theorem(计算几何)
			Description 题目链接 Solution 计算几何入门题 只要求出三角形DEF的一个点就能推出其他两个点 把一条边往内旋转a/3度得到一条射线,再做一条交点就是了 Code #include ... 
- SpringMVC文件上传——bean的配置【org.springframework.web.multipart.commons.CommonsMultipartResolver】
			一.简介 Spring MVC支持一个通用的多路上传解析器CommonsMultipartResolver,在Spring的配置文件中对CommonsMultipartResolver Bean进行配 ... 
- TouTiao开源项目 分析笔记2
			1.Constant常量定义类 1.1.源代码 public class Constant { public static final String USER_AGENT_MOBILE = " ... 
