vijos - P1456最小总代价 (状态压缩DP + 记忆化搜索)

P1456最小总代价

Accepted

标签:[显示标签]

描述

n个人在做传递物品的游戏,编号为1-n。

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

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

求当物品经过所有n个人后,整个过程的总代价是多少。

格式

输入格式

第一行为n,表示共有n个人(16>=n>=2);

以下为n*n的矩阵,第i+1行、第j列表示物品从编号为i的人传递到编号为j的人所花费的代价,特别的有第i+1行、第i列为-1(因为物品不能自己传给自己),其他数据均为正整数(<=10000)。

(对于50%的数据,n<=11)。

输出格式

一个数,为最小的代价总和。

样例1

样例输入1[复制]

2
-1 9794
2724 –1

样例输出1[复制]

2724

限制

所有数据时限为1s

来源

jszx

此题可以直接用记忆化搜索来更加简便,这里暂时提到的是状态压缩DP;

dp[0][i][j][k]为选择了几个人

dp[1][i][j][k]为最小代价

所以:

dp[1][i][j][k] = min(dp[1][i][j][k], dp[1][i - 1][f][j] + V[j][k]){k 不属于S{已经选了的人}}。

//package ds;

import java.util.*;
import java.math.*;
import java.io.*;

public class Main {
	static Scanner cin = null;
	static PrintStream cout = null;
	static final int MAXN = 20;
	static int INF = 0x3f3f3f3f;
	static int[][] V = new int[MAXN][MAXN];
	static int[][][][] dp = new int[2][MAXN][MAXN][MAXN];

	static int Bit_Count(int val) {//有几个人已经传递过
		int ret = 0;
		while (val > 0) {
			val &= (val - 1);
			ret++;
		}
		return ret;
	}

	public static void main(String[] agrs) throws IOException {
		// System.setIn(new FileInputStream(new File("D:" + File.separator +
		// "imput.txt")));
		cin = new Scanner(System.in);
		cout = new PrintStream(System.out);

		while (cin.hasNext()) {
			int n = cin.nextInt();
			for (int i = 1; i <= n; i++) {
				for (int j = 1; j <= n; j++) {
					V[i][j] = cin.nextInt();
				}
			}
			for (int i = 0; i < 2; i++) {
				for (int f = 1; f <= n; f++) {
					for (int j = 1; j <= n; j++) {
						for (int k = 1; k <= n; k++) {
							if (i == 1)
								dp[i][f][j][k] = 0;
							else
								dp[i][f][j][k] = INF;
							if (i == 0 && f == 0)
								dp[i][f][j][k] = 0;
						}
					}
				}
			}
			int Min = INF;
			for (int i = 1; i < n; i++) {
				for (int j = 1; j <= n; j++) {
					for (int k = 1; k <= n; k++) {
						if (j == k)
							continue;
						for (int f = 1; f <= n; f++) {
							if (f == j)
								continue;
							if ((dp[1][i - 1][f][j] & (1 << k)) > 0)
								continue;
							if (dp[0][i][j][k] >= dp[0][i - 1][f][j] + V[j][k]) {
								dp[0][i][j][k] = dp[0][i - 1][f][j] + V[j][k];
								dp[1][i][j][k] = dp[1][i - 1][f][j] | (1 << k);
								if (i == n - 1
										&& Bit_Count(dp[1][i][j][k]) == n - 1) {
									Min = Math.min(Min, dp[0][i][j][k]);
								}
							}
						}
					}
				}
			}
			if (Min == INF)
				cout.println("0");
			else
				cout.println(Min);
		}
		cout.flush();
	}
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-12 20:46:50

vijos - P1456最小总代价 (状态压缩DP + 记忆化搜索)的相关文章

hdu1143 状态压缩dp 记忆化搜索写法

http://poj.org/problem?id=1143 Description Christine and Matt are playing an exciting game they just invented: the Number Game. The rules of this game are as follows. The players take turns choosing integers greater than 1. First, Christine chooses a

hdu 4628 Pieces(状态压缩+记忆化搜索)

Pieces Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others) Total Submission(s): 1811    Accepted Submission(s): 932 Problem Description You heart broke into pieces.My string broke into pieces.But you will recover one

