暴力枚举:
直接用next_pernutation,只能得到70分
#include<bits/stdc++.h>
using namespace std;
int a[20];
int b[20][20];
int main()
{
int n,ans=0;
cin>>n;
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
cin>>b[i][j];
for(int i=1;i<=n;i++)
a[i]=i;
do
{
int temp=0;
for(int i=1;i<=n;i++)
temp+=b[i][a[i]];
ans=max(ans,temp);
}while(next_permutation(a+1,a+1+n));
cout<<ans<<endl;
return 0;
}
状压dp
#include<bits/stdc++.h>
using namespace std;
int dp[14][1<<14];//i个人选择状态为j的最大收益
int a[14][14];
int main()
{
int n;
cin>>n;
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
cin>>a[i][j];
for(int i=1;i<=n;i++)//第i个人
{
for(int j=0;j<(1<<n);j++)//枚举状态
{
for(int k=1;k<=n;k++)//选第k个学生
{
if((j&(1<<k-1))==0)//没有
dp[i][j|(1<<k-1)]=max(dp[i][j|(1<<k-1)],dp[i-1][j]+a[i][k]);
}
}
}
cout<<dp[n][(1<<n)-1]<<endl;
return 0;
}
DFS加枝减
#include <stdio.h>
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 20;
int g[N][N], p[N];
bool use[N];
int n, ans;
void DFS(int d, int s)
{
if (d > n)
{
ans=max(ans, s);
return;
}
for (int i = 1; i <= n; ++i)
if (!use[i] && s + g[d][i] + p[d + 1] > ans) //如果下面加p仍然不大于答案则不继续
{
use[i] = 1;
DFS(d + 1, s + g[d][i]);
use[i] = 0;
}
}
int main()
{
cin >> n;
for (int i = 1; i <= n; ++i)
for (int j = 1; j <= n; ++j)
scanf("%d", &g[i][j]);
for (int i = n; i >= 1; --i)
{
int mx = 0;
for (int j = 1; j <= n; ++j)
mx=max(mx, g[i][j]);
p[i] = p[i + 1] + mx; //不限制不同列的最大值前缀
}
DFS(1, 0);
cout << ans << endl;
return 0;
}