2018宁夏acm网络赛-G-Trouble of Tyrant-斜率优化-决策单调性

2023-11-06 19:50

本文主要是介绍2018宁夏acm网络赛-G-Trouble of Tyrant-斜率优化-决策单调性,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

(有任何问题欢迎留言或私聊 && 欢迎交流讨论哦

题目:传送门

 (原题目描述及样例在最下面。)

 我的描述请看图:
这里写图片描述
 无向图,n个节点,2n-3条边。n-1条边从1->i .(2<=i<=n)。n-2条边从i->i+1.(2<=i<=n-1)
 k次询问,问将每条边长度增加d后,从1到n的最短路径。


思路:

 非常明显要用到斜率优化这一技巧。

(显然为了方便,要将点从新逆序编号)
ar[i]表示从1经过i条边到达点n,所经过的路径长度。
dp[i]表示每条边增加长度d后,新的ar[i]值。

第一步:

假设i > j 且ar[i] < ar[j],如果边增加长度d后,在点 i 的决策劣于在点 j 的决策,则:

dp[i]=ar[i]+id dp[j]=ar[j]+jd 且 dp[i] > dp[j]
=> ar[i]-ar[j] >= d*(j-i)
=> (ar[i]-ar[j])/(i-j) >= -d

将上述式子定义为i和j的斜率。注意:j < i。
 如果在点 i 处决策劣于在点 j 处决策,则有:斜率(j, i) >= -d

(ar[i]-ar[j])/(i-j) >= -d

第二步:

所以我们就利用单调队列构造一个斜率上升的凸包,注意斜率是负的。大概长这个样子:
这里写图片描述

因为构造出来的是斜率变大的下凸包,它的决策单调性的优度是递减的。
为了保证计算每个 d 的答案正确。
我选择把d从小到大排序,这样 -d 就是从大到小排序。
最后从构造出来的凸包末尾开始依次计算排序后的 -d 的值。

从开头还是从末尾开始计算不同d值下的最短路径,是差不多的,改一下细节就可以了,两种方法我都过了这题。(从开头开始计算要快一点点)

具体看代码实现。

结构体c数组储存路径的长度及边数。
结构体op数组储存d值
结构体数组que为队列


AC代码
/*
**链接**
传送门: [here](https://nanti.jisuanke.com/t/A1670)
**题意**
$N,K 1e5$
无向图,n个节点,2n-3条边。n-1条边从1->i .(2<=i<=n)。n-2条边从i->i+1.(2<=i<=n-1)k次询问,问将每条边长度增加d后,从1到n的最短路径。
**思路**
[2018宁夏acm网络赛-G-Trouble of Tyrant](https://blog.csdn.net/qq_39599067/article/details/80671353)
类似题:[BZOJ1010](https://blog.csdn.net/qq_39599067/article/details/80668515)
**备注**
*/
const int MXN = 2e6 + 5;
const int MXE = 2e6 + 6;int n, m;
int64 dp[MXN], ar[MXN], sum[MXN], id[MXN], qry[MXN], ans[MXN];
int Q[MXN], hd, tl;
void read_data() {n = read(), m = read();for(int i = n - 1; i >= 1; --i) ar[i] = read();for(int i = n - 1; i >= 2; --i) sum[i] = read();for(int i = 2; i < n; ++i) sum[i] += sum[i - 1];for(int i = 1; i < n; ++i) ar[i] += sum[i];for(int i = 1; i <= m; ++i) {id[i] = i;qry[i] = read();}
}
int64 KX(int j, int k) {return ar[k] - ar[j];
}
int64 KY(int j, int k) {return k - j;
}
bool cmp1(const int &a, const int &b) {return qry[a] < qry[b];
}
bool cmp2(const int &a, const int &b) {return qry[a] > qry[b];
}
void gao_solve() {sort(id + 1, id + m + 1, cmp1);hd = tl = 0;for(int i = 1; i < n; ++i) {if(tl > hd && ar[i] >= ar[Q[tl - 1]]) continue;while(tl - hd >= 2 && KX(Q[tl-2], Q[tl-1]) * KY(Q[tl-1], i) >= KX(Q[tl-1], i) * KY(Q[tl-2], Q[tl-1])) -- tl;Q[tl++] = i;}for(int i = 1; i <= m; ++i) {while(tl - hd >= 2 && KX(Q[tl - 2], Q[tl - 1]) >= KY(Q[tl - 2], Q[tl - 1]) * (-qry[id[i]])) -- tl;ans[id[i]] = ar[Q[tl - 1]] + Q[tl - 1] * qry[id[i]];}
}
void print_ans() {for(int i = 1; i <= m; ++i) printf("%lld%c", ans[i], " \n"[i==m]);
}
int main() {
#ifdef LH_LOCALfreopen("D:in.in", "r", stdin);freopen("D:out.out", "w", stdout);
#endifint tim = read();while(tim --) {read_data();gao_solve();print_ans();}
#ifdef LH_LOCAL// cout << "time cost:" << 1.0 * clock() / CLOCKS_PER_SEC << "s" << endl;// system("pause");
#endifreturn 0;
}
#include<bits/stdc++.h>
#define fang(a) (a)*(a)
using namespace std;
typedef long long LL;
const int N = 100000+7;
int n,k,head,tail;
LL ar[N],f[N],d[N];
struct lp{int id;LL v;
}op[N],c[N],que[N];
bool cmp1(lp &a,lp &b){return a.v>b.v;
}
bool cmp2(lp &a,lp &b){return a.v<b.v;
}
/*
i>j
dp[i]=ar[i]+i*k
dp[j]=ar[j]+j*k
dp[i] >= dp[j]
ar[i]-ar[j] >= k*(j-i)
(ar[i]-ar[j])/(i-j) >= -k
维持一个斜率单调上升的队列(斜率是负的)
把k从小到大排序,则-k从大到小排序
从队列末尾开始匹配-k
*/
bool cd(lp a,lp b,lp c){return ((b.id-a.id)*(c.v-a.v)-(b.v-a.v)*(c.id-a.id))<0;
}
double fx(lp x,lp y){return (y.v-x.v*1.0)/(y.id-x.id);
}
int main(){int tim;scanf("%d",&tim);while(tim--){scanf("%d%d",&n,&k);/***********************/for(int i=n-1;i>=1;--i){scanf("%lld",&ar[i]);}for(int i=n-1;i>=2;--i){scanf("%lld",&d[i]);}for(int i=3;i<=n-1;++i){d[i]+=d[i-1];}for(int i=2;i<=n-1;++i){ar[i]+=d[i];}for(int i=1;i<=n-1;++i){c[i].id=i;c[i].v=ar[i];}/*以上是处理路径的长度和边数的代码*/for(int i=0;i<k;++i){scanf("%lld",&op[i].v);op[i].id=i;}sort(op,op+k,cmp2);for(int i=1;i<=n-1;++i){//printf("%d %lld\n",c[i].id,c[i].v );}head=tail=1;memset(f,0,sizeof(f));for(int i=1;i<=n-1;++i){//构造凸包if(head<tail&&c[i].v>=que[tail].v)continue;//这个新点代表的路径长度大于或等于上一个点且边数较多,则可以pass掉,画个图理解下吧while(head<tail-1&&cd(que[tail-1],que[tail],c[i]))tail--;que[++tail]=c[i];}//printf("*%d\n",tail );for(int i=0;i<k;++i){//按照上面红色的式子计算最优值while(head<tail-1&&fx(que[tail-1],que[tail])>=-op[i].v)tail--;f[op[i].id]=que[tail].v+que[tail].id*op[i].v;}for(int i=0;i<k;++i){printf(i!=k-1?"%lld ":"%lld\n",f[i] );}}return 0;
}

数据很水,错的代码都能ac
比如数据:
3 3
1 3
4
0 2 4
应该输出:
3 5 7

#include<bits/stdc++.h>
#define fang(a) (a)*(a)
using namespace std;
typedef long long LL;
const int N = 100000+7;
int n,k,head,tail;
LL ar[N],f[N],d[N];
struct lp{int id;LL v;
}op[N],c[N],que[N];
bool cmp1(lp &a,lp &b){return a.v>b.v;
}
bool cmp2(lp &a,lp &b){return a.v<b.v;
}
/*
i>j
dp[i]=ar[i]+i*k
dp[j]=ar[j]+j*k
dp[i] >= dp[j]
ar[i]-ar[j] >= k*(j-i)
(ar[i]-ar[j])/(i-j) >= -k
维持一个斜率单调上升的队列(斜率是负的)
把k从小到大排序,则-k从大到小排序
从队列末尾开始匹配-k
*/
bool cd(lp a,lp b,lp c){return ((b.id-a.id)*(c.v-a.v)-(b.v-a.v)*(c.id-a.id))<0;
}
double fx(lp x,lp y){return (y.v-x.v*1.0)/(y.id-x.id);
}
int main(){int tim;scanf("%d",&tim);while(tim--){scanf("%d%d",&n,&k);/***********************/for(int i=n-1;i>=1;--i){scanf("%lld",&ar[i]);}for(int i=n-1;i>=2;--i){scanf("%lld",&d[i]);}for(int i=3;i<=n-1;++i){d[i]+=d[i-1];}for(int i=2;i<=n-1;++i){ar[i]+=d[i];}for(int i=1;i<=n-1;++i){c[i].id=i;c[i].v=ar[i];}/*以上是处理路径的长度和边数的代码*/for(int i=0;i<k;++i){scanf("%lld",&op[i].v);op[i].id=i;}sort(c+1,c+n,cmp1);sort(op,op+k,cmp2);for(int i=1;i<=n-1;++i){//printf("%d %lld\n",c[i].id,c[i].v );}head=tail=1;memset(f,0,sizeof(f));for(int i=1;i<=n-1;++i){//构造凸包if(head<tail&&c[i].id<=que[tail].id)continue;//这个新点代表的路径长度小于或等于上一个点且边数较少,则可以pass掉,画个图理解下吧while(head<tail-1&&cd(que[tail-1],que[tail],c[i]))tail--;que[++tail]=c[i];}//printf("*%d\n",tail );for(int i=0;i<k;++i){//按照上面红色的式子计算最优值while(head<tail-1&&fx(que[tail-1],que[tail])>=-op[i].v)tail--;f[op[i].id]=que[tail].v+que[tail].id*op[i].v;}for(int i=0;i<k;++i){printf(i!=k-1?"%lld ":"%lld\n",f[i] );}}return 0;
}

####题面和样例 Description:

Tyrant has a private island on the Pacific Ocean. He has built many luxury villas on the island. He flies here every vacation to enjoy life. He will drive his sports car between his villas. Tyrant has n villas on the island and the last villa which is numbered by n is his favorite. His money is only allowed to build one airpor located at the first villa which is numbered by 1 so every time he come to the island, he will land at the villa numbered 1. What’s more Tyrant build 2*n-3 roads on the island. For every villa except the first island, there is a two-way street connecting this villa and the first villa. For every villa numbered by i (i >= 3) there is a two-way street connecting this villa and the villa numbered by i - 1. Tyrant is not satisfied, he think the road is not long enough. He asks q problems, every problem has a non negative integer d. He want to know if the length of each road is increaced by d what is the shortest distance between the first villa and his favorite villa. Notice that the increment in each problem is not cumulative – in other words it doesn’t take effect on the real road after query.

Input:

The first integer indicate the number of test cases T (T <= 10).

In each test cases the first line contains two integers n and q – the number of villas and the number of queries (3 <= n <= 100000, 1 <= q <= 100000).

The second line contains n - 1 non-negative integers – ai describe a road with length ai connecting villa 1 and i (2 <= i <= n)

The third line contains n - 2 non-negative integers – bi describe a road with length bi connecting villa i - 1 and i (3 <= i <= n)

The next line contains q non-negative integers – di means Tyrant want to know what is the shortest distance between the first villa ans his favorite villa when the length of each road is increaced by di.

All integers are 32-bit signed integer.

Output:

For each test case you should output q integers in one line means the answer for each problem.

样例输入
1
3 3
1 5
2
0 2 4
样例输出
3 7 9

这篇关于2018宁夏acm网络赛-G-Trouble of Tyrant-斜率优化-决策单调性的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/358708

相关文章

SpringBoot3实现Gzip压缩优化的技术指南

《SpringBoot3实现Gzip压缩优化的技术指南》随着Web应用的用户量和数据量增加,网络带宽和页面加载速度逐渐成为瓶颈,为了减少数据传输量,提高用户体验,我们可以使用Gzip压缩HTTP响应,... 目录1、简述2、配置2.1 添加依赖2.2 配置 Gzip 压缩3、服务端应用4、前端应用4.1 N

Linux系统配置NAT网络模式的详细步骤(附图文)

《Linux系统配置NAT网络模式的详细步骤(附图文)》本文详细指导如何在VMware环境下配置NAT网络模式,包括设置主机和虚拟机的IP地址、网关,以及针对Linux和Windows系统的具体步骤,... 目录一、配置NAT网络模式二、设置虚拟机交换机网关2.1 打开虚拟机2.2 管理员授权2.3 设置子

揭秘Python Socket网络编程的7种硬核用法

《揭秘PythonSocket网络编程的7种硬核用法》Socket不仅能做聊天室,还能干一大堆硬核操作,这篇文章就带大家看看Python网络编程的7种超实用玩法,感兴趣的小伙伴可以跟随小编一起... 目录1.端口扫描器:探测开放端口2.简易 HTTP 服务器:10 秒搭个网页3.局域网游戏:多人联机对战4.

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

MyBatis 动态 SQL 优化之标签的实战与技巧(常见用法)

《MyBatis动态SQL优化之标签的实战与技巧(常见用法)》本文通过详细的示例和实际应用场景,介绍了如何有效利用这些标签来优化MyBatis配置,提升开发效率,确保SQL的高效执行和安全性,感... 目录动态SQL详解一、动态SQL的核心概念1.1 什么是动态SQL?1.2 动态SQL的优点1.3 动态S

Python如何使用__slots__实现节省内存和性能优化

《Python如何使用__slots__实现节省内存和性能优化》你有想过,一个小小的__slots__能让你的Python类内存消耗直接减半吗,没错,今天咱们要聊的就是这个让人眼前一亮的技巧,感兴趣的... 目录背景:内存吃得满满的类__slots__:你的内存管理小助手举个大概的例子:看看效果如何?1.

一文详解SpringBoot响应压缩功能的配置与优化

《一文详解SpringBoot响应压缩功能的配置与优化》SpringBoot的响应压缩功能基于智能协商机制,需同时满足很多条件,本文主要为大家详细介绍了SpringBoot响应压缩功能的配置与优化,需... 目录一、核心工作机制1.1 自动协商触发条件1.2 压缩处理流程二、配置方案详解2.1 基础YAML

SpringBoot使用OkHttp完成高效网络请求详解

《SpringBoot使用OkHttp完成高效网络请求详解》OkHttp是一个高效的HTTP客户端,支持同步和异步请求,且具备自动处理cookie、缓存和连接池等高级功能,下面我们来看看SpringB... 目录一、OkHttp 简介二、在 Spring Boot 中集成 OkHttp三、封装 OkHttp

MySQL中慢SQL优化的不同方式介绍

《MySQL中慢SQL优化的不同方式介绍》慢SQL的优化,主要从两个方面考虑,SQL语句本身的优化,以及数据库设计的优化,下面小编就来给大家介绍一下有哪些方式可以优化慢SQL吧... 目录避免不必要的列分页优化索引优化JOIN 的优化排序优化UNION 优化慢 SQL 的优化,主要从两个方面考虑,SQL 语

Linux系统之主机网络配置方式

《Linux系统之主机网络配置方式》:本文主要介绍Linux系统之主机网络配置方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、查看主机的网络参数1、查看主机名2、查看IP地址3、查看网关4、查看DNS二、配置网卡1、修改网卡配置文件2、nmcli工具【通用