poj 1191 棋盘切割 (压缩dp+记忆化搜索)

一,题意: 中文题 二.分析: 主要利用压缩dp与记忆化搜索思想 三,代码: #include <iostream> #include <stdio.h> #include <string.h> #include <stdlib.h> #include <math.h> using namespace std; const int Big=20000000; int Mat[10][10]; int N; int sum[10][10]; int

poj 1191 棋盘分割 (压缩dp+记忆化搜索)

一,题意: 中文题 二,分析: 主要利用压缩dp与记忆化搜索思想 三,代码: #include <iostream> #include <stdio.h> #include <string.h> #include <stdlib.h> #include <math.h> using namespace std; const int Big=20000000; int Mat[10][10]; int N; int sum[10][10]; int

UVa 10817 (状压DP + 记忆化搜索) Headmaster&#39;s Headache

题意: 一共有s(s ≤ 8)门课程,有m个在职教师,n个求职教师. 每个教师有各自的工资要求,还有他能教授的课程,可以是一门或者多门. 要求在职教师不能辞退,问如何录用应聘者,才能使得每门课只少有两个老师教而且使得总工资最少. 分析: 因为s很小,所以可以用状态压缩. dp(i, s1, s2)表示考虑了前i个人,有一个人教的课程的集合为s1,至少有两个人教的集合为s2. 在递归的过程中,还有个参数s0,表示还没有人教的科目的集合. 其中m0, m1, s0, s1, s2的计算用到位运算,还

poj 1651 dp 记忆化搜索

题意: 给出n个整数a1,a2,-,an,要求从中取出中间的n-2个数(两端的数不能取),取出每个数的代价为它两边的数和它的乘积,问取出这n-2个数的最小代价为多少? 限制: 3 <= n <= 100; 1 <= ai <= 100 思路: dp 记忆化搜索 对于每个过程其实就是,枚举最后取的数a[i],然后把区间[l,r]分割成[l,i]和[i,r]两部分. dp[l][r]=min(gao(l,i)+a[left]*a[i]*a[right]+gao(i,r))

[hihocoder 1033]交错和 数位dp/记忆化搜索

#1033 : 交错和 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 给定一个数 x,设它十进制展从高位到低位上的数位依次是 a0,?a1,?...,?an?-?1,定义交错和函数: f(x)?=?a0?-?a1?+?a2?-?...?+?(?-?1)n?-?1an?-?1 例如: f(3214567)?=?3?-?2?+?1?-?4?+?5?-?6?+?7?=?4 给定 输入 输入数据仅一行包含三个整数,l,?r,?k(0?≤?l?≤?r?≤?1018,?|k|

UVA - 10817 Headmaster&#39;s Headache (状压dp+记忆化搜索)

题意:有M个已聘教师,N个候选老师,S个科目,已知每个老师的雇佣费和可教科目,已聘老师必须雇佣,要求每个科目至少两个老师教的情况下,最少的雇佣费用. 分析: 1.为让雇佣费尽可能少,雇佣的老师应教他所能教的所有科目. 2.已聘老师必须选,候选老师可选可不选. 3.dfs(cur, subject1, subject2)---求出在当前已选cur个老师,有一个老师教的科目状态为 subject1,有两个及以上老师教的科目状态为 subject2的情况下,最少的雇佣费用. dp[cur][subje

【DP】树形DP 记忆化搜索

DP中的树形DP,解决方法往往是记忆化搜索.显然,树上递推是很困难的.当然做得时候还是得把状态定义和转移方程写出来:dp[u][1/0]表示以u为根节点的树 涂(1) 或 不涂(0) 颜色的最少方案数.树上DP有两个经典问法:一条边两端至少有个一个端点涂色,问整个tree最少涂色次数:还有一种忘了...此题是前种问法. #include<cstdio> #include<cstring> #include<algorithm> using namespace std;