顺序表经典算法

顺序表经典算法

1.移除元素

题目:

给你一个数组 nums 和一个值 val,你需要 原地 移除所有数值等于 val 的元素,并返回移除后数组的新长度。

不要使用额外的数组空间,你必须仅使用 O(1) 额外空间并 原地 修改输入数组

元素的顺序可以改变。你不需要考虑数组中超出新长度后面的元素。

解答:

对于这个题我们首先可以想到一个思路

就是创建另外一个数组tmp去接受原数组不是val的值

这样也可以实现移除元素 但是 题目把这个思路限制了

image-20240418162502281

我们来看题目的示例:

/*
示例 1:输入:nums = [3, 2, 2, 3], val = 3
输出:2, nums = [2, 2]
解释:函数应该返回新的长度 2, 并且 nums 中的前两个元素均为 2。你不需要考虑数组中超出新长度后面的元素。
例如,函数返回的新长度为 2 ,而 nums = [2, 2, 3, 3] 或 nums = [2, 2, 0, 0],也会被视作正确答案。
示例 2:输入:nums = [0, 1, 2, 2, 3, 0, 4, 2], val = 2
输出:5, nums = [0, 1, 3, 0, 4]
解释:函数应该返回新的长度 5, 并且 nums 中的前五个元素为 0, 1, 3, 0, 4。
注意这五个元素可为任意顺序。你不需要考虑数组中超出新长度后面的元素。
*/

根据题目的思路和实例

我们采用双指针法来解决这个问题:

image-20240418163941523

// 我们采用双指针法
int removeElement(int* nums, int numsSize, int val)
{int src, dst;src = dst = 0;while (src < numsSize){if (nums[src] == val){src++;}else{nums[dst] = nums[src];src++;dst++;}}return dst;
}int main()
{int nums[5] = { 1,2,2,3,5 };int ret = removeElement(nums, 5, 2);printf("长度为:%d  [", ret);for (int i = 0; i < ret; i++){printf("%d ", nums[i]);}printf("]\n");return 0;
}

2.删除有序数组的重复项

删除有序数组中的重复项

采用双指针法:

  1. 首先我们让l1 和 l2 分别指向数组的第一个元素和第二个元素
  2. 我们让l1和l2指向的元素进行对比
  3. 如果相等 就让l2向后遍历 直至找到不同的元素为止(注意不要越界)
  4. 找到了就让l1++ 让新元素覆盖掉重复的元素
  5. 如果不相等,那就让l1++,l2++
  6. 最后返回l1+1 这个就是有效长度

我们来看代码如何实现:

int removeDuplicates(int* nums, int numsSize)
{// 创建两个指针 分别指向数组的首元素和第二个元素int l1 = 0;int l2 = 1;// 遍历数组  让l1 和 l2 去对比  while (l2 < numsSize){// 如果nums[l1] == nums[l2] 那就让l2往后找到不一样的元素为止if (nums[l1] == nums[l2]){while (l2 < numsSize && nums[l2] == nums[l1]){l2++;}// 走到这里 l2找到了不一样的元素// 找到了 就让l1往前走一步 让新元素覆盖掉重复的元素if (l2 < numsSize){l1++;nums[l1] = nums[l2];}}else // 不等于的话就让两个一起往后走{l1++;l2++;}}return l1 + 1; // 返回新的长度  + 1是因为 从0开始的
}

3.合并两个有序数组

题目:

给你两个按 非递减顺序 排列的整数数组 nums1nums2,另有两个整数 mn ,分别表示 nums1nums2 中的元素数目。

请你 合并 nums2nums1 中,使合并后的数组同样按 非递减顺序 排列。

**注意:**最终,合并后数组不应由函数返回,而是存储在数组 nums1 中。为了应对这种情况,nums1 的初始长度为 m + n,其中前 m 个元素表示应合并的元素,后 n 个元素为 0 ,应忽略。nums2 的长度为 n

示例:

