参考博客: 春季测试 2023] 圣诞树 题解 - 洛谷专栏 (luogu.com.cn)

题意:给定二维平面上一个凸多边形的 \(n\) 个顶点, 求一种方案,使得从最高点开始,不重复地经过所有点后距离的最小值。只要求输出这种方案。

经典的 TSP 问题。

\(n<=18\) 可以直接状压dp.

\(n<=1000\):

就是要大胆发现一些结论。

根据三角形法则,

对比图1和图2,发现如果电线交叉肯定不优。

所以对于逆时针给定的若干个点,当处于 \(\) 时,最优决策一定只能是下一步到 \(−1\) 或 \(i+1\),否则将会存在一个点被隔离,导致最终去往该点时一定形成交叉路径。并且 \(i\) 之前的决策并不影响当前这一步的决策。

此时无后效性和子问题的雏形已经出现了,考虑动态规划。

  • 定义:\(f[i][j][0/1]\), 区间\([i,j]\), 停在左边界/右边界, 最短距离
  • 转移:
\[f[i][j][0] = \min(dis(i,i+1) + f[i+1][j][0] , dis(i,j) + f[i+1][j][1])
\]
\[f[i][j][1] = \min(dis(j-1,j) + f[i][j-1][1] , dis(i,j) + f[i][j-1][0])
\]
  • 初始化:\(f[s][s][0/1] = 0\), 其他的初始化为 \(inf\)
  • 扩展顺序:先从小到大枚举长度,再从左到右依次枚举所有区间
  • 答案:由\(f[1][n][1]\) 往回倒推(最后一步仍然要遵循三角形法则,所以只考虑 \(f[1][n][1].\))
#include<bits/stdc++.h>
#define F(i,l,r) for(int i(l);i<=(r);++i)
#define G(i,r,l) for(int i(r);i>=(l);--i)
#define int ll
using namespace std;
using ll = long long;
using ull = unsigned long long;
const int N=1e3+5;
const double inf = 1e18;
struct node{
double x,y;
int id;
}a[N],tmp[N];
double f[N][N][2];
int pre[N][N][2];
inline double dis(int i,int j){
return sqrt((a[i].x-a[j].x) * (a[i].x-a[j].x) + (a[i].y-a[j].y) * (a[i].y-a[j].y));
}
int n,k=1;
inline void dfs(int i,int j,int op){
if(i==j) return cout<<a[i].id,void();
if(!op) cout<<a[i].id<<" ",dfs(i+1,j,pre[i][j][0]);
else cout<<a[j].id<<" ", dfs(i,j-1,pre[i][j][1]);
return ;
}
signed main(){
// freopen("tree.in","r",stdin);
// freopen("tree.out","w",stdout);
ios::sync_with_stdio(0); cin.tie(0); cout.tie(0);
cin>>n; F(i,1,n) cin>>a[i].x>>a[i].y,a[i].id=i,tmp[i]=a[i];
F(i,2,n) if(a[i].y>a[k].y) k=i;
F(i,1,k) a[i+n-k]=tmp[i];
F(i,k+1,n) a[i-k]=tmp[i];
F(len,2,n-1){
for(int i=1,j=len;j<=n;++i,++j){
f[i][j][0] = f[i][j][1] = inf;
if(dis(i,i+1) + f[i+1][j][0] < dis(i,j) + f[i+1][j][1])
f[i][j][0] = dis(i,i+1) + f[i+1][j][0], pre[i][j][0] = 0;
else
f[i][j][0] = dis(i,j) + f[i+1][j][1], pre[i][j][0] = 1;
if(dis(j-1,j) + f[i][j-1][1] < dis(i,j) + f[i][j-1][0]) f[i][j][1] = dis(j-1,j) + f[i][j-1][1], pre[i][j][1] = 1;
else f[i][j][1] = dis(i,j) + f[i][j-1][0], pre[i][j][1] = 0;
}
}
cout<<a[n].id<<" ";
if(dis(n-1,n) + f[1][n-1][1] > dis(1,n) + f[1][n-1][0]) dfs(1,n-1,0);
else dfs(1,n-1,1);
return fflush(0),0;
}

总结一下,感觉拿到这道题还是没有冷静去分析条件,比如起点固定能不能利用?逆时针给定的这张图的顺序可能长什么样?不是从最高点开始给的图,我怎么通过一些简单的设计去调整?有哪些情况是显然不优的?敢不敢大胆排除?

