秋招突击——6/14——复习{(树形DP)树的最长路径,(单调队列优化DP)——最大子序和}——新作{四数之和}

news2025/1/16 0:45:23

文章目录

    • 引言
    • 复习
      • 树形DP——树的最长路径
        • 实现代码
        • 答疑
      • 单调队列优化DP——最大子序和
        • 个人实现思路
        • 参考思路分析
        • 实现代码
      • 无重复最长字串
        • 思路分析
        • 实现代码
    • 新作
      • 四数之和
        • 实现思路
          • 需要注意的问题
        • 参考代码
          • 分析思路
          • 实现代码
    • 总结

引言

  • 今天好好看看树的最长的路径,自己重新写一下,之前看过了 ,这个思路并不难的,然后在学一下单调队列优化DP,这个得看准时间,不能一次学过了,能学多少学多少。

复习

树形DP——树的最长路径

  • 以往的学习链接

  • 第一次分析

  • 第二次分析

  • 第三次分析

  • 这道题总得来事主要分为两个部分,分别是

    • 使用一维数组表示邻接链表,然后的进行逐个遍历
    • 在遍历邻接链表的过程中,分别使用动态规划,计算最大值和次大值
  • 已经分析过很多次了,这里直接给出实现过程

实现代码
#include <iostream>
#include <cstring>
using namespace std;

const int N = 10010;
int h[N],e[2*N],ne[2*N],w[2*N],idx;
int f1[N],f2[N],res = 0;
void add(int a,int b,int c){
    e[idx] = b;
    w[idx] = c;
    ne[idx] = h[a];
    h[a] = idx ++;
}

void dfs(int r,int father){
    // 这里需要对参数进行说明
    // r表示以当前节点为跟节点的子树
    // father表示节点r的父节点

    // 如果传入的节点是father,跳过
    if (r == father )   return ;

    // 遍历当前跟节点的所有的子节点
    f1[r] = f2[r] = 0;
    for (int i = h[r]; ~i ; i = ne[i]) {
        int j = e[i];
        // 进行递归,遍历更新对应的子节点
        dfs(j,r);
        // 情况一,大于最大的边
        if(w[i] + f1[j] > f1[r]) f1[r] = w[i] + f1[j],f2[r] = f1[r];
        // 情况二,不大于最大的边,大于次大的边
        if(w[i] + f1[j] > f2[r]) f2[r] = w[i] + f1[j];
    }
    res = max(res , f1[r] + f2[r]);
}

int main(){
    int n;
    cin>>n;
    memset(h,-1,sizeof(h));
    for (int i = 0; i < n - 1; ++i) {
        int a,b,c;
        cin>>a>>b>>c;
        add(a,b,c),add(b,a,c);
    }
    // 这里是有一个问题,本质上是一个无向树,是否需要遍历每一个根节点作为子节点
    // 个人猜测是需要,因为认为规定了父子节点的逆序关系
    // 但是这里的代码有没有保存,根本没有办法进行遍历
    dfs(1,-1);
    cout<<res;
    return 0;
}
答疑
  • 这里有两个地方搞不清楚

是否需要遍历所有的定点都当做树的根节点

  • 不需要,因为每一次计算的时候,都是计算了最长边和最短边,最终的路径是最长边和最短边同时相加,所以是一个双向的路径,所以不需要。
  • 任选一个节点作为根节点,然后指定父节点,这个树的唯一拓扑结构就确定了,不会存在出现歧义的情况。
  • 所以不需要遍历每一个节点作为根节点

判定是否为父节点的情况

  • 因为是无向边,所以在遍历所有邻接链表的过程中,会遍历到父节点,所以这里需要跳过,所以实在遍历子节点过程中,判定是否为父节点,我原来的代码写错了。

在这里插入图片描述
在这里插入图片描述

单调队列优化DP——最大子序和

  • 题目内容如下,这是一个单调队列优化DP,先给我五分钟,看一下,写一下思路。
    在这里插入图片描述
