https://vjudge.net/problem/CodeForces-8C

题意:

一个平面上放着许多东西,每个东西都有一个坐标,最开始一个人在一个起始坐标,她出发去拿东西,一次要么拿一件东西,要么拿两件东西,拿了之后必须返回起始坐标。

每次花费的时间是两个坐标距离的平方,问拿完所有的东西需要的最少的时间。

思路:

由于数据范围比较小,所以可以考虑用状压dp来写。由于每次拿东西之后都要返回起点,那么其实拿东西的顺序是没有影响的,所以利用题目给定的顺序进行剪枝,即每次进行扩展的时候都考虑在前面的点已经取完了。

然后每次记录的时候,非常巧妙的方法,如果一个点的话,直接记录这个点,如果有两个点的话,那么就用(i+1)*100 + (j+1)来记录,因为点数最多时只有24,输出的时候递归输出就行了,递归输出这里还是比较巧妙的。

代码:(有详细的注释)

 #include <stdio.h>
#include <string.h>
#include <vector>
using namespace std; struct node
{
int x,y;
} a[]; int mp[][];
vector<int> ans; int cal(int i,int j)
{
return (a[i].x - a[j].x) * (a[i].x - a[j].x) + (a[i].y - a[j].y) * (a[i].y - a[j].y);
} const int maxn = ( << )+;
const int gg = 0x3f3f3f3f; int dp[maxn],last[maxn],rec[maxn]; void output(int s)
{
if (~last[s])
output(last[s]);//递归输出 if (rec[s])
{
if (rec[s] > ) ans.push_back(rec[s] / );//第一个扩展的点 ans.push_back(rec[s] % );
ans.push_back();//每次都要返回起点
}
} int main()
{
int n; node tmp; scanf("%d%d",&tmp.x,&tmp.y); scanf("%d",&n); for (int i = ;i < n;i++)
{
scanf("%d%d",&a[i].x,&a[i].y);
} a[n] = tmp; for (int i = ;i <= n;i++)
for (int j = i + ;j <= n;j++)
mp[i][j] = mp[j][i] = cal(i,j); memset(dp,gg,sizeof(dp));
memset(last,-,sizeof(last)); dp[] = ; for (int s = ;s < (<<n);s++)
{
if (dp[s] >= gg) continue; for (int i = ;i < n;i++)
{
if ((( << i) & s) == )//这个点没有被走过
{
int val = dp[s] + mp[n][i] * ; if (dp[s|( << i)] > val)
{
dp[s|( << i)] = val;
last[s|( << i)] = s;//记录前驱,下同
rec[s|( << i)] = i+;
} for (int j = i + ;j < n;j++)//从i+1开始枚举保证了不会有重复情况
{
if (((<<j)&s) == )
{
int tmp = dp[s] + mp[n][i] + mp[i][j] + mp[j][n]; if (dp[s|(<<i)|(<<j)] > tmp)
{
dp[s|(<<i)|(<<j)] = tmp;
last[s|(<<i)|(<<j)] = s;
rec[s|(<<i)|(<<j)] = (i+) * + (j+);//巧妙的记录
}
}
} break;//强行顺序剪枝
}
} //printf("%d\n",dp[s]);
} ans.clear(); output((<<n)-); printf("%d\n",dp[(<<n)-]); printf("0 "); for (int i = ;i < ans.size();i++)
printf("%d%s",ans[i],i == ans.size() - ? "\n" :" "); return ;
}

