解剖—顺序表相关OJ练习题

news2025/1/19 23:18:26

目录

一、删除有序数组中的重复项,返回出现一次元素的个数。

二、原地移除数组中所有数值等于val的元素

三、合并两个有序数组

四、旋转数组

五、数组形式的整数加法


一、删除有序数组中的重复项,返回出现一次元素的个数。

26. 删除有序数组中的重复项 - 力扣(LeetCode)

给你一个 非严格递增排列 的数组 nums ,请你 原地 删除重复出现的元素,使每个元素 只出现一次 ,返回删除后数组的新长度。元素的 相对顺序 应该保持 一致 。然后返回 nums 中唯一元素的个数。数组大小numsSize已给出。

考虑 nums 的唯一元素的数量为 k ,你需要做以下事情确保你的题解可以被通过:

  • 更改数组 nums ,使 nums 的前 k 个元素包含唯一元素,并按照它们最初在 nums 中出现的顺序排列。nums 的其余元素与 nums 的大小不重要。
  • 返回 k 。
int removeDuplicates(int* nums, int numsSize) {
    int src = 1;
    int dst = 0;
    while (src < numsSize) {
        if (nums[src] == nums[dst])
            src++;
        else
            nums[++dst] = nums[src++];
    }
    return dst + 1;
}

思路:采用双指针一前一后

1、src负责寻找与当前dst相等的值,dst负责修改数组。

2、如果src等于dst,则src向后移动一位,向后寻找不相同的值

3、如果src不等于dst,则dst向后移动一位,将src的值赋值给dst,src向后移动一位继续循环比较后项元素。

4、src指向数组最后一个元素时,比较后src大于数组大小numsSize时停止循环。

5、返回值为新数组的大小dst+1。

二、原地移除数组中所有数值等于val的元素

27. 移除元素 - 力扣(LeetCode)

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

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

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

int removeElement(int* nums, int numsSize, int val) {
    int src = 0;
    int dst = 0;
    while (src < numsSize) {
        if (nums[src] != val)
            nums[dst++] = nums[src++];
        else
            src++;
    }
    return dst;
}

思路:采用双指针

src负责寻找值为val的位置,dst负责修改数组。假设当前val等于7

1、如果src不等于val,则dst赋值为src,然后整体向后移动。

2、如果src等于val(dst也等于val),src向后移动一位。

3、再次判断时src不等于val,dst(当前为val)被赋值为src(val后一项)。成功删除数组元素val。

4、删除后dst和src均向后移动一位 。

5、之后在数组中依次查找,没有等于val的元素则将src赋值给dst(后项值一次赋值给前项),赋值后二者向后移动一位。

结果如下:

6、最终返回数组长度dst。

三、合并两个有序数组

88. 合并两个有序数组

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

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

注意:最终,合并后数组不应由函数返回,而是存储在数组 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]
void merge(int* nums1, int nums1Size, int m, int* nums2, int nums2Size, int n) {
    int end1 = m - 1;
    int end2 = n - 1;
    int i = m + n - 1;
    while (end1 >= 0 && end2 >= 0) {
        if (nums2[end2] > nums1[end1])
            nums1[i--] = nums2[end2--];
        else {
            nums1[i--] = nums1[end1--];
        }
    }
    while (end2 >= 0)
        nums1[i--] = nums2[end2--];
}

思路:三指针法

假设合并这两个数组:

1、定义 end1 和 end2 分别为数组 num1 和 num2 的最后一个元素下标,定义 i 为合并后num1新的最后一个元素下标.

2、因为是递增数列,所以我们从后向前先比较两个数组最后一个元素 (最大元素) 的大小。

大的元素赋值给数组num1尾节点,然后将 i、end1和end2向前移动一位继续比较。

3、如果end2位置元素不大于end1位置元素,则将end1位置元素赋值给 i 位置,end1和 i 向前移动一位。

4、最后end1已将小于零时,end2还等于0,证明数组num2中还有未合并的元素且该元素小于num1中最小元素(首元素)。

 

5、我们通过第二个循环将数组num2中小于num1最小元素的元素赋值到num1剩余位置,end2小于零时赋值结束。

 当然也可以使用qsot排序做这道题,但时间复杂度为O(nlogn)大于上述解法的O(m+n)。

 所以综合考虑推荐三指针法!!!

 四、旋转数组

189. 轮转数组 - 力扣(LeetCode)

给定一个整数数组 nums,将数组中的元素向右轮转 k 个位置,其中 k 是非负数。

