题源 递归进行曲
题目描述
Perket 是一种流行的美食。为了做好 Perket,厨师必须谨慎选择食材,以在保持传统风味的同时尽可能获得最全面的味道。你有 n 种可支配的配料。对于每一种配料,我们知道它们各自的酸度 s 和苦度 b。当我们添加配料时,总的酸度为每一种配料的酸度总乘积;总的苦度为每一种配料的苦度的总和。
众所周知,美食应该做到口感适中,所以我们希望选取配料,以使得酸度和苦度的绝对差最小。
另外,我们必须添加至少一种配料,因为没有任何食物以水为配料的。
输入格式
第一行一个整数 n,表示可供选用的食材种类数。
接下来 n 行,每行 2 个整数 si 和 bi,表示第 i 种食材的酸度和苦度。
输出格式
一行一个整数,表示可能的总酸度和总苦度的最小绝对差。
输入输出样例
输入 #1复制
1 3 10输出 #1复制
7输入 #2复制
2 3 8 5 8输出 #2复制
1输入 #3复制
4 1 7 2 6 3 8 4 9输出 #3复制
1说明/提示
数据规模与约定
对于 100% 的数据,有 1≤n≤10,且将所有可用食材全部使用产生的总酸度和总苦度小于1×10^9,酸度和苦度不同时为 1 和 0。
两种递归解法:
法一:
#include<bits/stdc++.h>
using namespace std;
long long ans=10000000;
int n;
long long s[15],b[15];
void dfs(int index,long long x,long long y){
if(index>=n){
if(x==1&&y==0){//这个位置一定一定要进行判断,否则就没法全过
return ;
}
ans=min(ans,abs(x-y));
return ;
}
dfs(index+1,x*s[index],y+b[index]);//选择
dfs(index+1,x,y);//不选择
}
int main()
{
cin>>n;
for(int i=0;i<n;i++){
cin>>s[i]>>b[i];
}
dfs(0,1,0);
cout<<ans;
return 0;
}
法二:
#include<bits/stdc++.h>
using namespace std;
long long ans=10000000;
int n;
long long s[15],b[15];
long long tp1=1,tp2=0;
int sg[15]={0};//用于标记
void dfs(int t)
{
if(t>=n)
{
return ;
}
else
{
for(int i=0;i<n;i++)
{
if(sg[i]==0) //标记筛选
{
tp1*=s[i];
tp2+=b[i];
sg[i]=1; //记录
ans=min(ans,abs(tp1-tp2)); //取最小值
dfs(t+1);
tp1/=s[i];//回溯一哈
tp2-=b[i];
sg[i]=0;
}
}
}
return ;
}
int main()
{
cin>>n;
for(int i=0;i<n;i++){
cin>>s[i]>>b[i];
}
dfs(0);
cout<<ans;
return 0;
}