codeforces 8c Looking for Order的更多相关文章

  1. codeforces 8C. Looking for Order 状压dp

    题目链接 给n个物品的坐标, 和一个包裹的位置, 包裹不能移动. 每次最多可以拿两个物品, 然后将它们放到包里, 求将所有物品放到包里所需走的最小路程. 直接状压dp就好了. #include < ...

  2. 【题解】codeforces 8c Looking for Order 状压dp

    题目描述 Lena喜欢秩序井然的生活.一天,她要去上大学了.突然,她发现整个房间乱糟糟的--她的手提包里的物品都散落在了地上.她想把所有的物品都放回她的手提包.但是,这里有一点问题:她一次最多只能拿两 ...

  3. Codeforces 993E Nikita and Order Statistics [FFT]

    洛谷 Codeforces 思路 一开始想偏想到了DP,后来发现我SB了-- 考虑每个\(a_i<x\)的\(i\),记录它前一个和后一个到它的距离为\(L_i,R_i\),那么就有 \[ an ...

  4. codeforces 637B B. Chat Order(map,水题)

    题目链接: B. Chat Order time limit per test 3 seconds memory limit per test 256 megabytes input standard ...

  5. codeforces 8C(非原创)

    C. Looking for Order time limit per test 4 seconds memory limit per test 512 megabytes input standar ...

  6. [Codeforces 993E]Nikita and Order Statistics

    Description 题库链接 给你一个长度为 \(n\) 的序列 \(A\) ,和一个数 \(x\) ,对于每个 \(i= 0\sim n\) ,求有多少个非空子区间满足恰好有 \(i\) 个数 ...

  7. Codeforces 8C 状压DP

    题意:有个人想收拾行李,而n个物品散落在房间的各个角落里(n < 24).现在给你旅行箱的坐标(人初始在旅行箱处),以及n个物品的坐标,你一次只能拿最多两个物品,并且拿了物品就必须放回旅行箱,不 ...

  8. [CF] 8C Looking for Order

    状压模板题 CF难度2000? 我得好好了解一下CF的难度机制了 反正CF的难度比洛谷真实就好了 Code #include<algorithm> #include<iostream ...

  9. CF dp 题(1500-2000难度)

    前言 从后往前刷 update 新增 \(\text{\color{red}{Mark}}\) 标记功能,有一定难度的题标记为 \(\text{\color{red}{红}}\) 色. 题单 (刷过的 ...

随机推荐

  1. sed修炼系列(四):sed中的疑难杂症

    本文目录:1 sed中使用变量和变量替换的问题2 反向引用失效问题3 "-i"选项的文件保存问题4 贪婪匹配问题5 sed命令"a"和"N" ...

  2. HTTP协议知多少-关于http1.x、http2、SPDY的相关知识

    作为网站开发的基础协议,我们知道浏览器上都有输出http这四个字母,这意味着什么呢? 这就是最基础的HTTP协议. 逐浪君今天为各位大人准备了一些HTTP技术的知识,来和大家分享. 以下图为例: 这一 ...

  3. IOS UIScrollView常用代理方法

    iOS UIScrollView代理方法有很多,从头文件中找出来学习一下 //只要滚动了就会触发 - (void)scrollViewDidScroll:(UIScrollView *)scrollV ...

  4. 短视频 SDK 功能点技术实现方式详解

    第三方短视频解决方案作为快速切入短视频行业的首选方式,选择一款功能齐全.性能优异的短视频解决方案十分重要. 今天我们来谈谈短视频 SDK 6大重要功能点及其技术实现方式. 短视频拍摄 断点续拍 指在拍 ...

  5. 换个角度审视NAT技术

    NAT (Network address translation,网络地址转换 )是局域网连接到互联网的一个对接工作. 首先要知道NAT是一个技术或者说软件而不是协议 后面你会知道NAT 是偏应用层但 ...

  6. Window下SVN服务器搭建以及客户端使用

    一.下载 上一篇博客是关于Jenkins的内容,在Jenkins自动化编译时可能会自动获取版本更新进行build,那就需要用到版本更新的工具.这里使用VisualSVN Server来作为搭建svn的 ...

  7. 201521123061 《Java程序设计》第十三周学习总结

    201521123061 <Java程序设计>第十三周学习总结 1. 本周学习总结 2. 书面作业 1. 网络基础 1.1 比较ping www.baidu.com与ping cec.jm ...

  8. 201521123087 《java程序设计》 第七周学习总结

    1. 本周学习总结 2. 书面作业 ArrayList代码分析1.1 解释ArrayList的contains源代码                                           ...

  9. 201521123049 《JAVA程序设计》 第7周学习总结

    1. 本周学习总结 2. 书面作业 1.ArrayList代码分析 1.1 解释ArrayList的contains源代码 //contains()方法 public boolean contains ...

  10. How To:禁用ubuntu全局菜单(global menu)的方法

    刚从windows转过来的新手可用会觉得ubuntu unity下的全局菜单(global menu)用起来很不方便.下边是介绍去除全局菜单的方法 1.打开终端(可以去dash主页里面搜,也可以直接按 ...