个人实现思路
  • 找一个连续子序列,使其和最大。一个子序列明显是包含了左端点和右端点,这里的直接暴力搜索,因该可以的。最终的时间复杂度是n的平方,但是这里有30万个节点,最终的结果肯定超过了运行的时间,不行。
  • 如何优化暴力搜索?
    • 会不会存在子串的情况,因为每一次都是完全从左到右的扫描的话,会存在重复使用子区间的情况,所以还是得想想看怎么弄,怎么优化?
    • 是不是状态DP,不是的,这是一个连续的数字,如果不是连续的数字,就可能是状态DP
    • 状态压缩DP,也不像,这里仅仅是有两个状态

真的服了,我这是完全没有看题,看成了任意长度,绝了,这是长度不超过m的连续子序列问题
看清题目!!!!!!!!!!

参考思路分析
  • 这里暂时理清了一部分思路,但是还是没有完全看懂,这个单调队列怎么推出来的,怎么写出来的。

  • 首先说一下问题转换

    • 将一个序列的最优值变成了的不同终点的序列中,计算最大值,再转成计算Sj的最小值
      在这里插入图片描述
  • 这里是有问题的,这里是计算{K-m,K-j}的最小值,是这个区间的最小值,怎么就变成了求他的最小值,编程了再{K-m,K}的序列中,计算以K-m为起点的,长度小于等于m的最小的连续序列====》计算最小值?怎么转换的?

  • 正常不应该是找出从{i-m}到j的不同的序列,找出其中的最小值吗?
    在这里插入图片描述

  • 上述前提有一个朴素的证明,那就是如果你下一个要增加的数字是不断递增的,那么你的结果就一定是越大的,但是如果你下一个数字不是单调递增的,那么你累加和得结果就有可能不是最大的,会变小,所以要比较。

在这里插入图片描述

实现代码
  • 这里直接贴代码,没看懂他的思路
  • 得,还是没有看懂,直接自己debug吧,自己理解吧。
#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;

typedef long long LL;

const int N = 3e5 + 10;

int n, m;
LL s[N], que[N];

int main()
{
    scanf("%d%d", &n, &m);
    for (int i = 1; i <= n; i ++ ) scanf("%lld", &s[i]), s[i] += s[i - 1];

    LL res = -1e18;
    int hh = 0, tt = 0; que[0] = 0;
    for (int i = 1; i <= n; i ++ )
    {
        while (hh <= tt && i - que[hh] > m) hh ++ ;
        res = max(res, s[i] - s[que[hh]]);
        while (hh <= tt && s[que[tt]] >= s[i]) tt -- ;
        que[ ++ tt] = i;
    }
    printf("%lld\n", res);
    return 0;
}

作者:一只野生彩色铅笔
链接:https://www.acwing.com/solution/content/67527/
来源:AcWing
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

无重复最长字串

  • 这道题是中等题,我记得我在华为面试手撕代码的时候做了,但是没有做出来,只通过了部分样例,那个时候是第三次做,没做好。

  • 关键是这道题第一次做,我做过了,第二遍是跟着解析做的,第三次没做出来,但是后续又刷了一遍,这是第四遍,看看哈。

  • 前几次的链接

    • 华为面试的结果
    • 第二次做
  • 好吧,看错了,原来拿到没做出来的题目是三数之和,这道题也是第二次做,看看哈。二十分钟,做出来。

思路分析
  • 这道题是双指针,使用hashmap维系左右指针所包含的所有的字符串的出现次数,具体实现代码如下
    在这里插入图片描述
实现代码
#include <iostream>
#include <unordered_map>
using namespace std;

int lengthOfLongestSubstring(string s){
    unordered_map<char,int > t;
    int len = s.size(),res = 0;
    int l = 0,r = 0;
    while(r < len){
        // 将右指针当前的字符的hash表加1
        t[s[r]]++;
        // 判定一下,当前的字母的是否出现过多次,如果出现过多次
        while (t[s[r]] > 1) {
            t[s[l]] --;
            l ++;
        }
        r ++;
        res = max(res,r - l);
    }
    return res;
}
int main(){

}

