题目戳这里

首先我们对坐标进行离散化,有用的点就变成了\(O(N)\)个。我们假设\(A\)点\(B\)的右边(从\(A\)往\(B\)跑和从\(B\)往\(A\)跑等价),然后我们很容易发现不会往左跑。于是我们就可以dp了。我们用\(f[i][j]\)表示\(A\)到\((i,j)\)的最小代价(\((i,j)\)是离散后的坐标),然后我们很容易得到dp方程。

\[f[i][j] = \min \{ f[i-1][j]+Tx[i-1][j] \times (x[i]-x[i-1]),\\f[i][j+1]+Ty[i][j] \times (y[j+1]-y[j]),f[i][j-1]+Ty[i][j-1]+(y[j]-y[j-1]) \}
\]

其中\(Tx[i][j]\)表示从\((i,j)\)走到\((i+1,j)\)走一个单位所需要的时间,\(Ty[i][j]\)表示从\((i,j)\)走到\((i,j+1)\)一个单位所需要的时间。这个我们可以预处理出来(大致就是看这中间有没有点在矩形的边界或中间)。

然后若\(B\)在\(A\)下方,则还可能只向下向左向右走。我们可以旋转坐标,就跟上面的一样处理了。

#include<algorithm>
#include<cstring>
#include<iostream>
#include<cstdio>
#include<cstdlib>
using namespace std; typedef long long ll;
const int maxn = 2010; const ll inf = 1LL<<60;
int Xa,Ya,Xb,Yb,N,totx,toty,bacx[maxn],bacy[maxn],in[maxn][maxn],Lb[maxn][maxn],Db[maxn][maxn];
ll f[maxn][maxn],Tx[maxn][maxn],Ty[maxn][maxn],ans; inline int gi()
{
char ch; int ret = 0,f = 1;
do ch = getchar(); while (!(ch >= '0'&&ch <= '9')&&ch != '-');
if (ch == '-') f = -1,ch = getchar();
do ret = ret*10+ch-'0',ch = getchar(); while (ch >= '0'&&ch <= '9');
return ret*f;
} struct Node
{
int X1,Y1,X2,Y2,T;
inline void read()
{
bacx[++totx] = X1 = gi(); bacy[++toty] = Y1 = gi();
bacx[++totx] = X2 = gi(); bacy[++toty] = Y2 = gi();
T = gi();
}
inline void convert() { swap(X1,Y1); swap(X2,Y2); }
}rec[maxn]; inline int find(int arr[],int r,int key)
{
int l = 1,mid;
while (l <= r)
{
mid = (l+r) >> 1;
if (arr[mid] < key) l = mid+1;
else r = mid-1;
}
return l;
} inline void work(int xx[],int tx,int yy[],int ty)
{
for (int i = 1;i <= tx;++i) for (int j = 1;j <= ty;++j) Lb[i][j] = Db[i][j] = in[i][j] = 0;
for (int i = 1;i <= N;++i)
{
int pX1 = find(xx,tx,rec[i].X1),pX2 = find(xx,tx,rec[i].X2);
int pY1 = find(yy,ty,rec[i].Y1),pY2 = find(yy,ty,rec[i].Y2);
for (int j = pX1+1;j < pX2;++j)
for (int k = pY1+1;k < pY2;++k) in[j][k] = i;
for (int k = pY1+1;k < pY2;++k) Lb[pX1][k] = i;
for (int j = pX1+1;j < pX2;++j) Db[j][pY1] = i;
}
for (int i = 1;i <= tx;++i)
for (int j = 1;j <= ty;++j)
{
if (i < tx)
{
Tx[i][j] = 10;
if (in[i][j]) Tx[i][j] = rec[in[i][j]].T;
else if (in[i+1][j]) Tx[i][j] = rec[in[i+1][j]].T;
else if (Lb[i][j]) Tx[i][j] = rec[Lb[i][j]].T;
}
if (j < ty)
{
Ty[i][j] = 10;
if (in[i][j]) Ty[i][j] = rec[in[i][j]].T;
else if (in[i][j+1]) Ty[i][j] = rec[in[i][j+1]].T;
else if (Db[i][j]) Ty[i][j] = rec[Db[i][j]].T;
}
}
if (Xa > Xb) swap(Xa,Xb),swap(Ya,Yb); for (int i = 1;i <= tx;++i) for (int j = 1;j <= ty;++j) f[i][j] = inf;
f[find(xx,tx,Xa)][find(yy,ty,Ya)] = 0;
for (int i = 2;i <= tx;++i)
{
for (int j = 1;j <= ty;++j)
{
f[i][j] = min(f[i-1][j]+Tx[i-1][j]*(xx[i]-xx[i-1]),f[i][j]);
if (j > 1) f[i][j] = min(f[i][j],f[i][j-1]+Ty[i][j-1]*(yy[j]-yy[j-1]));
}
for (int j = ty;j;--j) if (j < ty) f[i][j] = min(f[i][j],f[i][j+1]+Ty[i][j]*(yy[j+1]-yy[j]));
}
ans = min(ans,f[find(xx,tx,Xb)][find(yy,ty,Yb)]);
} int main()
{
freopen("4374.in","r",stdin);
freopen("4374.out","w",stdout);
while (scanf("%d %d %d %d",&Xa,&Ya,&Xb,&Yb) != EOF)
{
totx = toty = 0; N = gi(); ans = inf;
bacx[++totx] = Xa; bacx[++totx] = Xb;
bacy[++toty] = Ya; bacy[++toty] = Yb;
for (int i = 1;i <= N;++i) rec[i].read(); sort(bacx+1,bacx+totx+1); sort(bacy+1,bacy+toty+1);
totx = unique(bacx+1,bacx+totx+1)-bacx-1; toty = unique(bacy+1,bacy+toty+1)-bacy-1; work(bacx,totx,bacy,toty); swap(Xa,Ya); swap(Xb,Yb);
for (int i = 1;i <= N;++i) rec[i].convert();
work(bacy,toty,bacx,totx);
cout << ans << endl;
}
fclose(stdin); fclose(stdout);
return 0;
}

