递归与递推

news2025/1/19 20:40:21

会独立敲一遍代码并debug(1,3题较难;2,4题较简单)

部分题需要买课,可到洛谷或其他OJ找原题

目录

📕空间复杂度(计算方法)

🌼1,费解的开关

🌼2,带分数

🌼3,飞行员兄弟

🌼4,翻硬币


📕空间复杂度(计算方法)

1byte(字节) = 8bit(位),位指二进制位

float = int = 4byte = 32bit

char = 1byte = 8bit

double = long long = 8byte = 64bit

已知bite = B, bit = b

64MB ≈ 64 * 1e6(B),M是兆,1G = 1024M,M也是百万的缩写

= 64 * 2^20(B) ≈ 6.4 * 10^7(B) = 1.6 * 10^7(int),即 1600万个int

考虑到额外占用的空间,一般开到1500万,换算成2维数组就是a[3800][3800]

🌼1,费解的开关

标签:递推,位运算,中等

95. 费解的开关 - AcWing题库

视频

AcWing 95. 费解的开关(蓝桥杯C++ AB组辅导课) - AcWing

题解 

AcWing 95. 费解的开关 - AcWing

位运算(👇目录 -- biset) 

STL入门 + 刷题(下)_千帐灯无此声的博客-CSDN博客

注意将 step = 0 和 memcpy() 放在 for(int op...) 里,因为32个种op代表第一行的32种方案数,每个op都要重新备份,都有自己的step

当你每次结果都输出 -1 时,就应该想到 step 或者 ret 的问题

AC  代码

#include<iostream>
#include<cstring> //memcpy()
using namespace std;
#define N 6

char m[N][N], backup[N][N]; //m[][]表示亮或不亮

int tx[5] = {0,1,0,-1,0}, ty[5] = {1,0,-1,0,0}; //上下左右中, 漏了中间

void turn(int a, int b) //对(a, b)所在点和上下左右共5个点操作
{
    for(int i = 0; i < 5; ++i) {
        int aa = a + tx[i], bb = b + ty[i];
        if(aa < 0 || aa > 4 || bb < 0 || bb > 4)
            continue; //越界
        //1^1 == 0, 0^1 == 1
        m[aa][bb] ^= 1; //等价于m[aa][bb] = m[aa][bb] ^ 1
    }
}

int main()
{
    int t;
    cin>>t;
    while(t--) {
        int ret = 10; //最小步数, 初始取尽可能的大

        for(int i = 0; i < 5; ++i)
            cin>>m[i]; //每一行按字符串输入到m[][], 表示亮或不亮

        for(int op = 0; op < 32; ++op) { //op表示第 0 行 按 / 不按
            
            //m拷贝到backup, 便于32组方案每组的恢复
            memcpy(backup, m, sizeof(m));
            
            int step = 0; //当前步数
            
            //操作第1行
            for(int i = 0; i < 5; ++i)  //比如op是13, 对应01101, 1按, 0不按
                if(op >> i & 1) { //右移 i 位 与 1 得到第 i 位的数字
                    step++;
                    turn(0, i); //第1行第i + 1列
                }
            //根据1234行操作2345行
            for(int i = 0; i < 4; ++i) //1234行
                for(int j = 0; j < 5; ++j)
                    if(m[i][j] == '0') { //主要是char
                        step++;
                        turn(i + 1, j);
                    }
            //是否能全部亮
            bool dark = false;
            for(int j = 0; j < 5; ++j)
                if(m[4][j] == '0') { //注意是char, 不要直接!
                    dark = true;
                    break;
                }
            if(!dark) ret = min(ret, step);
            memcpy(m, backup, sizeof(m)); //不要忘记拷贝回去
        }
        if(ret <= 6) cout<<ret<<endl;
        else cout<<-1<<endl;
    }

    return 0;
}

复杂度

由代码可知,32 * 25 * 5 * 500 = 2e6

1,第一行的 op 有32种方案