新作

四数之和

实现思路
  • n个整数组成的数组:是否是有序的,是否是重复的
  • 要求
    • 不重复的四元组
    • 累加和为target
  • 这个可以联想到三数之和问题,三数之和问题是拆解成两数之和解决的。
  • 需要将二维数组进行排序。
  • 如果将之转成三数之和,再转成两数之和,时间复杂度就是n的三次方,百万级别,应该能够接受,实现一下
  • 写是写完了,但是又遇到了重复的问题,除了使用set,并不知道如何进行去重了,我得趁着有时间,做一下尝试。
  • 在不使用set的情况下,只能写成这样了,不过肯定还有问题的,这里还是使用set实现一下。

在这里插入图片描述

  • 使用set实现了,效果如下
    在这里插入图片描述
    实现代码如下
class Solution {
public:
    vector<vector<int>> fourSum(vector<int>& nums,int target ){
    vector<vector<int>> res;
    sort(nums.begin(),nums.end());
    set<vector<int>> temp;
    int s = nums.size();
    if(s < 4)   return res;
    for (int i = 0; i < s; ++i) {
        for (int j = i + 1; j < s; ++j) {
            for (int l = j + 1; l < s; ++l) {
                int r = s - 1;
                while (r - 1 > l &&  long(nums[r - 1]) + long(nums[i]) + long(nums[j]) + long(nums[l]) >= target) r--;
                if (r > l && long(nums[r]) + long(nums[i]) + long(nums[j]) + long(nums[l]) == target){
                    temp.insert({ nums[i], nums[j], nums[l],nums[r]});
                }
            }
        }
    }
    for (auto x: temp) {
        res.push_back(x);
    }
    return res;
}

};
需要注意的问题
  • 如何更加有效的去重并不知道
  • 如何处理大数相加越界的问题,目前没有好办法。
参考代码
分析思路
  • 基本思路和我的相同
    如何去重
  • 这里是和前一个指针进行比较,如果和前一个状态一样,说明后续遍历的结果也是相同的
  • 这里很有细节,如果是第一个定位的元素,就要注明是大于0
  • 后续所有元素,都要确保第一个元素已经访问过了,然后在进行遍历访问。

在这里插入图片描述

  • 和之前的元素进行比较!!!!

处理越界问题

  • 这里直接使用特定情况进行处理,就几个样例。
实现代码
#include <iostream>
#include <vector>
#include <algorithm>
#include <set>
using namespace std;

vector<vector<int>> fourSum(vector<int>& nums,int target ){
    vector<vector<int>> res;
    sort(nums.begin(),nums.end());
    int s = nums.size();
    for (int i = 0; i < s; ++i) {
        if (i && nums[i - 1] == nums[i]) continue;
        for (int j = i + 1; j < s; ++j) {
            // 确保第一个元素已经访问过,然后在和前一个状态进行比较
            if (j > i + 1 && nums[j - 1] == nums[j]) continue;
            for (int l = j + 1; l < s; ++l) {
                if(l > j + 1 && nums[j] == nums[j - 1]) continue;
                int r = s - 1;
                while (r - 1 > l && nums[r - 1] + nums[i] + nums[j] + nums[l] >= target) r--;
                if (nums[r] + nums[i] + nums[j] + nums[l] == target){
                    res.push_back({nums[r], nums[i], nums[j], nums[l]});
                }
            }
        }
    }
    return res;
}

int main(){

}

总结

  • 今天关于单调队列有超时了,不应该,这个学习算法的时间不应该太多,不过就是有点烦,居然又没有看懂。
  • 不能再看了,超过了半个多少小时,一上午都在看这个了,不能浪费时间了。
  • 有点挫败,还是没有看懂,跳过吧,明天再来看,控制一个小时,学完java才是关键,算法不是关键。

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

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

相关文章

《Deep learning practice》learning notes