示例 1:

输入: nums = [1,2,3,4,5,6,7], k = 3
输出: [5,6,7,1,2,3,4]
解释:
向右轮转 1 步: [7,1,2,3,4,5,6]
向右轮转 2 步: [6,7,1,2,3,4,5]
向右轮转 3 步: [5,6,7,1,2,3,4]
void reserve(int* a, int left, int right)
{
    while (left < right) {
        int tmp = a[left];
        a[left] = a[right];
        a[right] = tmp;
        left++;
        right--;
    }
}
void rotate(int* nums, int numsSize, int k) {
    if (k > numsSize)
        k %= numsSize;
    reserve(nums, numsSize - k, numsSize - 1);
    reserve(nums, 0, numsSize - k - 1);
    reserve(nums, 0, numsSize - 1);
}

思路:使用三次逆转法,让数组旋转k次

  • 逆转子数组[numsSize - k, numsSize - 1]
  • 逆转子数组[0, numsSize - k - 1]
  • 整体逆转[0,numsSize - 1]

假设轮转次数k为3 

1、首先写出逆转数组的函数reserve,逆置的原理就是依次交换首尾元素。

 2、然后找到要逆转的数组对应部分进行逆转。

3、 逆转过程如下:

五、数组形式的整数加法

 989. 数组形式的整数加法 - 力扣(LeetCode)

整数的 数组形式  num 是按照从左到右的顺序表示其数字的数组。

  • 例如,对于 num = 1321 ,数组形式是 [1,3,2,1] 。

给定 num ,整数的 数组形式 ,和整数 k ,返回 整数 num + k 的 数组形式 。

示例 1:

输入:num = [1,2,0,0], k = 34
输出:[1,2,3,4]
解释:1200 + 34 = 1234
int* addToArrayForm(int* A, int ASize, int K, int* returnSize) {
    int* addRet = (int*)malloc(10001 * sizeof(int));
    //reti: 第i位的结果
    int reti = 0;
    //从低位开始相加
    int ai = ASize - 1;
    int next = 0; // 进位值
    while (ai >= 0 || K > 0)
    {

        int x1 = 0;
        //如果ai没有越界,还有未相加的位,取出一位存入x1
        if (ai >= 0)
        {
            x1 = A[ai];
            --ai;
        }

        int x2 = 0;
        //如果k大于0,获取k的第i位
        if (K > 0)
        {
            x2 = K % 10;
            K /= 10;
        }
        //第i位的结果:每一位的值 + 进位
        int ret = x1 + x2 + next;
        //如果结果大于9,需要进位
        if (ret > 9)
        {
            ret %= 10;
            next = 1;
        }
        else
        {
            next = 0;
        }
        //存入第i位的结果到数组中
        addRet[reti++] = ret;
    }
    //如果最高位有进位,需要在存入1
    if (next == 1)
    {
        addRet[reti++] = 1;
    }
    //逆置结果
    reverse(addRet, 0, reti - 1);
    *returnSize = reti;

    return addRet;
}

 思路:模拟加法进行逐位相加, 从低位向高位相加,最后整体逆置,得到最终结果

1、定义一个整数数组 addRet 用于存储相加的结果。数组的大小设为10001,足够容纳可能的最大结果。

定义变量 reti,表示结果数组的当前索引,初始化为0。

定义变量 ai,表示数组 A 的当前索引,初始化为 ASize - 1,即数组 A 的最高索引。

定义变量 next,表示进位值,初始化为0。

int* addRet = (int*)malloc(10001 * sizeof(int));
int reti = 0;
int ai = ASize - 1;
int next = 0; 

2、使用循环处理每一位的相加,循环条件是 ai >= 0 或 K > 0,即数组 A 还有位数未相加,或者 K 还有位数未相加。

while (ai >= 0 || K > 0)

3、在循环内部,首先获取 A 数组当前位的值 x1,如果 ai 没有越界,就取出对应位置的值,并将 ai 减1。

int x1 = 0;
if (ai >= 0)
{
    x1 = A[ai];
    --ai;
}

4、获取整数 K 的当前位值 x2,如果 K 大于0,就取出 K 的最低位,同时将 K 除以10,以准备处理下一位。

 int x2 = 0;
 if (K > 0)
 {
     x2 = K % 10;
     K /= 10;
 }

5、计算当前位的结果 ret,是 x1x2 和之前的进位 next 三者之和。如果结果大于9,表示需要进位,就对结果取模10,然后将 next 设置为1;否则,next 设置为0。

