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

相关文章

Debian 13升级后网络转发等功能异常怎么办? 并非错误而是管理机制变更

《Debian13升级后网络转发等功能异常怎么办?并非错误而是管理机制变更》很多朋友反馈,更新到Debian13后网络转发等功能异常,这并非BUG而是Debian13Trixie调整... 日前 Debian 13 Trixie 发布后已经有众多网友升级到新版本,只不过升级后发现某些功能存在异常,例如网络转

从原理到实战解析Java Stream 的并行流性能优化

《从原理到实战解析JavaStream的并行流性能优化》本文给大家介绍JavaStream的并行流性能优化:从原理到实战的全攻略,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的... 目录一、并行流的核心原理与适用场景二、性能优化的核心策略1. 合理设置并行度:打破默认阈值2. 避免装箱

Python实战之SEO优化自动化工具开发指南

《Python实战之SEO优化自动化工具开发指南》在数字化营销时代,搜索引擎优化(SEO)已成为网站获取流量的重要手段,本文将带您使用Python开发一套完整的SEO自动化工具,需要的可以了解下... 目录前言项目概述技术栈选择核心模块实现1. 关键词研究模块2. 网站技术seo检测模块3. 内容优化分析模

Java实现复杂查询优化的7个技巧小结

《Java实现复杂查询优化的7个技巧小结》在Java项目中,复杂查询是开发者面临的“硬骨头”,本文将通过7个实战技巧,结合代码示例和性能对比,手把手教你如何让复杂查询变得优雅,大家可以根据需求进行选择... 目录一、复杂查询的痛点:为何你的代码“又臭又长”1.1冗余变量与中间状态1.2重复查询与性能陷阱1.

Python内存优化的实战技巧分享

《Python内存优化的实战技巧分享》Python作为一门解释型语言,虽然在开发效率上有着显著优势,但在执行效率方面往往被诟病,然而,通过合理的内存优化策略,我们可以让Python程序的运行速度提升3... 目录前言python内存管理机制引用计数机制垃圾回收机制内存泄漏的常见原因1. 循环引用2. 全局变

Python多线程应用中的卡死问题优化方案指南

《Python多线程应用中的卡死问题优化方案指南》在利用Python语言开发某查询软件时,遇到了点击搜索按钮后软件卡死的问题,本文将简单分析一下出现的原因以及对应的优化方案,希望对大家有所帮助... 目录问题描述优化方案1. 网络请求优化2. 多线程架构优化3. 全局异常处理4. 配置管理优化优化效果1.

MySQL中优化CPU使用的详细指南

《MySQL中优化CPU使用的详细指南》优化MySQL的CPU使用可以显著提高数据库的性能和响应时间,本文为大家整理了一些优化CPU使用的方法,大家可以根据需要进行选择... 目录一、优化查询和索引1.1 优化查询语句1.2 创建和优化索引1.3 避免全表扫描二、调整mysql配置参数2.1 调整线程数2.

Python开发简易网络服务器的示例详解(新手入门)

《Python开发简易网络服务器的示例详解(新手入门)》网络服务器是互联网基础设施的核心组件,它本质上是一个持续运行的程序,负责监听特定端口,本文将使用Python开发一个简单的网络服务器,感兴趣的小... 目录网络服务器基础概念python内置服务器模块1. HTTP服务器模块2. Socket服务器模块

Go语言网络故障诊断与调试技巧

《Go语言网络故障诊断与调试技巧》在分布式系统和微服务架构的浪潮中,网络编程成为系统性能和可靠性的核心支柱,从高并发的API服务到实时通信应用,网络的稳定性直接影响用户体验,本文面向熟悉Go基本语法和... 目录1. 引言2. Go 语言网络编程的优势与特色2.1 简洁高效的标准库2.2 强大的并发模型2.

深入解析Java NIO在高并发场景下的性能优化实践指南

《深入解析JavaNIO在高并发场景下的性能优化实践指南》随着互联网业务不断演进,对高并发、低延时网络服务的需求日益增长,本文将深入解析JavaNIO在高并发场景下的性能优化方法,希望对大家有所帮助... 目录简介一、技术背景与应用场景二、核心原理深入分析2.1 Selector多路复用2.2 Buffer