题目
设某一机器由n个部件组成,每一种部件都可以从m个不同的供应商处购得。设wij是从供应商j 处购得的部件i的重量,cij是相应的价格。
试设计一个算法,给出总价格不超过d的最小重量机器设计。输入格式:
第一行有3 个正整数n ,m和d, 0<n<30, 0<m<30, 接下来的2n 行,每行m个数。前n行是c,后n行是w。
输出格式:
输出计算出的最小重量,以及每个部件的供应商
输入样例:
3 3 4 1 2 3 3 2 1 2 2 2 1 2 3 3 2 1 2 2 2
输出样例:
在这里给出相应的输出。例如:
4 1 3 1
1. 请用回溯法的方法分析“最小重量机器设计问题”
对于每个部件,在m个供应商里面依次尝试每个供应商,通过总价值进行截枝记录最小重量的供应商
1.1 说明“最小重量机器设计问题"的解空间
一共需要n个部件,每个部件可以选择m个供应商,由所有的可能组成的集合为该问题的解空间
1.2 说明 “最小重量机器设计问题"的解空间树
一个n+1层m叉树
1.3 在遍历解空间树的过程中,每个结点的状态值是什么
每个结点的状态值包括剩余价格,当时的选择部件的重量和,x[t]表示第t个部件选择第x[t]供应商
void backtrack(int t)
{
if(t > n)
{
if(bestm > cm)
{
for(int i=1; i <= n; i++)
{
last[i] = x[i];
}
bestm = cm;
}
}
else
{
for(int i=1; i <= m; i++)
{
if(bestm < cm+w[t][i])
continue;
if(cv >= c[t][i])
{
cv-=c[t][i];
cm+=w[t][i];
x[t] = i;
backtrack(t+1);
cv+=c[t][i];
cm-=w[t][i];
x[t] = 0;
}
}
}
}
1.4 如何利用限界函数进行剪枝
约束函数:总价格小于d
限界函数:此时的重量和小于之前得到的最小重量
2. 你对回溯算法的理解
回溯法
实际上回溯算法是不断地尝试搜索寻找到解(或者不满足题意的时候)就返回其他路径的过程,其中利用约束函数和限界函数减少时间复杂度。
回溯法有两种典型的解空间树:子集树和排列树
回溯法子集树的模型
void backtrack(int t)
{
if(t > n)
{
output(x);
return;
}
if()
{
x[t] = 1;
backtrack(t+1);
}
if()
{
x[t] = 0;
backtrack(t+1);
}
}
回溯法排列树的模型
void backtrack(int t)
{
int i;
if(t > n)
{
output(x);
return;
}
for(int i=t; i <= n; i++)
{
if()
{
swap(x[t],x[i]);
backtrack(t+1);
swap(x[t],x[i]);
}
}