int ret = x1 + x2 + next;
if (ret > 9)
{
    ret %= 10;
    next = 1;
}
else
{
    next = 0;
}

7、将计算得到的 ret 存入结果数组 addRet 的当前位置 reti,递增 reti,以准备处理下一位。

addRet[reti++] = ret;

8、循环结束后,检查最高位是否有进位。如果 next 为1,表示有进位,因此将1存入结果数组的当前位置 reti

 if (next == 1)
 {
     addRet[reti++] = 1;
 }

9、调用 reverse 函数来逆置结果数组 addRet,以得到正确的结果顺序。

reverse(addRet, 0, reti - 1);

最后,将结果数组的大小 reti 赋值给 returnSize 指向的变量,以告知结果数组的大小。

返回结果数组 addRet,它包含了相加的结果。

*returnSize = reti;
return addRet;

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

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

相关文章

Service Mesh和Kubernetes:加强微服务的通信与安全性

文章目录 什么是Service Mesh&#xff1f;Service Mesh的优势1. 流量控制2. 安全性3. 可观测性 Istio&#xff1a;Service Mesh的领军者流量管理安全性可观测性 Linkerd&#xff1a;轻量级Service Mesh流量管理安全性可观测性 Istio vs. Linkerd实际应用结论 &#x1f388;个人…

vscode中4个json的区别和联系

在vscode中快捷键ctrlshiftp&#xff0c;然后输入setting&#xff0c;会出现下图几个选项 当不同设置之间出现冲突时&#xff0c;听谁的&#xff1a; Open Workspace Settings(JSON) > Open Settings(JSON) Open User Settings > Open Default Settings(JSON) Open Wo…

openstack 云主机 linux报 login incorrect

还未输入密码就提示login incorrect 不给输密码位置 完全不给输密码的机会 关机进入单用户 检查登录安全记录 vi /var/log/secure 发现 /usr/lib64/security/pam_unix.so 报错 将正常的机器提取/usr/lib64/security/pam_unix.so 比对MD5一致&#xff0c; 另外判断 libtir…

车载开发学习——CAN总线

CAN总线又称为汽车总线&#xff0c;全程为“控制器局域网&#xff08;Controller Area Network&#xff09;”&#xff0c;即区域网络控制器&#xff0c;它将区域内的单一控制单元以某种形式连接在一起&#xff0c;形成一个系统。在这个系统内&#xff0c;大家以一种大家都认可…

市值缩水90%以上,泛生子何以败退美股?

癌症是人类面临的最大健康威胁之一&#xff0c;也是医学界最难攻克的难题之一。随着科技的发展&#xff0c;癌症精准医疗逐渐成为治疗癌症的新方向&#xff0c;癌症精准医疗能通过对癌细胞的基因检测和分析&#xff0c;为患者提供个性化的治疗方案。然而&#xff0c;这一领域的…

redis(其它操作、管道)、django中使用redis(通用方案、 第三方模块)、django缓存、celery介绍(celery的快速使用)