P9119 [春季测试 2023] 圣诞树的更多相关文章

  1. 2023 年 CCF 春季测试赛模拟赛 - 2 题解

    T1 约数和 标准解法 \(n = a_1^{b_1} \times a_2^{b_2} \dots a_k^{b_k}\) 那么根据算术基本定理的推广,约数个数和约数和都是可以快速计算得到 约数和 ...

  2. 2023 年 CCF 春季测试赛模拟赛 - 2

    T1 分治,\(a^b + \dots + 1 = (a^{\lfloor\frac{b}{2}\rfloor} + \dots + 1) \times (a^{\lfloor\frac{b}{2}\ ...

  3. 2023 年 CCF 春季测试赛模拟赛 - 1

    T1 个人思路: 询问:求 \(1\) 到 \(t_i\) 路径上离 \(1\) 最远的 \(p\),使得 \(dis_{1,p} \times 2 \le d_i\).即 \(dis_{1,t} \ ...

  4. 【kAriOJ】离散数学春季学期编程测试 1

    A.凯撒密码 题意: 给你k1,k2,和一串明文,一串密文. 明文用k1加密,密文用k2解密. 对于明文要把字母转换成大写字母,非字母全部删除. 额:要考虑到取模可能会变成负数,所以要加一下26再取模 ...

  5. 躬身入局,干货分享,2023年春招后端技术岗(Python)面试实战教程,Offer今始为君发

    早春二月,研发倍忙,杂花生树,群鸥竟飞.为什么?因为春季招聘,无论是应届生,还是职场老鸟,都在摩拳擦掌,秣马厉兵,准备在面试场上一较身手,既分高下,也决Offer,本次我们打响春招第一炮,躬身入局,让 ...

  6. 百度前端技术学院(IFE)2016春季学期总结

    今天(5月16日)作为第八个提交者提交了任务五十:RIA微型问卷管理平台 这样一个综合性的大任务,宣告我的IFE春季学期课程学习顺利完成.其实任务五十并不复杂,现在再让我来做,可能一周不到就写出来了, ...

  7. 云server 性能测试web压力测试

    前言:如今,云server主流.它已成为许多中小型企业的首选server,但是云server它是一个虚拟机.所以性能是一个大问题,从这里开始介绍云server性能测试,云webserver压力测试. ...

  8. HDU 2023 求平均成绩

    Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u Submit Status Practice HDU ...

  9. 个人待办事项工具的设计和搭建(IFE前端2015春季 任务3)

    这是我几个月之前的项目作品,花了相当的时间去完善.博客人气不高,但拿代码的人不少,所以一直处于保密状态.没有公开代码.但如果对你有帮助,并能提出指导意见的,我将十分感谢. IFE前端2015春季 任务 ...

  10. Win10《芒果TV》春季商店版更新v3.3.0:全新视觉蜕变&支持快男直播

    在微软发布Win10创意者更新正式版前夕,Win10版<芒果TV>迅速更新至v3.3.0,主要是全新升级视觉交互,新增大咖快男个人直播,全面优化底层架构,启动大提速. Win10版< ...

随机推荐

  1. Python-目标检测-将xml文件转换成.txt文件

    代码说明:labels文件夹是工程下的一个文件夹,里面存放的是一些xml文件. 然后我们将这些xml文件中的内容取出来,放在路径path1的文件名下.这样也就完成了xml文件到txt文件的转化. 该代 ...

  2. 「TCP/UDP」一个端口号可以同时被两个进程绑定吗?

    一.1个端口号可以同时被两个进程绑定吗? 根据端口号的绑定我们分以下几种情况来讨论: 2个进程分别建立TCP server,使用同一个端口号8888 2个进程分别建立UDP server,使用同一个端 ...

  3. 15. 从0开始学ARM-位置无关码

    @ 目录 十九.位置无关码 一.为什么需要位置无关码? 1. exynos 4412启动流程 二.怎么实现位置无关码? 1. 什么是<编译地址>?什么是<运行地址>? 2. 举 ...

  4. 优秀架构设计与web文档

    架构风格与基于网络的软件架构设计 https://docs.huihoo.com/rest/REST_cn.pdf rfc https://tools.ietf.org/html/rfc2616

  5. RabbitMQ相关总结

    //connection //channel //broke //exchange:fanout. dirct. topic(*:一个单词.#:多个单词). header //queue //bind ...

  6. mariadb5.5.56二进制离线安装

    在生产环境中一般使用发布好的二进制版本,简单概括一下安装过程: 1. 下载 地址为:https://downloads.mariadb.org/mariadb/5.5.56/ 这里选择最新版本的5.5 ...

  7. 在 Windows 中启用 Administrator 帐户

    打开管理员终端. 启用: net user administrator /active:yes 关闭: net user administrator /active:no

  8. 【Jmeter】之进行单接口批量压力测试

    目录: 一.安装Jmeter 二.接口压力测试 p.p1 { margin: 0; font: 14px ".PingFang SC"; color: rgba(17, 31, 4 ...

  9. 手把手教你安装Jupyter Notebook(保姆级教程)

    来源于:https://blog.csdn.net/weixin_43855159/article/details/137738714 1. 什么是Jupyter Notebook Jupyter N ...

  10. C++17: 用折叠表达式实现一个IsAllTrue函数

    前言 让我们实现一个 IsAllTrue 函数,支持变长参数,可传入多个表达式,必须全部计算为true,该函数才返回true. 本文记录了逐步实现与优化该函数的思维链,用到了以下现代C++新特性知识, ...