示例 1:输入:nums1 = [1,2,3,0,0,0], m = 3, nums2 = [2,5,6], n = 3
输出:[1,2,2,3,5,6]
解释:需要合并 [1,2,3][2,5,6] 。
合并结果是 [1,2,2,3,5,6] ,其中斜体加粗标注的为 nums1 中的元素。
示例 2:输入:nums1 = [1], m = 1, nums2 = [], n = 0
输出:[1]
解释:需要合并 [1][] 。
合并结果是 [1] 。
示例 3:输入:nums1 = [0], m = 0, nums2 = [1], n = 1
输出:[1]
解释:需要合并的数组是 [][1] 。
合并结果是 [1] 。
注意,因为 m = 0 ,所以 nums1 中没有元素。nums1 中仅存的 0 仅仅是为了确保合并结果可以顺利存放到 nums1 中。

解答:

对于这个问题,我们先来看一个之前我们熟悉的思路

思路1:

我们把nums2数组的元素 放到 nums1数组里面去 然后再用冒泡排序去对nums1进行排序。

这个思路是可以的 但是我们知道冒泡排序的由于有两个for循环嵌套,效率没有那么理想, 因此我们这个时候就可以考虑另外一个思路

思路2:

我们可以让三个指针分别指向nums1的有效数字的最后一个数字、最后一个空间,以及nums2的最后一个数字 这三个指针分别是 l1 l3 l2。

第一种情况:(l2先出循环)

让l1 和 l2 指向的数字去比较 如果l2大就放到l3 然后 l3-- l2–

然后继续比较

直到遇到l1大于l2 那就把l1放到l3 直到l2–到数组外边

如果所示

image-20240418182123239

第二种情况:(l1先出循环)

逻辑是和第一种情况一样的

l1 和 l2 指向的数字进行比较,谁大谁就放到l3

并且和l3一起–

但是由于l1先出了循环 导致nums2 还有数字没有存放到l1中

如图所示:

image-20240418182653981

因此我们还需要将剩余的数字放到nums1中

我们通过循环 去把nums2中的数据去放到l3中

每放一个数字 l2和l3都要–

image-20240418182812022

直至l2也出了边界 这个时候就完成了

讲完了思路 我们来看代码是如何实现的:

void merge(int* nums1, int m, int* nums2, int n)
{// 首先我们创建三个int变量作为下标  // l1指向nums1的最后一个数字 l2指向nums2的最后一个数字 l3指向nums1的最后一个空间int l1, l2, l3;l1 = m - 1;l2 = n - 1;l3 = m + n - 1;while (l1 >= 0 && l2 >= 0) // 只要l1 和 l2 < 0 就要退出循环 单独处理{// 判断l1 和 l2 指向的数字谁大 谁大 就放到l3处if (nums1[l1] > nums2[l2]){nums1[l3--] = nums1[l1--]; // 别忘了--}else // 这里说明l2大{nums1[l3--] = nums2[l2--];}}// 走到这里说明 要不就排好了 要不就是l2 或者 l1 出了边界// 而我们只需要对l1出边界的情况做好处理  (因为l1和l2 不会同时出边界 如果l2出了边界就说明排好了)// l1出边界 就说明 nums2还有数字没有放到nums1中 while (l2 >= 0){nums1[l3--] = nums2[l2--];}
}int main()
{int nums1[] = { 1 };int nums2[] = {0};merge(nums1, 1, nums2, 0);for (int i = 0; i < 1; i++){printf("%d ", nums1[i]); // 1}return 0;
}

4.顺序表的问题及思考

这里有关顺序表的三个问题:

  1. 中间/头部的插入删除,时间复杂度为O(N)
  2. 增容需要申请新空间,拷贝数据,释放旧空间。会有不小的消耗
  3. 增容一般是呈2倍的增长,势必会有一定的空间浪费,例如容量为100,增容到200,但是只补充5个数据,那就浪费了95个数据

其实这三个问题总结下来就是:

  1. 中间/头部的插入效率地下
  2. 增容降低运行效率
  3. 增容造成空间浪费

那么我们要如何解决以上问题呢?

答案就是数据结构中的——链表

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

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

相关文章

Linux编辑器调试器 gcc/g++ gdb 编译过程及使用讲解

这恋爱呀 我有两不谈 第一异性不谈 因为我们性别不一样 我知道的她不知道相处起来太累 第二同性不谈 因为我们性别一样 我知道的他也知道相处起来太无聊了 –❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀–❀-正文开始-❀–❀–❀–❀–❀–❀–…

雅思(IELTS)优秀小作文分享

IELTS优秀小作文分享 柱状图 本篇范文个人评分是8分或者8.5分&#xff0c;属于能找到的最优质的范文了 题目如下: The two sets of bar charts illustrate the amount of time that teenagers (boys, girls, and all) in the UK spend chatting online and playing game c…

DHCPv4_CLIENT_ALLOCATING_01: 在其本地物理子网上广播DHCPDISCOVER消息

测试目的&#xff1a; 确保客户端能够在其本地物理子网上广播DHCPDISCOVER消息。 描述&#xff1a; 该测试用例旨在验证DHCP客户端是否能够正确地在其本地物理子网上广播DHCPDISCOVER消息&#xff0c;以便进行IP地址的自动分配。 测试拓扑&#xff1a; 测试步骤&#xff1a…

汇报进度26届cpp,目前来说之后的规划,暑假打算回家10天就留校沉淀了

汇报一下进度吧&#xff0c;26双非菜鸡&#xff0c;cpper. 但目前学了一些go &#xff0c;辅修吧&#xff0c;距离发的上个动态已经过去3个月了&#xff0c;真的觉得找实习时间来不及&#xff0c;现在leetcode 100多道题&#xff0c;前几天蓝桥杯整了个省二&#xff0c;把OS和…

利用大语言模型(KIMI)构建智能产品的控制信息模型

数字化的核心是数字化建模&#xff0c;为一个事物构建数字模型是一项十分复杂的工作。不同的应用场景&#xff0c;对事物的关注重点的不同的。例如&#xff0c;对于一个智能传感器而言&#xff0c;从商业的角度看&#xff0c;产品的信息模型中应该包括产品的类型&#xff0c;名…

Linux蛋疼笔记之无法安装软件

Ubuntu在安装软件时&#xff0c;一直安装失败&#xff0c;提示&#xff1a; dpkg: error processing package gconf2-common (--configre):installed gconf2-common package post-installation script susprocess returned error exit status 10 Error wre encountered while …

数字滤波器设计笔记1

系统结构 1.先利用matlab的simulink和FDA进行滤波器建模设计&#xff0c;通过仿真后&#xff0c;确定模型达到相应的性能要求&#xff0c;再利用verilog进行电路设计。最后使用modelsim进行功能验证。其中testbench的输入数据&#xff0c;利用matlab模型的输入数据。 2.Matlab…

基于openwrt和libssh2实现ssh的远程登录

libssh2的支持 openwrt本身带有libssh和libssh2两种三方库。我们需要修改编译使其参与编译./scripts/feeds update -a ./scripts/feeds install -a执行make menuconfig操作&#xff0c;勾选你想要的三方库 ssh服务器的连接 这里需要注意的主要就是makefile里面记得添加ssh2…

刷代码随想录有感(52):用数组最大值及其两侧构造最大二叉树

题干&#xff1a; 代码&#xff1a; class Solution { public:TreeNode* constructMaximumBinaryTree(vector<int>& nums) {if(nums.size() 1)return new TreeNode(nums[0]);int maxvalue INT_MIN;int maxindex;for(int i 0; i < nums.size(); i){if(nums[i] …

[C++][数据结构]红黑树的介绍和模拟实现

前言 之前我们简单学习了一下搜索树和平衡搜索树&#xff0c;今天我们来学习一下红黑树 简介 概念 红黑树&#xff0c;是一种二叉搜索树&#xff0c;但在每个结点上增加一个存储位表示结点的颜色&#xff0c;可以是Red或Black。 通过对任何一条从根到叶子的路径上各个结点着…

地图产业的困局与破局:高精地图“上车”难 轻量化渐成主流方案 | 最新快讯

《科创板日报》5月3日讯&#xff08;编辑 邱思雨&#xff09; 近期&#xff0c;特斯拉与百度的“绯闻”成为智驾、地图行业的焦点。 有媒体消息称&#xff0c;特斯拉将与百度地图独家深度定制车道级高辅地图。《科创板日报》记者也获悉&#xff0c;自5月1日起&#xff0c;百度…

pygame鼠标绘制

pygame鼠标绘制 Pygame鼠标绘制效果代码 Pygame Pygame是一个开源的Python库&#xff0c;专为电子游戏开发而设计。它建立在SDL&#xff08;Simple DirectMedia Layer&#xff09;的基础上&#xff0c;允许开发者使用Python这种高级语言来实时开发电子游戏&#xff0c;而无需被…

ESXi8 中FreeBSD启动失败记录

一天突然发现ESXi8 中的FreeBSD启动失败&#xff0c;且自动挂载了FreeBSD的安装光盘&#xff0c;进入安装步骤。 惊了一身冷汗。 到虚拟主机设置里&#xff0c;发现引导选项里面&#xff0c;选择应当用来引导虚拟机的固件为EFI&#xff0c;原来是前段时间手残修改了&#xff0…

图片倾斜矫正处理(Hough Transform)

目录 倾斜矫正原理及实现方式Canny边缘检测非极大值抑制霍夫变换 倾斜矫正原理及实现方式 代码连接&#xff1a;https://github.com/shuyeah2356/Image-Angel-correction/tree/main 倾斜矫正的实现原理&#xff1a; 使用霍夫变换检测图片中的直线&#xff1b; 计算直线与水平方…

TCP四次挥手分析

TCP四次挥手分析 概念过程分析为什么连接的时候是三次握手&#xff0c;关闭的时候却是四次握手&#xff1f;为什么要等待2MSL&#xff1f; 概念 四次挥手即终止TCP连接&#xff0c;就是指断开一个TCP连接时&#xff0c;需要客户端和服务端总共发送4个包以确认连接的断开。 在…

机器视觉系统-条形光源安装位置计算

使用条形光对反光材质物体打光时&#xff0c;常常出现强烈的光斑反射&#xff0c;影响图像处理。如果不想图像中出现光源的光斑&#xff0c;可以通过计 算得出条形光源的安装范围。 检则PCB板上的二维码字符&#xff0c;使用两个条形光打光的效果图 以及等效模型&#xff1a; …

机器学习:深入解析SVM的核心概念【二、对偶问题】

对偶问题 **问题一&#xff1a;什么叫做凸二次优化问题&#xff1f;而且为什么符合凸二次优化问题&#xff1f;**为什么约束条件也是凸的半空间&#xff08;Half-Space&#xff09;凸集&#xff08;Convex Set&#xff09;半空间是凸集的例子SVM 约束定义的半空间总结 **问题二…

领域驱动设计(DDD)笔记(三)后端工程架构

文章链接 领域驱动设计(DDD)笔记(一)基本概念-CSDN博客领域驱动设计(DDD)笔记(二)代码组织原则-CSDN博客领域驱动设计(DDD)笔记(三)后端工程架构-CSDN博客前导 领域驱动设计(Domain Driven Design,简称DDD)是业内主导的业务工程理论。它在各中权威人士被广泛讨论…

ArrayList应用

1.简单的洗牌算法 基本要求&#xff1a; 人数为3个人没人轮流抽一张牌&#xff0c;抽五轮&#xff0c;也就是每人五张牌去除大小王&#xff0c;一共52张牌&#xff0c;要求牌打乱顺序 思路&#xff1a; 创建Card对象&#xff0c;有花色和牌面值两个成员属性生成一副扑克牌&…

ICode国际青少年编程竞赛- Python-1级训练场-for循环练习

ICode国际青少年编程竞赛- Python-1级训练场-for循环练习 1、 for i in range(3):Dev.step(4)Dev.turnLeft()2、 for i in range(3):Dev.step(2)Dev.turnRight()Dev.step(2)Dev.turnLeft()3、 for i in range(3):Dev.step(2)Dev.turnRight()Dev.step(2)Dev.turnLeft()4、 for…