UVALive4374 Drive through MegaCity的更多相关文章

  1. AHCI: Failed to attach drive to Port1 (VERR_GENERAL_FAILURE).

    在mac操作系统下,安装VirtualBoxVm虚拟机,虚拟机里面安装wind7操作系统.在启动虚拟机的时候报错:AHCI: Failed to attach drive to Port1 (VERR ...

  2. vmware安装win7提示No CD-ROM drive to use:GCDROM not loaded

    今天安装win7 64位的操作系统到vmware虚拟机,以为一切事如此的简单,因为自己以前经常拿vmware来装系统,结果确出现下面莫名其妙的错误: 提示说没有CD-ROM,可是我明明在vmware的 ...

  3. Ignite 配置更新Oracle JDBC Drive

           如果使用Oracle 12C 作为Ignite 的Repository的话,在Repository Createion Wizard的配置过程中,会出现ORA-28040:No matc ...

  4. 网盘的选择,百度网盘、google drive 还是 Dropbox

    我是国内用户,需要越过Chinawall 我使用的是一枝红杏,用着还行 如果要买,结账时输入'laod80' 一枝红杏官网:官网地址 Dropbox: 稳定,速度快 Dropbox官网:链接 操作十分 ...

  5. UEFI Bootable USB Flash Drive - Create in Windows(WIN7 WIN8)

    How to Create a Bootable UEFI USB Flash Drive for Installing Windows 7, Windows 8, or Windows 8.1 In ...

  6. Create a SQL Server Database on a network shared drive

    (原文地址:http://blogs.msdn.com/b/varund/archive/2010/09/02/create-a-sql-server-database-on-a-network-sh ...

  7. 【转】Expire Google Drive Files 让Google Docs云盘共享连接在指定时间后自动失效

    最近在清理Google Docs中之前共享过的文件链接,发现Google Docs多人协作共享过的链接会一直存在,在实际操作中较不灵活.正好订阅的RSS推送了Pseric写的这篇文章 - Expire ...

  8. vs2010 A selected drive is no longer valid

    visual studio 2010重新安装添加组件,报A selected drive is no longer valid错误. 这个是由于已经安装了sp1,此时需要将sp1卸载掉,然后就可以安装 ...

  9. 24.编写一个Car类,具有String类型的属性品牌,具有功能drive; 定义其子类Aodi和Benchi,具有属性:价格、型号;具有功能:变速; 定义主类E,在其main方法中分别创建Aodi和Benchi的对象并测试对象的特 性。

    package zhongqiuzuoye; public class Car { String brand; public void drive() {} } package zhongqiuzuo ...

随机推荐

  1. OCCI的迭代修改

    传统的在执行多行DML(INSERT.UPDATE.DELETE)时,我们是多次调用executeUpdate():注意!当我们调用一次此函数时,则执行一次网络往返,当数据量大时则效率非常低.不过 O ...

  2. Mybatis与Hibernate区别

    Mybatis与Hibernate区别 mybatis: 1. 入门简单,即学即用,提供了数据库查询的自动对象绑定功能,而且延续了很好的SQL使用经验,对于没有那么高的对象模型要求的项目来说,相当完美 ...

  3. 描述linux目录结构以及目录结构命名规定

    FHS全称(Filesystem Hierarchy Standard),中文意思是目录层次标准,是linux的目录规范标准. 详情点击查看 FHS定义了两层规范: 第一层:“/”目录下的各个目录应该 ...

  4. java 基础面试

    1. &和&&的区别 答: &是位运算符,表示按位与运算,&&是逻辑运算符,表示逻辑与(and) 2.int 和 Integer 有什么区别 答: Ja ...

  5. MARK 一条关于Linux 运维方面个人向收藏网址

    吴钧泽博客 https://wujunze.com/archives.html Linux运维笔记 https://blog.linuxeye.cn/ Linux中文网 http://www.ppze ...

  6. 初见spark-04(高级算子)

    今天,这个是spark的高级算子的讲解的最后一个章节,今天我们来介绍几个简单的算子, countByKey val rdd1 = sc.parallelize(List(("a", ...

  7. sql查询作业答案

    sql查询作业答案   阅读目录 一 题目 二 答案 一 题目 1.查询所有的课程的名称以及对应的任课老师姓名 2.查询学生表中男女生各有多少人 3.查询物理成绩等于100的学生的姓名 4.查询平均成 ...

  8. Delphi中的关键字与保留字

    Delphi中的关键字与保留字 分类整理 Delphi 中的“关键字”和“保留字”,方便查询 感谢原作者的收集整理! 关键字和保留字的区别在于,关键字不推荐作标示符(编译器已经内置相关函数或者留给保留 ...

  9. 什么是 IRC?

    IRC是Internet Relay Chat 的英文缩写,中文一般称为互联网中继聊天.它是由芬兰人Jarkko Oikarinen于1988年首创的一种网络聊天协议.经过十年的发展,目前世界上有超过 ...

  10. Linux忘记root密码的解决办法

    这里以centos6为例: 第一步:先将系统重新启动,在读秒的时候按下任意键就会出现如下图的菜单界面: 第二步:按下『e』就能够进入grub的编辑模式,如图: 第三步:将光标移动到kernel那一行, ...