秋招突击——6/22——复习{区间DP——加分二叉树,背包问题——买书}——新作{移除元素、实现strStr()}

news2025/1/7 6:44:16

文章目录

    • 引言
    • 复习
      • 区间DP——加分二叉树
        • 个人实现
      • 背包问题——买书
        • 个人实现
        • 参考实现
    • 新作
      • 移除元素
        • 个人实现
        • 参考思路
      • 找出字符串中第一个匹配项的下标
        • 个人实现
        • 参考实现
    • 总结

引言

  • 今天做了一个噩梦,然后流了一身汗,然后没起来,九点多才起床背书。十点钟才开始把昨天那道题题目过一遍,然后十一点才开始复习题目,为了不耽误下午的时间,所以这里的就单纯做已经做过的题目,主打一个有量,不在学习新的dp。

复习

区间DP——加分二叉树

  • 这道题差不多看了第二遍,第一次学的时候就蛮难的,不过有固定的编程范式,不过这里没想起来。提供一下之前的分析思路。

    • 第一次分析
    • 区间DP——合并石子
  • 之前做了两次,这次看看能不能一次过。

个人实现
  • 区间DP集合表示是f[i][j],
    • i是区间的左端点
    • j是区间的右端点,
  • 集合划分是区间的中间的,所以需要遍历枚举区间[i,j]中所有的区间分割点
  • 具体的代码流程如下
    • 列举区间长度
    • 列举区间左端点
    • 枚举的区间的分割点

实现问题

  • 区间长度的上下限有没有特殊情况,需不需要特殊考虑?如果区间长度为1,或者为n怎么办?
  • 特殊情况1
    • 区间长度为1,表示的左右子节是重合的,所以需要特殊考虑,当前情况是叶子节点
    • 区间长度就是1到n,不用考虑n+1,之前那个题目使用n+1是特殊情况,实际情况下,不需要使用使用n+1,而且这道题使用n+1也没有任何意义,那道题股票买卖题n+1一天的某种情况,是之前最大的。
#include <iostream>

using namespace std;

const int N = 35;
int f[N][N],n,rp[N][N];  // f表示区间的状态记录,r表示根节点的记录状态
int s[N];

void dfs(int l,int r){
    if (l > r)  return ;
    int k = rp[l][r];
    cout<<k<<" ";
    if (k - 1 >= l) dfs(l,k - 1);
    if (k + 1 <= r) dfs(k + 1,r);
}