1 redis其它操作 2 redis管道 3 django中使用redis 3.1 通用方案 3.2 第三方模块 4 django缓存 5 celery介绍 5.1 celery的快速使用 1 redis其它操作 delete(*names) exists(name) keys(pattern*) expire(name ,time) rename(src, dst) move(name, db)) randomkey() type(na…

VBA技术资料MF71:查找所有空格并替换为固定字符

我给VBA的定义&#xff1a;VBA是个人小型自动化处理的有效工具。利用好了&#xff0c;可以大大提高自己的工作效率&#xff0c;而且可以提高数据的准确度。我的教程一共九套&#xff0c;分为初级、中级、高级三大部分。是对VBA的系统讲解&#xff0c;从简单的入门&#xff0c;到…

实现实时美颜:主播直播美颜SDK的技术细节

在今天的数字时代&#xff0c;直播和实时互动成为了日常生活的一部分&#xff0c;而主播直播美颜SDK的出现为用户提供了更加精美的视觉体验。这项技术的背后有着复杂的技术细节&#xff0c;从图像处理到机器学习&#xff0c;本文将深入探讨主播直播美颜SDK的技术细节&#xff0…

四边形不等式

区间dp问题&#xff0c;状态转移方程&#xff1a; dp[i][j] min( dp[i][k] dp[k1][j] w[i][j] ) //w[i][j]是从i到j的&#xff0c;一个定值 不随k改变&#xff0c;而且w的值只和i j有关&#xff0c;是它们的二元函数。 其中i<k<j ,初始值dp[i][i]已知。 含义&#x…

第三类医疗器械经营许可证经营范围

在我国&#xff0c;医疗器械监督管理条例规定:医械经营企业要依据主营产品办理相应许可证。医疗器械根据其风险性又分为三类&#xff0c;一类医疗器械实行产品备案管理&#xff0c;第二类、第三类医疗器械实行产品注册管理&#xff0c;经营第二类、第三类医疗器械应当持有《医疗…

Day 2 Qt

#include "my_widget.h" #include "ui_my_widget.h"My_Widget::My_Widget(QWidget *parent): QWidget(parent), ui(new Ui::My_Widget) {ui->setupUi(this);//窗口的相关设置 // this -> resize(800,500);this -> setWindowTitle("QQ聊天…

APP备案公钥、证书MD5指纹/签名MD5值获取方法

本文只详细讲解android app获取方法&#xff0c;三种方式&#xff1a; 1. 你的应用已安装到手机&#xff0c;android应用市场搜索下载安装 APP备案助手&#xff0c;此app可直接获取所有已安装app的公钥、证书MD5指纹/签名MD5值&#xff0c;示例&#xff1a;获取 抖音app公钥、…

每日一练 | 华为认证真题练习Day120

1、MPLS域中的LER全称为Label Egress Router。 A. 对 B. 错 2、如果一个以太网数据帧的Type/Length字段的值为0x0800&#xff0c;则此数据帧所承载的上层报文首部长度范围为20-60B。 A. 对 B. 错 3、在VRP平台上&#xff0c;可以通过下面哪种方式访问上一条历史命令&#x…

AI爆文变现-写作项目-脚本配置教程-解放双手

之前给大家分享过AI爆文的写作教程&#xff0c;没看过的可以看下对应的教程&#xff1a; AI爆文撸流量主保姆级教程2.0 因为是怼量&#xff0c;为了高效完成文章&#xff0c;我用python脚本实现了自动写文章的功能&#xff0c;发布文章目前还是要手动进行。 AI爆文教程参考&…

C++11——包装器与lambda表达式

目录 一.背景 二.lambda 1.见一见lambda 2.lambda表达式语法 3.lambda捕捉列表说明 三.函数对象与lambda表达式 四.包装器 1.function包装器 2.包装类的成员函数 五.bind 1.调整参数位置 2.减少函数参数 一.背景 在C98中&#xff0c;如果想要对一个数据集合中的元素…

c++_learning-模板元编程

模板元编程 元编程&#xff1a;元函数&#xff1a;定义&#xff1a;数值元函数&#xff1a;编译期间&#xff0c;能够被调用的类模板&#xff1a;constexpr修饰的函数&#xff1a;constexpr修饰的变量模板&#xff1a; 类型元函数&#xff1a;元函数总结&#xff1a; 混合元编程…

C++基础——内存分区模型

1 概述 C程序在执行是&#xff0c;将内存大致分为4个区域&#xff1a; 代码区&#xff1a;用于存放二进制代码&#xff0c;由操作系统进行管理全局区&#xff1a;存放全局变量和静态变量及常量栈区&#xff1a;由编译器自动分配释放&#xff0c;存放函数的参数、局部变量等堆…

电脑版便签软件下载用哪个?

在面对每天繁忙的工作日程&#xff0c;电脑是许多上班族不可或缺的工作助手&#xff0c;而一款得心应手的电脑便签软件&#xff0c;更是可以帮助大家记录、提醒、督促各项任务按时完成的得力助手。那么&#xff0c;究竟在众多的电脑便签软。件中&#xff0c;哪一位能够真正成为…

你真的理解“感受野”了吗?

大家好啊&#xff0c;我是董董灿。 在卷积神经网络中&#xff0c;有一个非常重要且绕不开的概念叫做卷积的“感受野”。 我刚开始学习卷积的时候&#xff0c;仅仅把“感受野”理解为感受视野&#xff0c;没错就是文字上的理解。 但随着后来工作的深入&#xff0c;慢慢发现感…

pytorch的安装【全官网流程】

1.准备python环境 python环境需要看pytorch上说明的版本本文用的是python3.9 conda create -n pytorch39 python3.92.安装pytorch【要使用GPU的先安装步骤3的CUDA在安装这个】 pytorch官方地址 &#xff08;1&#xff09;官方指出了python版本&#xff1a; &#xff08;2…