解题思路:

题意:A的下面只能AA或BB,B的下面只能是AB或BA(各位想到二进制的异或没有);

如果:有1个A和2个B,那么有三种方案:

 


                     A(0 =1 ^ 1)
                                       B(1)                     ^B(1)

                     B(1 = 0 ^ 1)
                                       A(0)                     ^B(1)

                     B(1 = 1 ^ 0)
                                       B(1)                     ^A(0)

那么就可以用0代表A,1代表B;

还要一点要说,题目的意思是从下向上摆(下面多上面少),为了写程序方便(二维数组从一行开始),我们从从上向下摆,显然结果是一样的。

注意:下面是按反过了的模型说的(倒三角型))

那么只要知道第一行的数据,就可以依次更新下面的数据(反对角线方向从上向下);也就是说只用枚举一行的所有可能就可以了,第2行开始的结果都是根据上一行来的。

如何枚举一行:第一行每一个位置不是1就是0,每一个位置每次从0开始枚举,到1结束;枚举方向从左到右;

例如:拿样例来说吧:1个A,2个B也就是1个0,2个1;

(注意:这里枚举0,1是在第一行,其他行都是根据第一行来更新的)

0              // 第一行第一个位置为0,此时不用更新


0 0           // 第一行第二个位置为0,此时更新第二行

1              // 不合格(约束:1个0)


0 1           // 第一行第二个位置为1,此时更新第二行

1              // 合格,方案数加一


1             // 第一行第一个位置为1,此时不更新


1 0          // 第一行第二个位置为0,此时更新

1             // 合格,方案数加1


1 1         //  第一行第二个位置为1,此时更新                                

0            // 合格,方案数加1

此时已经枚举完毕,总方案数为3。

(注意:一直都是枚举的第一行的每一种方案,其他行都是根据第一行更新得到的)


如何统计A,B的个数(也就0,1的个数):加入用cnt计数,每次枚举的时候直接cnt+=枚举的那个位置上的数,如果是1那就加上了,是0就相当于没加,那这不就是统计了1的个数吗!! 0的个数用当前规模的总个数减去1的个数就可以了。


注意事项:


题目中提到:测试数据都合法。也就是说可以算出一行都多少个数字;

*    *    *     *

*    *    *

*    * 

*

上面的这个图,就是每次更新的范围,所有*的总个数就是A,B的总和;

在知道了A,B的个数之和,就可以算出第一行有几个数(人);

怎么算?? ::: 等差数列求和,公差为1;前n项和就是 (1 + n)* n / 2 

假设第一行只有一个数(x=1),计算以边前n项和看是否相等,如果相等那么第一行就有x个数,如果不x+1

,然后看是否相等,最后就得到了第一行能放几个数(x)。



注意:题目的要点就是枚举第一行有多少种可能,每种情况用来去更新其他行。

参考代码:

#include <bits/stdc++.h>
using namespace std;

typedef long long ll;
const int N = 510;

int m; // A的个数(即0的个数)
int n; // B的个数(即1的个数) 
int p[N][N]; // c存放当前的摆放方案 
int cnt; // 当前B的个数(即1的个数) 
int x = 1; // 第一行最多有多少个人 
ll ans; // 总的方案数 

void dfs(int cur) {
        // 0和1的个数大于给定的个数,直接返回上一层
	if(cnt > n || cur * (cur + 1) / 2 - cnt > m) return ;
	if(cur == x) {
		ans++;
		return ;
	}
	for(int i = 0; i < 2; ++i) {
		p[0][cur] = i;
		cnt += i; // 加号的数量
		for(int j = 1; j <= cur; ++j) { // 更新受影响的位置
			p[j][cur - j] = p[j - 1][cur - j] ^ p[j - 1][cur - j + 1];
			cnt += p[j][cur - j];
		}
		dfs(cur + 1); // 去下一个位置
		for(int j = 1; j <= cur; ++j) cnt -= p[j][cur - j]; // 回溯
		cnt -= i; // 回溯
	}
}

int main() {
	scanf("%d%d", &m, &n);
	while(x * (x + 1) / 2 != m + n) x++; // 第一行有多少个数字(题目中的人)
	dfs(0);
	printf("%lld\n", ans);
	return 0;
}


点赞(17)
 

0.0分

11 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 2 条评论

LBD西瓜泥 3年前 回复TA
666
shī 4年前 回复TA
强啊