学习笔记&#xff1a; 【公开课】旷视x北大《深度学习实践》&#xff08;28课时全&#xff09; R Talk | 旷视科技目标检测概述&#xff1a;Beyond RetinaNet and Mask R-CNN 文章目录 Lecture 1: Introduction to Computer Vision and Deep Learning&#xff08;孙剑&#x…

apt-get update和apt-get upgrade的区别

apt-get update apt-get update 命令用于更新本地软件包列表。具体来说&#xff0c;做了以下事情&#xff1a; ①从 /etc/apt/sources.list 文件和 /etc/apt/sources.list.d/ 目录下的所有文件中读取软件源配置。 ②连接到这些软件源&#xff0c;并下载最新的软件包列表。 ③将…

1832javaERP管理系统之车间计划管理Myeclipse开发mysql数据库servlet结构java编程计算机网页项目

一、源码特点 java erp管理系统之车间计划管理是一套完善的web设计系统&#xff0c;对理解JSP java编程开发语言有帮助采用了serlvet设计&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统采用web模式&#xff0c;系统主要采用B/S模式开发。开发环境为TOMCAT7.0,Mye…

欧洲杯赛况@20240615

点击标题下「蓝色微信名」可快速关注 欧洲杯首战&#xff0c;德国5:1狂胜苏格兰&#xff0c;大比分、红点套餐、超新星登场进球&#xff0c;好像这些能想到的元素都发挥了作用&#xff0c;作为东道主&#xff0c;聚集了天时地利人和&#xff0c;可以说是完美&#xff0c;这就是…

【python】Sklearn—Cluster

参考学习来自 10种聚类算法的完整python操作示例 文章目录 聚类数据集亲和力传播——AffinityPropagation聚合聚类——AgglomerationClusteringBIRCH——Birch&#xff08;✔&#xff09;DBSCAN——DBSCANK均值——KMeansMini-Batch K-均值——MiniBatchKMeans均值漂移聚类——…

21. 第21章 算法分析

21. 算法分析 这个附录选自OReilly Media出版的Alen B.Downey的Think Complexity(2012)一书. 当你读完本书之后, 可能会像继续读读那本书.算法分析是计算机科学的一个分支, 研究算法的性能, 尤其是他们的运行时间和空间需求. 参见http://en.wikipedia.org/wiki/Analysis_of_al…

Mac M3 Pro安装Hadoop-3.3.6

1、下载Hadoop安装包 可以到官方网站下载&#xff0c;也可以使用网盘下载 官网下载地址&#xff1a;Hadoop官网下载地址 网盘地址&#xff1a;https://pan.baidu.com/s/1p4BXq2mvby2B76lmpiEjnA?pwdr62r提取码: r62r 2、解压并添加环境变量 # 将安装包移动到指定目录 mv …

回归预测 | Matlab实现GWO-ESN基于灰狼算法优化回声状态网络的多输入单输出回归预测

回归预测 | Matlab实现GWO-ESN基于灰狼算法优化回声状态网络的多输入单输出回归预测 目录 回归预测 | Matlab实现GWO-ESN基于灰狼算法优化回声状态网络的多输入单输出回归预测预测效果基本介绍程序设计参考资料 预测效果 基本介绍 1.Matlab实现GWO-ESN基于灰狼算法优化回声状态…

学习前端第一关

作者&#xff1a;周棋洛 文章目录 前言超文本标记语言&#xff1f;编辑器推荐手刃一个网页&#xff0c;Hello World&#xff01;网页骨架讲解标签学习HTML标题HTML段落HTML换行HTML居中HTML水平线HTML格式化HTML注释HTML链接 前言 网站并不是多么复杂的技术&#xff0c;今天这…

leetcode LRU 缓存

leetcode: LRU 缓存 LRU 全称为 Least Recently Used&#xff0c;最近最少使用&#xff0c;常常用于缓存机制&#xff0c;比如 cpu 的 cache 缓存&#xff0c;使用了 LRU 算法。LRU 用于缓存机制时&#xff0c;关键的是当缓存满的时候有新数据需要加载到缓存的&#xff0c;这个…

