首页 > 其他分享 >传递游戏【题解】

传递游戏【题解】

时间:2023-01-12 16:23:54浏览次数:43  
标签:状态 游戏 题解 状压 传递 枚举 物品 dp

Description

毛大神最近在玩一个传递游戏,即有 \(N\) 个人在做传递物品的游戏,这 N 个人的编号为 \(1,2,3,...,N-1,N\)。

游戏规则是这样的:开始时物品可以在任意一人手上,他可把物品传递给其他人中的任意一位;下一个人可以传递给未接过物品的任意一人。

即物品只能经过同一个人一次,而且每次传递过程都有一个代价;不同的人传给不同的人的代价值之间没有联系;

毛大神想知道当物品经过所有 \(N\) 个人后,整个过程的最小的代价总和是多少。

Format

Input

第一行为 \(N\),表示共有 \(N\) 个人;

以下为 N × N 的矩阵,第\(i\)行第 j 列的数为 \(A[i][j]\),表示物品从编号为 i 的人传递到编号为 j 的人所花费的代价,其中 \(A[i][i]=-1\)(因为物品不能自己传给自己)。

Output

输出共一行一个数,为最小的代价总和。

Samples

输入数据 1
2
-1 9794
2724 -1
输出数据 1
2724
Limitation
对于 50% 的数据,\(1 ≤ N ≤ 11\);

对于 100% 的数据,\(1 ≤ N ≤ 16;1 ≤ A[i][j] ≤ 10000;A[i][i]=-1\)。

