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

相关文章

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

MySQL不使用子查询的原因及优化案例

《MySQL不使用子查询的原因及优化案例》对于mysql,不推荐使用子查询,效率太差,执行子查询时,MYSQL需要创建临时表,查询完毕后再删除这些临时表,所以,子查询的速度会受到一定的影响,本文给大家... 目录不推荐使用子查询和JOIN的原因解决方案优化案例案例1:查询所有有库存的商品信息案例2:使用EX

SSID究竟是什么? WiFi网络名称及工作方式解析

《SSID究竟是什么?WiFi网络名称及工作方式解析》SID可以看作是无线网络的名称,类似于有线网络中的网络名称或者路由器的名称,在无线网络中,设备通过SSID来识别和连接到特定的无线网络... 当提到 Wi-Fi 网络时,就避不开「SSID」这个术语。简单来说,SSID 就是 Wi-Fi 网络的名称。比如

MySQL中my.ini文件的基础配置和优化配置方式

《MySQL中my.ini文件的基础配置和优化配置方式》文章讨论了数据库异步同步的优化思路,包括三个主要方面:幂等性、时序和延迟,作者还分享了MySQL配置文件的优化经验,并鼓励读者提供支持... 目录mysql my.ini文件的配置和优化配置优化思路MySQL配置文件优化总结MySQL my.ini文件

Java实现任务管理器性能网络监控数据的方法详解

《Java实现任务管理器性能网络监控数据的方法详解》在现代操作系统中,任务管理器是一个非常重要的工具,用于监控和管理计算机的运行状态,包括CPU使用率、内存占用等,对于开发者和系统管理员来说,了解这些... 目录引言一、背景知识二、准备工作1. Maven依赖2. Gradle依赖三、代码实现四、代码详解五

正则表达式高级应用与性能优化记录

《正则表达式高级应用与性能优化记录》本文介绍了正则表达式的高级应用和性能优化技巧,包括文本拆分、合并、XML/HTML解析、数据分析、以及性能优化方法,通过这些技巧,可以更高效地利用正则表达式进行复杂... 目录第6章:正则表达式的高级应用6.1 模式匹配与文本处理6.1.1 文本拆分6.1.2 文本合并6

Vue3 的 shallowRef 和 shallowReactive:优化性能

大家对 Vue3 的 ref 和 reactive 都很熟悉,那么对 shallowRef 和 shallowReactive 是否了解呢? 在编程和数据结构中,“shallow”(浅层)通常指对数据结构的最外层进行操作,而不递归地处理其内部或嵌套的数据。这种处理方式关注的是数据结构的第一层属性或元素,而忽略更深层次的嵌套内容。 1. 浅层与深层的对比 1.1 浅层(Shallow) 定义

HDFS—存储优化(纠删码)

纠删码原理 HDFS 默认情况下,一个文件有3个副本,这样提高了数据的可靠性,但也带来了2倍的冗余开销。 Hadoop3.x 引入了纠删码,采用计算的方式,可以节省约50%左右的存储空间。 此种方式节约了空间,但是会增加 cpu 的计算。 纠删码策略是给具体一个路径设置。所有往此路径下存储的文件,都会执行此策略。 默认只开启对 RS-6-3-1024k