HTML5的未来:掌握最新技术,打造炫酷网页体验

引言 随着互联网技术的飞速发展&#xff0c;HTML5已经成为构建现代网页和应用的核心技术之一。HTML5不仅提供了丰富的语义化标签&#xff0c;还引入了多项前沿技术&#xff0c;使得网页体验更加丰富多彩。本文将探讨HTML5的最新技术&#xff0c;并结合行业实践&#xff0c;提供…

智能体(Agent)实战——从gpts到auto gen

一.GPTs 智能体以大模型作为大脑&#xff0c;同时配备技能&#xff0c;使其能够完成具体的任务。同时&#xff0c;为了应用于垂直领域&#xff0c;我们需要为大模型定义一个角色&#xff0c;并构建知识库。最后&#xff0c;定义完整的流程&#xff0c;使其完成整个任务。以组会…

GenICam标准(五)

系列文章目录 GenICam标准&#xff08;一&#xff09; GenICam标准&#xff08;二&#xff09; GenICam标准&#xff08;三&#xff09; GenICam标准&#xff08;四&#xff09; GenICam标准&#xff08;五&#xff09; GenICam标准&#xff08;六&#xff09; 文章目录 系列文…

什么是浏览器指纹

在数字互联网时代&#xff0c;我们的在线活动几乎都会留下痕迹。其中&#xff0c;浏览器指纹就像我们的数字身份证&#xff0c;让网站能够识别和追踪用户。本文将详细介绍浏览器指纹是什么&#xff0c;它如何工作。 一、什么是浏览器指纹 浏览器指纹&#xff08;Browser Fing…

【odoo】odoo.conf文件配置

概要 odoo.conf 文件是 Odoo 服务器的配置文件&#xff0c;它用于定义和管理 Odoo 运行时的各种参数。这个文件包含了许多配置选项&#xff0c;可以帮助管理员根据特定的需求和环境来调整 Odoo 服务器的行为。 主要功能 数据库连接设置&#xff1a;定义 Odoo 连接到 PostgreSQL…

vue项目问题汇总

1.el-select&#xff1a; 下拉框显示到了top:-2183px , 添加属性 :popper-append-to-body"false" 2. el-upload: 选过的文件在使用过后记得清空&#xff0c;因为如果有limit1的时候&#xff0c;没有清空会导致不触发onchange 使用自定义上传方法http-request的时…

C++ 47 之 函数调用运算符重载

#include <iostream> #include <string> using namespace std;class MyPrint{ public:// 重载小括号() 重载谁operator后就紧跟谁的符号void operator()(string txt){cout << txt << endl;} };class MyAdd{ public:int operator()(int a, int b){retur…

Android 断点续传实现原理

下载原理 在介绍断点续传之前&#xff0c;我们先说说下载的原理。代码示例用 OkHttp 作为示例。 下载核心思路是把 responseBody 写入文件&#xff0c;核心代码如下&#xff1a; 但是这种做法有个明显的问题&#xff0c;假如手机在下载文件的时候下载了80%&#xff0c;某些原…

[大模型]XVERSE-7B-chat langchain 接入

XVERSE-7B-Chat为XVERSE-7B模型对齐后的版本。 XVERSE-7B 是由深圳元象科技自主研发的支持多语言的大语言模型&#xff08;Large Language Model&#xff09;&#xff0c;参数规模为 70 亿&#xff0c;主要特点如下&#xff1a; 模型结构&#xff1a;XVERSE-7B 使用主流 Deco…

树莓派等Linux开发板上使用 SSD1306 OLED 屏幕,bullseye系统 ubuntu,debian

Raspberry Pi OS Bullseye 最近发布了,随之而来的是许多改进,但其中大部分都在引擎盖下。没有那么多视觉差异,最明显的可能是新的默认桌面背景,现在是大坝或湖泊上的日落。https://www.the-diy-life.com/add-an-oled-stats-display-to-raspberry-pi-os-bullseye/ 通过这次操…