浅浅的谈一下

  • 第一反应肯定是搜索,然后回溯,时间复杂度乍一看可以,我们算算

  • ……其实我也不会算(一个蒟蒻),为什么不行呢,因为有人试过了,\(45\)分

  • 怎么办?

  • 一看这题目的问法, 就有一种\(dp\)的思路,配合这数据范围,配合这题目,状压\(dp\)应运而生!

  • 什么是状压\(dp\)(其实我写过了,就copy过来了)

  • “状压DP 又叫集合动态规划。是以结合信息为状态的特殊的动态规划的问题。主要有传统集合动态规划和基于连通性状态压缩的动态规划两种。” ————百度

  • 是不是感觉很高大尚?(我也觉得

  • 其实很简单(不是我说的

  • 状压dp的样子

  • 我们回到题目:我们把每个人看成每一个点,被传递过标记成一,他就变成了这样一个样子,很想二进制枚举是吧
    长这样
  • 怎么转移?

  • (\(<<\)):左移

    • 在十进制上是乘法
    • 在二进制上是把整体往左挪一位,例如:\(100 << 1 = 1000\)
    • (\(>>\)): 右移,和左移原理一样,把整体往右移一位
    • 对于每个状态,我们枚举这个点有没有选到:如\(100010\)这个状态没有选第三个点
  • 这就要用到左移与运算

  • 如果我想要表示一个\(100\),表示第三个状态已被选怎么办?

  • 很容易发现\(100=1<<(3-1)\)

  • 于是我们可以总结:\(想要表示第i个点被选=1<<(i-1)\)

  • 于是我们就可以用\(i\&1<<(j-1)\)(\(i\)表示当前状态,\(j\)表示当前枚举到的点)来表示\(i\)状态有没有选点\(j\)

  • 考虑转移?

  • 枚举每个状态,再枚举这个状态选了那些点,再由这些点进行转移

  • 什么意思?

  • 例如状态\(10001\),这个状态包含\(1,5\)这\(2\)个点

  • 如果点\(2\)可以接到点\(1\)后面

  • 那么\(f[11001]=\max(\min)(f[10001]+v,f[11001])\)

  • 然后输出\(f[(1<<n)-1]\)全部点被选即可

  • 这道题咋写

  • 假如枚举到一个状态\(T\),枚举这个状态有多少个存在的点,再由这个点去转移到另外没有被选过的点

  • 等等,怎么判断\(T\)中,球传到哪里了?

  • 于是,我们就要开二维数组$dp_{iT}到了第i个人,状态为T $

  • 于是就有我们的满分代码了(注意\(<<\)的优先级!)

\(Code\)

#include<bits/stdc++.h>
//值得写解题报告! 
using namespace std;
int dp[20][1<<20],n,g[20][20];
//dp[i][T]到了第i个人,状态为T 
int main(){
	cin>>n;
	for(int i=1;i<=n;i++) 
		for(int j=1;j<=n;j++)
			cin>>g[i][j];
	memset(dp,0x3f,sizeof(dp));
	for(int i=1;i<=n;i++)dp[i][1<<(i-1)]=0;//初始 
	for(int t=1;t<1<<n;t++)//枚举状态
		 for(int i=1;i<=n;i++)
		 	if(t&(1<<i-1))//在这里面 
		 		for(int j=1;j<=n;j++)
		 			if(i!=j&&!(t&(1<<j-1)))//不在这里面
					 	dp[j][t|(1<<j-1)]=min(dp[j][t|(1<<j-1)],dp[i][t]+g[i][j]);
	int ans=1e9;
	for(int i=1;i<=n;i++)
		ans=min(ans,dp[i][(1<<n)-1]);
	cout<<ans<<endl;
	return 0;//注意"<<"优先级比"-"小 
}

题外话:普及组的题竟然会考状压,没想到

标签:状态,游戏,题解,状压,传递,枚举,物品,dp
From: https://www.cnblogs.com/Phrvth/p/17046985.html

相关文章

  • 表达式的值【题解】
    [NOIP2011普及组]表达式的值题目描述对于1位二进制变量定义两种运算:运算的优先级是:先计算括号内的,再计算括号外的。“×”运算优先于“⊕”运算,即计算表达式......
  • [NOIP2017 普及组]跳房子 【题解】
    题目背景NOIP2017普及组T4题目描述跳房子,也叫跳飞机,是一种世界性的儿童游戏,也是中国民间传统的体育游戏之一。跳房子的游戏规则如下:在地面上确定一个起点,然后在起点......
  • P1005 [NOIP2007 提高组] 矩阵取数游戏
    题目传送门前言今天依旧是不写高精的一天呢!(是的,这位作者又只拿了开\(LL\)的\(\color{yellow}{60}\)分)思路描述看到数据\(n,m\le80(30)\)就知道数组可以任性开,......
  • 洛谷 P8077 [WC2022] 序列变换 题解
    题目链接。WC2023之前补一下WC2022的题,参考了官方题解。首先,把括号序列转化为二叉树,\(\texttt{(A)B}\)转为一个点的左子树是\(A\),右子树是\(B\)。相当于括号序列先......
  • 1.8模拟赛题解
    T1考虑每次反弹后,球的运动轨迹都会偏移\(2\beta\),总偏移量即为\(2k\beta\),而最后需要回到原点,因此\(360|2k\beta\),简单求\(\gcd\)即可。T2设\(ans_k\)表示出现过......
  • 1.11模拟赛题解
    T1对于方阵\(A\),考虑其反方阵\(A'\)。容易发现\(A\)与\(A'\)的权值和相同,而其中必有一个与\(B\)的差不超过\(\lfloor\frac{nm}{2}\rfloor\),因此判断一下哪个满足......
  • 1.9模拟赛题解
    T1从左到右扫描,首先如果\(a_i<b_i\)那么一定无解,否则不断在其右边找最近的\(j\)使得\(a_j\in[b_i,a_i]\),把\(a_i\)和\(a_j\)交换。感性理解这是对的。T2先证操......
  • 1.12模拟赛题解
    T1容易知道答案为原图的最大子二分图大小。枚举每个点在二分图的左边还是右边,计算出答案。时间复杂度\(O(2^n\timesm)\)。T2考虑递推构造方案。假设现在已经有了一组......
  • LeetCode刷题(73)~Nim 游戏
    题目描述你和你的朋友,两个人一起玩Nim游戏:桌子上有一堆石头,每次你们轮流拿掉1-3块石头。拿掉最后一块石头的人就是获胜者。你作为先手。你们是聪明人,每一步都是最优......
  • POI Excel格式报表生成 同步下载问题解决
    前言解决POI导出功能,过时方法和新增样式放在最下面或者参考下文POI样式调节0.maven(新版本)<poi.version>4.1.2</poi.version> <dependency> <groupId>org.ap......