C++课后习题训练记录Day120

1.练习项目:

问题描述

给定一个 n×n 的棋盘。现在要向棋盘中放入 n 个黑皇后和 n 个白皇后,使任意的两个黑皇后都不在同一行、同一列或同一条对角线上,任意的两个白皇后都不在同一行、同一列或同一条对角线上。

问总共有多少种放法?

输入格式

输入的第一行包含一个整数 n。

输出格式

输出一行包含一个整数,表示答案。

2.选择课程

在蓝桥云课中选择题库,选择题号17013并开始练习。

3.开始练习

(1)源码 :

#include<bits/stdc++.h>

using namespace std;

typedef long long ll;

const int N=15;

ll n,ans;

ll vis_bNN,vis_wNN;

bool posNN;

void dfs_w(int dep)

{

if(dep==n+1){

ans++;

return;

}

for(int i=1;i<=n;i++){

if(vis_wdepi)continue;

if(posdepi)continue;

for(int j=1;j<=n;j++)vis_wji++;

for(int k=i,t=dep;k<=n&&t<=n;k++,t++)vis_wtk++;

for(int k=i,t=dep;k>=1&&t>=1;k--,t--)vis_wtk++;

for(int k=i,t=dep;k<=n&&t>=1;k++,t--)vis_wtk++;

for(int k=i,t=dep;k>=1&&t<=n;k--,t++)vis_wtk++;

dfs_w(dep+1);

for(int j=1;j<=n;j++)vis_wji--;

for(int k=i,t=dep;k<=n&&t<=n;k++,t++)vis_wtk--;

for(int k=i,t=dep;k>=1&&t>=1;k--,t--)vis_wtk--;

for(int k=i,t=dep;k<=n&&t>=1;k++,t--)vis_wtk--;

for(int k=i,t=dep;k>=1&&t<=n;k--,t++)vis_wtk--;

}

}

void dfs_b(int dep) {

if(dep == n+1) {

dfs_w(1);

return;

}

for(int i = 1; i <= n; i++) {

if(vis_bdepi) continue;

posdepi = true;

for(int j = 1; j <= n; j++)vis_bdepj++,vis_bji++;

for(int k=i,t=dep; k<=n&&t<=n; k++,t++) vis_btk++;

for(int k=i,t=dep; k>=1&&t>=1; k--,t--) vis_btk++;

for(int k=i,t=dep; k<=n&&t>=1; k++,t--) vis_btk++;

for(int k=i,t=dep; k>=1&&t<=n; k--,t++) vis_btk++;

dfs_b(dep + 1);

posdepi = false;

for(int j = 1; j <= n; j++)vis_bdepj--,vis_bji--;

for(int k=i,t=dep; k<=n&&t<=n; k++,t++) vis_btk--;

for(int k=i,t=dep; k>=1&&t>=1; k--,t--) vis_btk--;

for(int k=i,t=dep; k<=n&&t>=1; k++,t--) vis_btk--;

for(int k=i,t=dep; k>=1&&t<=n; k--,t++) vis_btk--;

}

}

int main()

{

ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);

cin>>n;

dfs_b(1);

cout<<ans<<'\n';

return 0;

}

(2)检验结果

对此代码进行检验,检验后无报错,提交此代码,判题结果为正确90分,发现最后一个案例超时,对源代码进行优化,优化过后的代码如下:

#include<bits/stdc++.h>

using namespace std;

const int N = 15;

int n, cnt = 0; // cnt: 单色皇后解的数量

int pos3000N; // poski 表示第k个解中第i行皇后所在的列

int visNN; // 攻击标记

int locN; // 当前解的各行皇后列位置

void dfs(int dep) {

if (dep == n + 1) {

cnt++;

for (int i = 1; i <= n; i++)

poscnti = loci;

return;

}

for (int i = 1; i <= n; i++) {

if (visdepi) continue;

locdep = i;

// 标记攻击范围(保留你熟悉的二维标记方式)

for (int j = 1; j <= n; j++) visdepj++, visji++;

for (int k = i, t = dep; k <= n && t <= n; k++, t++) vistk++;

for (int k = i, t = dep; k >= 1 && t >= 1; k--, t--) vistk++;

for (int k = i, t = dep; k <= n && t >= 1; k++, t--) vistk++;

for (int k = i, t = dep; k >= 1 && t <= n; k--, t++) vistk++;

dfs(dep + 1);

// 回溯

for (int j = 1; j <= n; j++) visdepj--, visji--;

for (int k = i, t = dep; k <= n && t <= n; k++, t++) vistk--;

for (int k = i, t = dep; k >= 1 && t >= 1; k--, t--) vistk--;

for (int k = i, t = dep; k <= n && t >= 1; k++, t--) vistk--;

for (int k = i, t = dep; k >= 1 && t <= n; k--, t++) vistk--;

}

}

int main() {

ios::sync_with_stdio(false); cin.tie(0);

cin >> n;

dfs(1);

int ans = 0;

for (int i = 1; i <= cnt; i++) { // 枚举黑皇后解

for (int j = 1; j <= cnt; j++) { // 枚举白皇后解

bool ok = true;

for (int r = 1; r <= n; r++) { // 检查每一行是否同列(同格)

if (posir == posjr) {

ok = false;

break;

}

}

if (ok) ans++;

}

}

cout << ans << '\n';

return 0;

}

对此代码进行检验,检验后无报错,提交此代码,判题结果为正确100分。

(3)练习心得:注意每段代码末尾的分号是否存在,如不存在则需即使补充;输入法是否切换 为英语模式;语法是否错误。

相关推荐
apocelipes1 天前
常用编程语言和库的正则表达式性能对比
c语言·c++·python·性能优化·golang·开发工具和环境
郝学胜_神的一滴2 天前
CMake 034:生成器表达式:解耦构建时序、精简分支逻辑的终极利器
c++·cmake
见过夏天3 天前
C++ 基础入门完全指南
c++
用户805533698034 天前
不止三件套:QObject 属性系统全关键字与运行时反射!
c++·qt
BadBadBad__AK5 天前
线段树维护区间 k 次方和
c++·数学·算法·stl
卷无止境5 天前
Eigen 库如何借助 OpenMP 加速计算
c++·后端
卷无止境5 天前
OpenMPI、MPICH 与 OpenMP:关系、核心概念与架构全解
c++·后端
郝学胜_神的一滴6 天前
CMake 30:循环语法全解|foreach_while双循环精讲、迭代技巧与实战避坑指南
c++·cmake
卷无止境8 天前
C++ 的Eigen 库全解析
c++
卷无止境8 天前
现代 C++特性大盘点:一门脱胎换骨的老语言
c++·后端