int main(){
    cin>>n;
    for (int i = 1; i <= n; ++i) {
        cin>>s[i];
        s[i] += s[i - 1];
    }
    
    // 区间DP
    for (int len = 1; len <= n; ++len) {
        // 列举区间左端点
        for (int l = 1; l + len <= n; ++l) {
            // 计算区间的右端点
            int r = l + len  - 1;
            // 列举区间的分割点
            for (int k = l; k <= r; ++k) {
            	if(看== 1int temp;
                if (k == l){
                    // 左端点为空
                    temp = (s[k] -s[k - 1]) + f[k + 1][r];
                }else if(k == r){
                    // 右子树为空
                    temp = (s[k] -s[k - 1]) + f[l][k - 1];
                }else{
                    // 左右子树都不为空
                    temp = (s[k] -s[k - 1]) + f[k + 1][r] * f[l][k - 1];
                }
                // 比较确定最大值的并进行记录
                if (f[l][r] < temp) {
                    rp[l][r] = k;
                    f[l][r] = temp;
                }
            }
        }
    }
    
    cout<<f[0][n];
    dfs(0,n);
    
    return f[0][n];
    
}

修改之后的如下

  • 其实这里有两个地方还是可以修改的,左右节点的情况是可以特殊讨论和考虑的。
#include <iostream>

using namespace std;

const int N = 35;
int f[N][N],n,rp[N][N];  // f表示区间的状态记录,r表示根节点的记录状态
int s[N];

void dfs(int l,int r){
    if (l > r)  return ;
    int k = rp[l][r];
    cout<<k<<" ";
    if (k - 1 >= l) dfs(l,k - 1);
    if (k + 1 <= r) dfs(k + 1,r);
}

int main(){
    cin>>n;
    for (int i = 1; i <= n; ++i) {
        cin>>s[i];
    }
    
    // 区间DP
    for (int len = 1; len <= n; ++len) {
        // 列举区间左端点
        for (int l = 1; l + len <= n; ++l) {
            // 计算区间的右端点
            int r = l + len  - 1;
            if (l == r) f[l][r] = s[l],rp[l][r] = l;
            else
                // 列举区间的分割点
                for (int k = l; k <= r; ++k) {
                    int temp;
                    if (k == l){
                        // 左端点为空
                        temp = s[k] + f[k + 1][r];
                    }else if(k == r){
                        // 右子树为空
                        temp = s[k] + f[l][k - 1];
                    }else{
                        // 左右子树都不为空
                        temp = s[k] + f[k + 1][r] * f[l][k - 1];
                    }
                    // 比较确定最大值的并进行记录
                    if (f[l][r] < temp) {
                        rp[l][r] = k;
                        f[l][r] = temp;
                    }
                }
        }
    }
    
    cout<<f[0][n];
    dfs(0,n);
    
    return f[0][n];
    
}

背包问题——买书

在这里插入图片描述

  • 标准的背包问题
个人实现
  • 标准的背包问题,但是是计算方案数量的,而且会存在重复方案的问题,需要去除重复?是一个问题,并不是那么好去掉重复的。
  • 但是如果一开始是按照顺序进行购买的,后续只会增加更贵的物品,我应该如何进行计算?
  • 方案数应该怎么计算?直接求,整个矩阵的累加和就行了。
  • 注意,这里有一个,所有书全部用来购买书,手里不能有剩钱。

这是一个重复背包问题,属实不会弄,该想想怎么弄了!

#include <iostream>

using namespace std;

const int N = 10010;
int f[N],n;
int w[4] = {10,20,50,100};



int main(){
    cin>>n;
    int sum = 0;
    if (n % 10 != 0) cout<<0<<endl;
    else {
        for (int i = 0; i < 4; ++i) {
            for (int j = n; j >= w[i]; j--) {
                f[j] = f[j - w[i]] + f[j];
            }
        }
        cout << sum;
        
    }
}
参考实现
  • 参考作者:一只野生彩色铅笔
    链接:https://www.acwing.com/solution/content/52967/
    来源:AcWing
    著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

在这里插入图片描述

  • 下述方法还是很朴素的,确实如此,这道题暂时先跳过,这个方案转换公式是想到了,但是没有写出来。关键是他还有优化,下次再来吧,还要看redis,没那么多时间。
#include <iostream>

using namespace std;

const int N = 5, M = 1010;

int n = 4, m;
int v[N] = {0, 10, 20, 50, 100};
int f[N][M];

int main()
{
    //input
    cin >> m;

    //dp
    f[0][0] = 1;
    for (int i = 1; i <= n; ++ i)
    {
        for (int j = 0; j <= m; ++ j)
        {
            for (int k = 0; v[i] * k <= j; ++ k)
            {
                f[i][j] += f[i - 1][j - v[i] * k];
            }
        }
    }

    //output
    cout << f[n][m] << endl;

    return 0;
}

新作

移除元素

题目链接
在这里插入图片描述
在这里插入图片描述

个人实现

重点

  • 原地移除所有等于val的元素,后续的元素向前移动
  • 保证前k个元素是不等于val的,并且原来的顺序不变
  • nums的其余元素和nums的大小并不重要

实现思路

  • 两个指针,一个从前往后,一个从后往前
  • 从前往后的负责遍历对应的数组,判定当前的元素是否等于val,然后从后往前的元素负责将不相等的元素移动到从前往后的索引下方。

在这里插入图片描述

class Solution {
public:
    int removeElement(vector<int>& nums, int val) {

        int i = 0,r = nums.size() - 1;
        for (; i <= r ; ++i) {
        if (nums[i] == val) {
            while (r >= 0 && nums[r] == val) r--;
            if (i <= r ) nums[i] = nums[r],r--;
            else return i;
        }
    }
        return i;
    }
};

总结

  • 做是做出来了 ,但是说实话,稀碎,明明是道简单的题目,但是还是调了很久,思路很快就想到了,但是总是遇到各种各样的bug。
  • 有以下几个细节需要注意
    • 凡是涉及到数组索引的,都要进行检测,防止数组越界。这次在遍历右指针的时候,有好几个地方都没有对指针进行检查,导致不断报错溢出,很难受。
参考思路
  • 这个思路太暴力了,时间复杂度比我的高,完整遍历了整个数组。不对,其实我也是遍历了整个数组,不过我是双指针。而且是反向的,不行,这样完全没有必要。

  • 就是两个指针

    • idx表示实际的指针,只有不同的时候才会赋值并且递增
    • i表示遍历的指针,如果不同就执行赋值,如果相同就跳过。
  • 老是会把问题想的太复杂

class Solution {
public:
    int removeElement(vector<int>& nums, int val) {
 int idx = 0;
    for (int i = 0; i < nums.size(); ++i) {
        if (nums[i] != val){
            nums[idx] = nums[i];
            idx ++;
        }
    }
    return idx;
    }
};

找出字符串中第一个匹配项的下标

  • 题目链接

在这里插入图片描述

个人实现

重点

  • 第一个字符串中找出第二个字符串的第一个匹配项目的下标
    • 第二个字符串是一个子串,然后在第一个字符串中找到字串出现的第一个索引位置

思路实现

  • 别把这个问题想的太复杂,这是一道简单题,并不是中等题,所以怎么想就怎么实现。
  • 最直白的方式,就是两个指针进行遍历

在这里插入图片描述

class Solution {
public:
    int strStr(string h, string n) {
        bool f;
        for (int i = 0; i < h.size(); ++i) {
            f = true;
            for (int j = 0; j < n.size(); ++j) {
                if (h[i + j] != n[j])   {
                    f = false;
                    break;   
                }
            }
            if (f)  return i;
    }
    return -1;
    }
};
参考实现
  • 跳过吧,这个是用KMP算法的模板题,没有必要!
  • 这里仅仅贴一个代码就行了。
class Solution {
public:
    int strStr(string s, string p) {
        if (p.empty()) return 0;
        int n = s.size(), m = p.size();
        s = ' ' + s, p = ' ' + p;

        vector<int> next(m + 1);
        for (int i = 2, j = 0; i <= m; i ++ ) {
            while (j && p[i] != p[j + 1]) j = next[j];
            if (p[i] == p[j + 1]) j ++ ;
            next[i] = j;
        }

        for (int i = 1, j = 0; i <= n; i ++ ) {
            while (j && s[i] != p[j + 1]) j = next[j];
            if (s[i] == p[j + 1]) j ++ ;
            if (j == m) return i - m;
        }

        return -1;
    }
};

作者:yxc
链接:https://www.acwing.com/activity/content/code/content/347914/
来源:AcWing
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

总结

  • 今天两道简单的leetcode做完了,发现自己容易把问题想复杂,完全没有意义,简单题,怎么想怎么来。
  • 然后算法题复习题,又被背包问题给难住了,实际上能够写出状态转移方程,但是觉得有点麻烦,那个完全背包问题就划水划过去了。明天重点看一下这道题,然后再随便找一道区间DP的新题。

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

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

相关文章

Python + Playwright(0):从零开始学习Playwright自动化框架

Python Playwright&#xff08;0&#xff09;&#xff1a;从零开始学习Playwright自动化框架 简介一、官方文档二、安装安装要求pip安装 三、基本使用方法录制脚本 四、代码示例结语 简介 Playwright 是一个强大的自动化库&#xff0c;由微软开发&#xff0c;主要用于web端的…

会声会影2024永久破解和谐版下载 包含激活码序列号

亲爱的创作伙伴们&#xff0c;今天我要分享一个让我的影视编辑生活大放异彩的神器——会声会影2024破解版本&#xff01;&#x1f389;&#x1f31f; &#x1f308;**功能全面升级**&#xff1a;作为一款专业的视频编辑软件&#xff0c;会声会影2024破解版本不仅继承了之前版本…

CRMEB-PHP多商户版安装系统配置清单

系统在安装完成之后&#xff0c;需要对系统进行一系列的配置&#xff0c;才能正常使用全部的功能&#xff0c;以下是官方整理的配置清单 平台后台 商户后台

Linux常用

很早以前的 ls: 查看文件夹内所有文件 rz: windows的文件传到linux服务器 sz filename: 将文件下载到windows本地 ctrlinsert:复制 shiftinsert:粘贴 ctrlD&#xff1a;退出spark-shell 运行脚本并输出日志 nohup sh filename.sh > log.log 2>&1 & 查看日…

计算机视觉解决什么问题?

本节课为「计算机视觉 CV 核心知识」第一节课正式课&#xff1b; 「AI秘籍」系列课程&#xff1a; 人工智能应用数学基础人工智能Python基础人工智能基础核心知识人工智能BI核心知识人工智能CV核心知识 Hi&#xff0c;大家好。我是茶桁。 老同学对我应该都很熟悉了&#xff…

std::bind与std::ref配合使用时要注意的几个问题

目录 1 假如输入函数的变量是左值非常量引用&#xff0c;则该变量在std::bind中只能用std::ref修饰&#xff0c;不能用cref&#xff0c;否则编译失败&#xff1a; 2 假如输入函数的变量是左值常量引用&#xff0c;则该变量在std::bind中既可以用std::ref修饰&#xff0c;也可…

如何在 Mac 上清空硬盘后恢复丢失的数据?

如果您不小心从 Mac 硬盘上删除了重要文件&#xff0c;您可能会感到非常沮丧。但您仍然可以找回丢失的信息。将 Mac 想象成一个大盒子&#xff0c;里面装着所有东西。丢弃某样东西就像撕掉盒子上的标签&#xff1a;房间现在可以放新东西了&#xff0c;但旧东西仍然在那里&#…

leetcode 二分查找·系统掌握 有效的完全平方数

题目&#xff1a; 题解&#xff1a; 就是一个非常普通的二分查找&#xff0c;但是需要注意的是查找的上下界&#xff0c;因为是完全平方&#xff0c;所以可以把上界设为这个数的一半&#xff0c;但是要特殊处理num等于1的时候。 bool isPerfectSquare(int num) {if(num1)retur…

指令微调数据集构建方法

指令微调&#xff08;Instruction Tuning&#xff09;&#xff0c;是指使用自然语言形式的数据对预训练后的大语言模型进行参数微调&#xff0c;在一些文章中也称为有监督微调&#xff08;Supervised Fine-tuning&#xff0c;SFT&#xff09;或多任务提示训练&#xff08;Multi…

Window和linux杀死进程的方式(命令行版)

在本文中&#xff0c;我们将探讨如何在Windows和Linux操作系统下高效地终止指定的进程&#xff0c;涵盖基本命令与高级技巧&#xff0c;确保您能灵活应对各种管理需求。 linux杀死进程 在终端中&#xff0c;我们通过下面命令找到端口运行的程序 lsof -i:72812. 然后输入下面…

高考填报志愿不容易,压线考生怎么救?

每年的高考季 就是高考生们水深火热的一大月份&#xff0c;很多考生都会纠结要报考哪些学校&#xff0c;哪些专业好&#xff0c;并非每个学生从小就有明确的目标&#xff0c;很多人到6月份才深思这个问题&#xff0c;此时难免手慌脚乱&#xff0c;更别说一些考生的分数处于一本…

在C++中,构造器(Builder)模式的思考(《C++20设计模式》及常规设计模式对比)

文章目录 一、前言二、为什么需要Builder Pattern,Builder Pattern到底解决了什么实际问题&#xff1f;为什么不用set()方法&#xff1f;2.1 初学者有那些对象的属性初始化方法呢&#xff1f;2.1.1 构造函数的弊端2.1.1.1 对于属性的初始化只能是固定的顺序 2.1.2 用set()函数初…

k8s部署grafana beyla实现app应用服务依赖图可观测

k8s部署grafana beyla OS: Static hostname: test Icon name: computer-vm Chassis: vm Machine ID: 22349ac6f9ba406293d0541bcba7c05d Boot ID: 83bb7e5dbf27453c94ff9f1fe88d5f02 Virtualization: vmware Operating System: Ubuntu 22.04.4 LTS Kernel: Linux 5.15.0-105-g…

oracle 外连接(+)和left join用法

案例1&#xff1a; select count(1) FROM TFUNDINFO A, TFUNDTYPE B WHERE A.VC_FUNDCODEB.VC_FUNDCODE() select count(1) FROM TFUNDINFO A, TFUNDTYPE B WHERE A.VC_FUNDCODEB.VC_FUNDCODE SELECT count(1): 这表示查询将返回一个计数&#xff0c;count(1)是一种常见的计数…

创建阿里云的免费镜像仓库

1、登录 阿里云 首先进入阿里云的官网&#xff0c;如果没有注册的需要先注册&#xff0c;这里就不过多的讲解了。 2、搜索 登录完毕后点击右上角的控制台 进入管理页面。或者直接在搜索框中输入容器镜像服务 点击进入 这里我是已经开通过了&#xff0c;如果你还没有开通的…

JSON序列化与反序列化

目录 JSON序列化 查看JSON文件&#xff0c;设置数据模板类 ​编辑 Newtonsoft.Json下载 运行结果展示 JSON反序列化 序列化是将对象或数据结构转换为可以存储或传输的格式&#xff08;如JSON字符串&#xff09;的过程&#xff0c;而反序列化则是将这个格式的数据转换回原…

python3GUI--ktv点歌软件By:PyQt5(附下载地址)

文章目录 一&#xff0e;前言二&#xff0e;展示1.启动2.搜索2.服务1.首页2.天气预报3.酒水饮料4.酒水饮料2 3.服务4.灯光5.调音6.排行榜7.分类点歌9.歌手点歌10.歌手个人页 三&#xff0e;心得体会1.关于代码2.关于设计3.关于打包 四&#xff0e;总结 文件大小&#xff1a;33.…

APP启动流程解析

简单概括启动微信的流程就是&#xff1a; 1.Launcher通知AMS 要启动微信了&#xff0c;并且告诉AMS要启动的是哪个页面也就是首页是哪个页面 2.AMS收到消息告诉Launcher知道了&#xff0c;并且把要启动的页面记下来 3.Launcher进入Paused状态&#xff0c;告诉AMS&#xff0c…

[数据概念]一分钟弄懂数据治理

“ 数据治理是数据资产化的起点。” 数据资产化的趋势正愈演愈烈。然而&#xff0c;我们必须清醒地认识到&#xff0c;资产化的前提条件是拥有实际的数据资产。那么&#xff0c;这些宝贵的数据资产究竟源自何处呢&#xff1f;答案显而易见&#xff0c;它们源自企业日常运营中积…

车载以太网权威指南阅读笔记

总体思路&#xff1a; 要基于车载以太网做出相应的机器人以太网神经中枢&#xff0c;需要按照以下步骤&#xff1a; 了解车载以太网&#xff0c;包括但不限于 车载以太网是如何基于汽车需求定义的车载以太网的工作模式车载以太网工作所需要的硬件车载以太网中的数据交互模式 …