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

相关推荐
我命由我123457 分钟前
由 ImageView 获取到的 Drawable 对象,它的 intrinsicWidth、intrinsicWidth 与实际图片的尺寸
java·开发语言·java-ee·android studio·android jetpack·android-studio·android runtime
xuankuxiaoyao8 分钟前
Axios-图书列表案例
开发语言·前端·javascript
guslegend10 分钟前
Java 创建对象有几种方式
java·开发语言
带娃的IT创业者12 分钟前
深度解析 Bun:重新定义 JavaScript 运行时的性能边界
开发语言·javascript·node.js·ecmascript·bun·运行时
布朗克16814 分钟前
29 反射机制
java·开发语言·反射
San813_LDD16 分钟前
[数据结构]共享栈与双端队列:算法思想分析及C语言实现
java·开发语言·数据结构
우리帅杰17 分钟前
【AI测试】Python AI大模型介绍
开发语言·人工智能·python·ai编程
我是一颗柠檬18 分钟前
【Java项目技术亮点】全链路分层限流:从网关到数据库的多层防护体系
java·开发语言·数据库
The Sheep 202318 分钟前
C#多线程学习
开发语言·学习·c#
Shadow(⊙o⊙)23 分钟前
QT常用控件1.0,enabled() geometry() QIcon的.qrc文件导入
开发语言·c++·qt