2,然后在32种方案的前提下,根据op对应的二进制对第一行操作,这里有5种

3,接着根据1~4行对2~5行操作,这里20种(5 + 20 = 25)

4,根据 turn() 函数,每个点要操作5次

5,最多500组测试

🌼2,带分数

标签:递归,搜索,剪枝

1209. 带分数 - AcWing题库

解法 -- 暴力

暴力的2种解法都是3000多ms

(1)全排列打乱1~9的顺序

(2)cal()函数得到对应的整数,比如345896712这个排列里,取第1~4位就会得到4589

(3)两层for循环遍历(两个分割点),以便得到a, b, c对应的位数

(4)最后判断满足条件,将除法转化为乘法,避免C++对小数的处理

复杂度

全排列最坏复杂度 < n! * n,视作 n! * n,加上2层for循环枚举位数,相当于9个数里插2个隔板,即 C_{8}^{2} = 28,已知 n = 9(9个数字全排列),

那么时间复杂度(运算次数) = 28 * 9! * 9 ≈ 1e8,刚好不超时

关于 next_permutation👇( 进入文章后,左侧目录跳转STL常用函数  第5个的next_permutation 

STL入门 + 刷题(下)_千帐灯无此声的博客-CSDN博客

AC  代码1

全排列 -- next_permutation() 

#include<iostream>
#include<algorithm> // next_permutation()
using namespace std;

int num[9] = {1,2,3,4,5,6,7,8,9};

int cal(int i, int j) // 比如 i=3,j = 5, 全排列前返回345
{
    int res = 0;
    for (int x = i - 1; x < j; ++x)
        res = res * 10 + num[x];
    return res;
}

int main()
{
    int target, a, b, c, ans = 0; // target = a + b/c
    cin>>target;
    do {
        // 两层for枚举左右边界(a, b, c的位数)
        for (int l = 1; l <= 7; ++l) // a=num[..l] b=num[l+1..r] c=num[r+1..]
            for (int r = l + 1; r <=8; ++r) {
                a = cal(1, l);
                b = cal(l+1, r);
                c = cal(r+1, 9);
                if (target * c == a * c + b) // 除法转乘法
                    ans++;
            }
    }while(next_permutation(num, num + 9)); // 全排列打乱顺序
    
    cout<<ans;
    
    return 0;
}

AC  代码2

dfs 深度优先搜索,手写全排列

#include<iostream>
using namespace std;

int num[9], vis[9];
int ans = 0, a, b, c, target;

int cal(int i, int j) // 比如 i=3,j = 5, 全排列前返回345
{
    int res = 0;
    for (int x = i - 1; x < j; ++x)
        res = res * 10 + num[x];
    return res;
}

void dfsabc() // 分配位数
{
    for (int l = 1; l <= 7; ++l)
        for (int r = l + 1; r <= 8; ++r) {
            a = cal(1, l);
            b = cal(l + 1, r);
            c = cal(r + 1, 9);
            if (target * c == a * c + b) 
                ans++;
        }
}

void dfs(int u) // dfs 全排列, u表示递归的深度(第u个数)
{
    if (u > 9) {
        dfsabc();
        return;
    }
    for (int i = 0; i <= 8; ++i) 
        if(!vis[i]) { // 未访问过
            num[u - 1] = i + 1; // u从1开始, i从0开始
            vis[i] = 1; // 标记
            dfs(u + 1); // 递归
            vis[i] = 0; // 取消标记
        }
}

int main()
{
    cin>>target;
    dfs(1);
    cout<<ans;
    
    return 0;
}

解释下第35行👇

num[u - 1] = i + 1; // u从1开始, i从0开始

dfs(int u)里的u表示递归的深度,即第 u 位数,因为 u 从1开始

i 从 0 开始,所以 num[u - 1] = i + 1

🌼3,飞行员兄弟

标签:枚举,位运算

116. 飞行员兄弟 - AcWing题库

视频讲解

AcWing 116. 飞行员兄弟 - AcWing

暴力解法

暴力解法,分数组和二进制优化,2份AC代码,前者用数组存储棋盘,后者用二进制存储棋盘

解释👇 

(1)每个位置最多操作1次(操作2次相当于没操作),最终结果与操作的顺序无关

(2)操作一个位置,所在的 行 和 列 全部反转

(3)采取暴力枚举

一共16个开关,每个开关摁或不摁2种可能,所以是 2^16 种方案

每个方案有16个开关要遍历,采取位运算的话,对7个位置(同一行 / 列)只需要操作一次

所以时间复杂度是 2^16 * 16 = 2^20 ≈ 1e6

----  当然,如果不用二进制优化,直接用数组存棋盘,复杂度 2^16 * (16 * 7 + 16 + 16)  ----

----  7表示每个开关共关联7个开关(同一行 / 列),两个16表示检查16个位置是否都开,以及都开的话,要记录结果  ----

(4)将4*4棋盘,按字符串读入4次,然后将读入的棋盘转化成一个整数(这个整数表示一个16位的二进制数),关于位运算👇

c++之位运算(详解,初学者绝对能看懂)_c++位运算_?!??的博客-CSDN博客

(5)常用技巧(判断一个二进制数 num 第 i 位是0还是1,第0位即低位第一位),只需要

num >> i & 1

(6)关于题目中的 “如果存在多种打开冰箱的方式,则按照优先级整体从上到下,同行从左到右打开”,只需要输出 0 ~ 2^16 - 1 里面字典序最小的方案(16位二进制中,1越靠前的方案,字典序越小),只需要从0开始遍历

常用技巧👇

 位运算常用技巧:用一个整数存储一个矩阵或者一维数组的信息(整数对应的二进制)

思路👇

1,棋盘的16个数字读入字符数组 g[][]

2,枚举 0 ~ 2^16 - 1 这 2^16 种方案 

3,每种方案,2层for循环遍历行 / 列,进行操作

补充👇 

(1)

为什么2^16种操作数里,每种操作数表示一种操作方案呢

一个操作数,比如6,的二进制表示为0000 0000 0000 0110

又比如2^16-1,二进制表示为1111 1111 1111 1111

这里的1表示需要操作,即开关是关着的,需要被打开

(2)

数组存棋盘,将 + 和 - 字符读入char[][]里,同时,为了判断某一种操作方案16个位置,某个位置是否需要打开,想象棋盘为

0  1  2  3  

4  5  6  7

8  9  10 11

12 13 14 15

只需要 操作数 << 棋盘数字,再 & 1,即可得到该位置是否需要打开

(3)

关于memcpy👇

std::memcpy - cppreference.com

// Defined in header <cstring>
void* memcpy( void* dest, const void* src, std::size_t count );

第 2 个参数,拷贝到第 1 个参数里,第 3 个参数为容器大小

(4)

关于pair👇

std::pair - cppreference.com

(5)

关于vector中operator=👇

std::vector<T,Allocator>::operator= - cppreference.com

(6)

关于for循环中的auto👇

Range-based for loop (since C++11) - cppreference.com

AC  数组

    op存储操作方案,op对应的二进制中,1表示需要操作,g[][]存储初始状态

#include<iostream>
#include<vector>
#include<algorithm>
#include<cstring> // memcpy()
using namespace std;

#define x first
#define y second

typedef pair<int, int> PII;
const int N = 5;

char g[N][N], backup[N][N]; // 存储棋盘,拷贝
vector<PII> ans; // 存储答案

int get(int x, int y)
{
    return x * 4 + y; // 返回第x行第y列的数字0~15
}

void turn_one(int x, int y) 
{
    if (g[x][y] == '-') g[x][y] = '+';
    else g[x][y] = '-'; // 改变一个开关
}

void turn_all(int x, int y) // 改变所有关联开关
{
    for (int i = 0; i < 4; ++i) {
            turn_one(x, i); // 列
            turn_one(i, y); // 行
    }
    turn_one(x, y); // 翻转了2次,多了1次
}

int main()
{
    // 读入棋盘
    for (int i = 0; i < 4; ++i)
        cin>>g[i]; // 读入一行字符
        
    // 遍历2^16种操作(数)方案
    for (int op = 0; op < 1 << 16; ++op) {
        memcpy(backup, g, sizeof g); // 备份: g拷贝到backup
        vector<PII> temp; // 存储当前所有操作
        // 枚举当前操作方案的16个开关
        for (int i = 0; i < 4; ++i)
            for (int j = 0; j < 4; ++j) 
                if (op >> get(i, j) & 1) { // 需要翻转
                    temp.push_back({i, j}); // 插入操作
                    turn_all(i, j); // 按一下开关
                }
                
        // 检查是否全部打开
        bool if_close = false;
        for (int i = 0; i < 4; ++i)
            for (int j = 0; j < 4; ++j)
                if (g[i][j] == '+')
                    if_close = true; 
        if (!if_close && (ans.empty() || ans.size() > temp.size())) 
            ans = temp; // 更新最小切换次数
        
        // 不要忘记备份回来, turn_all会修改原棋盘
        memcpy(g, backup, sizeof g);
    }
    
    cout<<ans.size()<<endl; // 最小次数
    for (auto op : ans) // 输出每一步操作
        cout<<op.x + 1<<" "<<op.y + 1<<endl; // 题意从1开始
    
    return 0;
}

AC  二进制

   比直接用数组存快了一倍

#include<iostream>
#include<vector>
using namespace std;

#define x first
#define y second
typedef pair<int, int> PII;

int change[5][5];

int get(int x, int y)
{
    return x * 4 + y;
}

int main()
{
    int state = 0; // state 存储棋盘初始状态
                   // 二进制中1表示关,0表示开
    // 读入
    for (int i = 0; i < 4; ++i) {
        string line; // 读入4行字符
        cin>>line;
        for (int j = 0; j < 4; ++j) 
            if (line[j] == '+') // 存下初始状态
                state += 1 << get(i, j); // +表示对应二进制位上的1
    }
    
    // 预处理一个开关以及同一行/列的所有数
    for (int i = 0; i < 4; ++i)
        for (int j = 0; j < 4; ++j) {
            for (int k = 0; k < 4; ++k) {
                // 注意3个都是 [i][j]
                change[i][j] += 1 << get(i, k); // 行
                change[i][j] += 1 << get(k, j); // 列
            }
            change[i][j] -= 1 << get(i, j); // 减去重复部分
        }
    
    // 遍历
    vector<PII> ans; // 存储答案的每一步操作
    for (int op = 0; op < 1 << 16; ++op) { // op 存储要操作的开关
        vector<PII> temp;
        int now = state; // 初始状态拷贝
        for (int i = 0; i < 16; ++i)  // 遍历16个开关,对应二进制的16位
            if (op >> i & 1) { // 这个开关需要操作
                int x = i / 4, y = i % 4; // 得到行 列
                now ^= change[x][y]; // 需要操作的数取反
                temp.push_back({x, y}); // 记录操作每一步
            }
        if (!now && (ans.empty() || ans.size() > temp.size()))
            ans = temp; // 满足更少操作次数
        
    }
    
    // 输出答案
    cout<<ans.size()<<endl;
    for (auto haha : ans)
        cout<<haha.x + 1<<" "<<haha.y + 1<<endl; // 题目从1开始
        
    return 0;
}

🌼4,翻硬币

1208. 翻硬币 - AcWing题库

标签:递推

思路

每次只改变相邻两个字符,由贪心可知,我们从第一个字符开始,如果第一个字符不一样,只能通过改变前2个字符来实现,由此可知,答案显然是唯一的

---- 从第一个字符遍历到最后,时间复杂度 O(n)

BUG

一开始,我用begin, end作为起始和目标字符串,报错,

error: reference to 'begin' is ambiguous

它与 C++ 标准库中的 begin 函数名产生了冲突

将begin改为Begin即可

AC 代码

#include<iostream>
using namespace std;

string Begin, End;

void change(int x)
{
    Begin[x] = Begin[x] == '*' ? 'o' : '*'; // 三目运算符
    Begin[x + 1] = Begin[x + 1] == '*' ? 'o' : '*';
}

int main()
{
    cin>>Begin>>End;
    
    int ans = 0;
    for (int i = 0; i < Begin.size() - 1; ++i) 
        if (Begin[i] != End[i]) {
            change(i);
            ans++;
        }
    cout<<ans;
    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/967041.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

图神经网络教程之HAN-异构图模型

异构图 包含不同类型节点和链接的异构图 异构图的定义&#xff1a;节点类别数量和边的类别数量加起来大于2就叫异构图。 meta-path元路径的定义&#xff1a;连接两个对象的复合关系&#xff0c;比如&#xff0c;节点类型A和节点类型B&#xff0c;A-B-A和B-A-B都是一种元路径。 …

[C++] STL_list常用接口的模拟实现

文章目录 1、list的介绍与使用1.1 list的介绍1.2 list的使用 2、list迭代器3、list的构造4、list常用接口的实现4.1 list capacity4.2 插入删除、交换、清理4.2.1 insert任意位置插入4.2.2 push_front头插4.2.3 push_back尾插4.2.4 erase任意位置删除4.2.5 pop_front头删4.2.6 …

Keil 编译 Debug

# 头文件无法导入进来 # 导入头文件&#xff0c;只有函数声明&#xff0c;但缺少函数实现 已经导入了air32f10x_gpio.h但是没有导入 .c&#xff0c;就导致 编译出错出现undefined symbol (某个函数)&#xff0c;这时候按照下面的操作&#xff0c;导入外设模块就好。

PQUEUE - Printer Queue

题目描述 The only printer in the computer science students union is experiencing an extremely heavy workload. Sometimes there are a hundred jobs in the printer queue and you may have to wait for hours to get a single page of output. Because some jobs are …

pip切换源

pip config set global.index-url https://pypi.tuna.tsinghua.edu.cn/simple

CLFS信息泄露漏洞CVE-2023-28266分析

引用 这篇文章的目的是介绍今年4月发布的CLFS信息泄露漏洞CVE-2023-28266分析. 文章目录 引用简介CVE-2023-28266漏洞分析CVE-2023-28266调试过程漏洞复现相关引用参与贡献 简介 文章结合了逆向代码和调试结果分析了CVE-2023-28266漏洞利用过程和漏洞成因. CVE-2023-28266漏洞…

两个线程同步执行:解决乱箭穿心(STL/Windows/Linux)

C自学精简教程 目录(必读) C并发编程入门 目录 多线程同步 线程之间同步是指线程等待其他线程执行完某个动作之后再执行&#xff08;本文情况&#xff09;。 线程同步还可以是像十字路口的红绿灯一样&#xff0c;只允许一个方向的车同行&#xff0c;其他方向的车等待。 本…

UART串口Shell软硬件模型分析总结

文章目录 层次一、最底层逻辑配置交互----如何从Uart硬件读写单个字节数据层次二、抽象串口软件模块交互----基于串口对接输入输出流 和 Printf适配层次三、类似Shell封装抽象交互----基于串口交互命令行界面&#xff08;命令解析、补全、修改、记录&#xff09;case1 依次输入…

自建音乐服务器Navidrome之一

这里写自定义目录标题 1.1 官方网站 2. Navidrome 简介2.1 简介2.2 特性 3. 准备工作4. 视频教程5. 界面演示5.1 初始化页5.2 专辑页 前言 之前给大家介绍过 Koel 音频流服务&#xff0c;就是为了解决大家的这个问题&#xff1a;下载下来的音乐&#xff0c;只能在本机欣赏&…

上海的正西边有哪些城市

背景 上海一路向西&#xff0c;来一趟拉萨之行&#xff0c;那么上海出现&#xff0c;所经过的那么多城市&#xff0c;哪些是在上海的正西边呢&#xff1f; 画一幅地图 基于这个背景需求&#xff0c;我们需要拿来一幅地图&#xff0c;一看便知。下面的python代码生成了一幅地…

通信原理板块——平稳随机过程

微信公众号上线&#xff0c;搜索公众号小灰灰的FPGA,关注可获取相关源码&#xff0c;定期更新有关FPGA的项目以及开源项目源码&#xff0c;包括但不限于各类检测芯片驱动、低速接口驱动、高速接口驱动、数据信号处理、图像处理以及AXI总线等 1、平稳随机过程的定义 (1)严平稳随…

UE4 显示遮挡物体

SceneDepth是你相机能够看见的物体的深度距离 CustomDepth是你相机包括看不见被遮挡的物体的深度距离 如果CustemDepth比SceneDepth的距离相等&#xff0c;那么就是没有被遮挡的物体&#xff0c;如果被遮挡那么就是CustemDepth比SceneDepth深度距离远&#xff0c;然后再做对应…

PYTHON知识点学习-循环语句

&#x1f680;write in front&#x1f680; &#x1f50e;大家好&#xff0c;我是Aileen★。希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流&#x1f50e; &#x1f194;本文由 Aileen_0v0★ 原创 CSDN首发&#x1f412; 如需转载还…

【机器学习】线性回归

Model Representation 1、问题描述2、表示说明3、数据绘图4、模型函数5、预测总结附录 1、问题描述 一套 1000 平方英尺 (sqft) 的房屋售价为300,000美元&#xff0c;一套 2000 平方英尺的房屋售价为500,000美元。这两点将构成我们的数据或训练集。面积单位为 1000 平方英尺&a…

Swift 如何从图片数据(Data)检测原图片类型?

功能需求 如果我们之前把图片对应的数据(Data)保持在内存或数据库中,那么怎么从 Data 对象检测出原来图片的类型呢? 如上图所示:我们将 11 张不同类型的图片转换为 Data 数据,然后从 Data 对象正确检测出了原图片类型。 目前,我们的代码可以检测出 jpeg(jpg), tiff,…

WebRTC 安全之一

WebRTC 的安全需要满足三个基本需求 Authentication 用户访问需要认证Authorization 用户访问需要授权Audit 用户的访问应该可被追踪和审查 其中前两项也可以归结为 CIA Confidentiality 机密性&#xff1a;信息需要保密&#xff0c; 访问权限也需要控制Integrity 完整性&#…

Spring Cloud集成Nacos配置中心/注册中心

Spring Cloud版本 2021.0.5 Spring Cloud Alibaba版本 2021.0.5.0 Spring Boot版本 2.7.10 pom文件 需要放在依赖管理的pom文件 <dependencyManagement><dependencies><!-- spring boot依赖 --><dependency><groupId>org.springframewor…

2023-9-3 试除法判定质数

题目链接&#xff1a;试除法判定质数 #include <iostream>using namespace std;bool is_prime(int n) {if(n < 2) return false;for(int i 2; i < n / i; i){if(n % i 0) return false;}return true; }int main() {int n;cin >> n;while(n--){int x;cin &g…

git大文件推送报错

报错信息 不多掰扯&#xff0c;直接上报错信息和截图 Delta compression using up to 8 threadsRPC failde; HTTP 413 curl 22 The requested URL returned error: 413 Request Entity Too Large从以上的报错信息不难看出推送仓库的时候&#xff0c;请求体过大&#xff0c;为…

C++ do...while 循环

不像 for 和 while 循环&#xff0c;它们是在循环头部测试循环条件。do…while 循环是在循环的尾部检查它的条件。 do…while 循环与 while 循环类似&#xff0c;但是 do…while 循环会确保至少执行一次循环。 语法 C 中 do…while 循环的语法&#xff1a; do {statement(s…