0、剪枝的方法
data:image/s3,"s3://crabby-images/0afcf/0afcfe4494cd294f4d31b1c2dc05ea0c94b0b19c" alt="在这里插入图片描述"
1、小猫爬山
data:image/s3,"s3://crabby-images/0a771/0a771d85c1b51428956332416575718c94ea8cf0" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/c11e5/c11e50cc971b7941e7f63b7ea0268a75d3068ae7" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/07cc5/07cc5266c1210201041366d8ca13bc2c31bebd30" alt="在这里插入图片描述"
data:image/s3,"s3://crabby-images/fbf21/fbf211d5722aa2cf4396e736e5ebae20726158e0" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/b60c3/b60c3e6683e651c3cb3fd23c350285fd28e76873" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/77e20/77e20d783a2dca44e1fd6ceb68e23c114bc4708d" alt="在这里插入图片描述"
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
const int N = 20;
int n, m;
int w[N];
int sum[N];
int ans = N;
void dfs(int u, int k)
{
if (k >= ans) return;
if (u == n)
{
ans = k;
return;
}
for (int i = 0; i < k; i ++ )
if (sum[i] + w[u] <= m)
{
sum[i] += w[u];
dfs(u + 1, k);
sum[i] -= w[u];
}
sum[k] = w[u];
dfs(u + 1, k + 1);
sum[k] = 0;
}
int main()
{
cin >> n >> m;
for (int i = 0; i < n; i ++ ) cin >> w[i];
sort(w, w + n);
reverse(w, w + n);
dfs(0, 0);
cout << ans << endl;
return 0;
}
2、数独
data:image/s3,"s3://crabby-images/5a32f/5a32fd423fc42a3ec87199f47e5549da50f61937" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/d653b/d653b4ba3f77babdc6c2fff41ed6838d1448ef59" alt="在这里插入图片描述"
data:image/s3,"s3://crabby-images/a651e/a651ea2574b377e243c3fcc865f386096a3f33b0" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/cdbef/cdbef51723dd947045d9d585ec7a3709b72201a5" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/7753b/7753b59fa8c281ec1fddf4b2270c809742221eb9" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/c0d31/c0d31cbf3f86fcf1c05183bafce690ccd50e6594" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/f0d97/f0d9767d399bf349f2ec40e2f67d76a1cbf555c1" alt="在这里插入图片描述"
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
const int N = 9, M = 1 << N;
int ones[M], map[M];
int row[N], col[N], cell[3][3];
char str[100];
void init()
{
for (int i = 0; i < N; i ++ )
row[i] = col[i] = (1 << N) - 1;
for (int i = 0; i < 3; i ++ )
for (int j = 0; j < 3; j ++ )
cell[i][j] = (1 << N) - 1;
}
void draw(int x, int y, int t, bool is_set)
{
if (is_set) str[x * N + y] = '1' + t;
else str[x * N + y] = '.';
int v = 1 << t;
if (!is_set) v = -v;
row[x] -= v;
col[y] -= v;
cell[x / 3][y / 3] -= v;
}
int lowbit(int x)
{
return x & -x;
}
int get(int x, int y)
{
return row[x] & col[y] & cell[x / 3][y / 3];
}
bool dfs(int cnt)
{
if (!cnt) return true;
int minv = 10;
int x, y;
for (int i = 0; i < N; i ++ )
for (int j = 0; j < N; j ++ )
if (str[i * N + j] == '.')
{
int state = get(i, j);
if (ones[state] < minv)
{
minv = ones[state];
x = i, y = j;
}
}
int state = get(x, y);
for (int i = state; i; i -= lowbit(i))
{
int t = map[lowbit(i)];
draw(x, y, t, true);
if (dfs(cnt - 1)) return true;
draw(x, y, t, false);
}
return false;
}
int main()
{
for (int i = 0; i < N; i ++ ) map[1 << i] = i;
for (int i = 0; i < 1 << N; i ++ )
for (int j = 0; j < N; j ++ )
ones[i] += i >> j & 1;
while (cin >> str, str[0] != 'e')
{
init();
int cnt = 0;
for (int i = 0, k = 0; i < N; i ++ )
for (int j = 0; j < N; j ++, k ++ )
if (str[k] != '.')
{
int t = str[k] - '1';
draw(i, j, t, true);
}
else cnt ++ ;
dfs(cnt);
puts(str);
}
return 0;
}
3、 木棒
data:image/s3,"s3://crabby-images/334eb/334eb341e1077c79da7621a7cd59f03e2bd852d0" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/3a215/3a2157c8370f2928846430bc23dbb392115f4c00" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/bbca2/bbca27ffe7e488b8244e8baea6f9e4f29f33510d" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/543c4/543c43d5b8aea6d0537da914268c9a4a125f0f6d" alt="在这里插入图片描述"
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
const int N = 70;
int n;
int w[N];
int sum, length;
bool st[N];
bool dfs(int u, int cur, int start)
{
if (u * length == sum) return true;
if (cur == length) return dfs(u + 1, 0, 0);
for (int i = start; i < n; i ++ )
{
if (st[i] || cur + w[i] > length) continue;
st[i] = true;
if (dfs(u, cur + w[i], i + 1)) return true;
st[i] = false;
if (!cur || cur + w[i] == length) return false;
int j = i;
while (j < n && w[j] == w[i]) j ++ ;
i = j - 1;
}
return false;
}
int main()
{
while (cin >> n, n)
{
memset(st, 0, sizeof st);
sum = 0;
for (int i = 0; i < n; i ++ )
{
cin >> w[i];
sum += w[i];
}
sort(w, w + n);
reverse(w, w + n);
length = 1;
while (true)
{
if (sum % length == 0 && dfs(0, 0, 0))
{
cout << length << endl;
break;
}
length ++ ;
}
}
return 0;
}
4、 生日蛋糕
data:image/s3,"s3://crabby-images/5cfe5/5cfe55d3560c0c2e63b8453dca97c644d97be9e9" alt="在这里插入图片描述"
#include <cstring>
#include <iostream>
#include <algorithm>
#include <cmath>
using namespace std;
const int N = 25, INF = 1e9;
int n, m;
int minv[N], mins[N];
int R[N], H[N];
int ans = INF;
void dfs(int u, int v, int s)
{
if (v + minv[u] > n) return;
if (s + mins[u] >= ans) return;
if (s + 2 * (n - v) / R[u + 1] >= ans) return;
if (!u)
{
if (v == n) ans = s;
return;
}
for (int r = min(R[u + 1] - 1, (int)sqrt(n - v)); r >= u; r -- )
for (int h = min(H[u + 1] - 1, (n - v) / r / r); h >= u; h -- )
{
int t = 0;
if (u == m) t = r * r;
R[u] = r, H[u] = h;
dfs(u - 1, v + r * r * h, s + 2 * r * h + t);
}
}
int main()
{
cin >> n >> m;
for (int i = 1; i <= m; i ++ )
{
minv[i] = minv[i - 1] + i * i * i;
mins[i] = mins[i - 1] + 2 * i * i;
}
R[m + 1] = H[m + 1] = INF;
dfs(m, 0, 0);
if (ans == INF) ans = 0;
cout << ans << endl;
return 0;
}
|