首页 > 编程语言 >【算法与数据结构】【数组篇】【题1-题5】

【算法与数据结构】【数组篇】【题1-题5】

时间:2024-06-11 10:33:02浏览次数:27  
标签:下标 matrix nums int 算法 intervals 数组 数据结构

1.数组基本知识点

1.1概念

数组就是一个集合。数组会用一些名为索引的数字来标识每项数据在数组中的位置,且在大多数编程语言中,索引是从 0 算起的。我们可以根据数组中的索引,快速访问数组中的元素。

数组中的元素在内存中是连续存储的,且每个元素占用相同大小的内存。

1.2 相关操作的时间复杂度和空间复杂度

访问元素时间复杂度都是O(1),空间复杂度O(1),因为对于固定大小的数组,访问时间不随数组大小而变化。通过下标可直接访问。

修改元素:时间复杂度O(1),空间复杂度O(1),与n无关,通过下标可直接修改

插入元素和删除元素:时间复杂度O(1),空间复杂度O(1),插入和删除元素都需要移动后面的元素,因此随n变化。

题1:寻找数组的中心索引

给你一个整数数组 nums ,请计算数组的 中心下标 。

数组中心下标是数组的一个下标,其左侧所有元素相加的和等于右侧所有元素相加的和。

如果中心下标位于数组最左端,那么左侧数之和视为 0 ,因为在下标的左侧不存在元素。这一点对于中心下标位于数组最右端同样适用。

如果数组有多个中心下标,应该返回 最靠近左边 的那一个。如果数组不存在中心下标,返回 -1 。

示例 1:

输入:nums = [1, 7, 3, 6, 5, 6]
输出:3
解释:
中心下标是 3 。
左侧数之和 sum = nums[0] + nums[1] + nums[2] = 1 + 7 + 3 = 11 ,
右侧数之和 sum = nums[4] + nums[5] = 5 + 6 = 11 ,二者相等。

示例 2:

输入:nums = [1, 2, 3]
输出:-1
解释:
数组中不存在满足此条件的中心下标。

示例 3:
输入:nums = [2, 1, -1]
输出:0
解释:
中心下标是 0 。
左侧数之和 sum = 0 ,(下标 0 左侧不存在元素),
右侧数之和 sum = nums[1] + nums[2] = 1 + -1 = 0 。

解答思路:

第一步:先用for循环求总和
第二步:当位置为i时,找到其左边所有元素的合。
第三步:然后用总和减去其左边所有元素的合再减去当前元素的值。就是右边元素的值。
第四步:判断其左边元素和和右边元素和是否相等。
第五步:如果相等,则返回i,如果不相等,循环判断下一个元素。
第六步:如果最后一个元素仍然不满足,返回-1

代码案例:

class Solution
{
public:
    int pivotIndex(vector<int> &nums)
    {
        int size = nums.size();
        if (size <= 0)
        {
            return 0;
        }

        int sum = 0;

        for (int &i : nums)
        {
            sum = sum + i; //第一步:先用for循环求总和
        }

        int leftsum = 0;

        for (int i = 0; i < size; i++)
        {

            if (leftsum == (sum -leftsum - nums[i]))//第二步和第三步和第四步
            {
                return i; //第五步
            }

            leftsum += nums[i];//第五步
        }

        return -1;//第六步
    }
};

题2:搜索插入位置

给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。

请必须使用时间复杂度为 O(log n) 的算法。

示例 1:

输入: nums = [1,3,5,6], target = 5
输出: 2

示例 2:

输入: nums = [1,3,5,6], target = 2
输出: 1

示例 3:

输入: nums = [1,3,5,6], target = 7
输出: 4

 解题思路:

思路:
第一步:此处特定说明了必须使用时间复杂度为 O(log n) 的算法。所以采用二分查找法。首先二分算法适用于已经排序过的数组。
第二步:二分法需要每次找到中间值,因此需要定义left,right,middle三个变量。
第三步:left=0,right = nums.size() - 1; middle = left + (right - left) / 2;
第四步:通过while循环进行查找
第五步:如果目标值刚好等于中间值,即target = nums[middle],则return middle;
第六步:如果目标值大于中间值,则说明其应该在右区间,因此需要挪动左边界值(left)为middle+1。
        如果目标值小于中间值,则说明其应该在左区间,因此需要挪动右边界值(right)为middle-1。
第七步:如果最后左边界和右边界相等,此时会退出whlie循环,此时说明区间只有一个值了,判断其是否等于此值,如果大于等于则return left+1;就是新插入的位置。
        如果此时目标值小于此值,则返回left的位置,要插入当前这个值的位置。

代码案例:

class Solution
{
public:
    int searchInsert(vector<int> &nums, int target)
    {
        if (nums.size() <= 0)
        {
            return -1;
        }

        int left = 0;
        int right = nums.size() - 1;
        int middle = left + (right - left) / 2;
        while (left < right)
        {
            middle = left + (right - left) / 2;
            if (target == nums[middle])
            {
                return middle;
            }
            else if (target < nums[middle])
            {
                right = middle - 1;
            }
            else if (target > nums[middle])
            {
                left = middle + 1;
            }
        }

        if (target >= nums[left])
        {
            return left + 1;
        }
        else if (target < nums[left])
        {
            return left;
        }
        else
        {
            return -1;
        }
    }
};

 题3:合并区间

以数组 intervals 表示若干个区间的集合,其中单个区间为 intervals[i] = [starti, endi] 。请你合并所有重叠的区间,并返回 一个不重叠的区间数组,该数组需恰好覆盖输入中的所有区间 。

示例 1:

输入:intervals = [[1,3],[2,6],[8,10],[15,18]]
输出:[[1,6],[8,10],[15,18]]
解释:区间 [1,3] 和 [2,6] 重叠, 将它们合并为 [1,6].

示例 2:

输入:intervals = [[1,4],[4,5]]
输出:[[1,5]]
解释:区间 [1,4] 和 [4,5] 可被视为重叠区间。

解题思路:

思路:


一般需要后一个比前一个小的题,首先先思考排序


第一步:排序,确保第一个start小于第二个start
第二步:如果intervals[i].end >= intervals[i+1].start,则将其组合,选择两个end值中最大的那个
        如果intervals[i].end < intervals[i+1].start,则无法合并,然后循环查找下一个区间和上一个区间再次进行循环对比

代码案例:

class Solution
{
public:
    vector<vector<int>> merge(vector<vector<int>> &intervals)
    {

        if (intervals.size() == 1)
        {
            return intervals;
        }

        sort(intervals.begin(), intervals.end()); // 第一步:排序,确保第一个start小于第二个start

        vector<vector<int>> returnvector;
        returnvector.push_back(intervals[0]);//第二步:先放入第一个值,以此为判断
        int j = 0;

        for (int i = 1; i < intervals.size(); i++)
        {
            if (returnvector[j][1] >= intervals[i][0])//第三步:如果第一个值的end大于第二个值的start,则合并
            {

                returnvector[j][1] = max(returnvector[j][1], intervals[i][1]);//选择第一个值的end和第二个值end中最大的那个
            }
            else //否则,说明没有公共区间,则放入第二个
            {
                returnvector.push_back(intervals[i]);//只有合并完成,才会往后判断,否则一直基于此子数组进行合并
                j++;
            }
        }

        return returnvector;
    }
};

题4:旋转矩阵

给你一幅由 N × N 矩阵表示的图像,其中每个像素的大小为 4 字节。请你设计一种算法,将图像旋转 90 度。

不占用额外内存空间能否做到?

示例 1:

给定 matrix =
[
  [1,2,3],
  [4,5,6],
  [7,8,9]
],

原地旋转输入矩阵,使其变为:
[
  [7,4,1],
  [8,5,2],
  [9,6,3]
]

示例 2:

给定 matrix =
[
  [ 5, 1, 9,11],
  [ 2, 4, 8,10],
  [13, 3, 6, 7],
  [15,14,12,16]
],

原地旋转输入矩阵,使其变为:
[
  [15,13, 2, 5],
  [14, 3, 4, 1],
  [12, 6, 8, 9],
  [16, 7,10,11]
]

思路:

这种题主打一个蛋疼。记住规律即可。

如果是顺时针旋转二维矩阵,则先对角翻转,再水平翻转。

如果是逆时针旋转二维矩阵,则先水平翻转,再对角翻转。

代码案例:

class Solution
{
public:
    void rotate(vector<vector<int>> &matrix)
    {
        int N = matrix.size();
        // 第一步:先对角线反转
        for (int i = 0; i < N; i++)
        {
            for (int j = i; j < N; j++)
            {
                swap(matrix[i][j], matrix[j][i]);
            }
        }

        // 第二步:水平翻转
        for (int i = 0; i < N; i++)
        {
            for (int j = 0; j < N / 2; j++)
            {
                swap(matrix[i][j], matrix[i][N - 1 - j]);
            }
        }
    }
};

题5:零矩阵

编写一种算法,若M × N矩阵中某个元素为0,则将其所在的行与列清零。

示例 1:

输入:
[
  [1,1,1],
  [1,0,1],
  [1,1,1]
 
]
输出:
[
  [1,0,1],
  [0,0,0],
  [1,0,1]
]

思路:

第一步:遍历二维数组,找出为0的x下标和y下标,并保存起来。

第二步:然后两次循环将其他位置清0

代码案例:

class Solution
{
public:
    void setZeroes(vector<vector<int>> &matrix)
    {
        int M = matrix.size();
        int N = matrix[0].size();

        vector<int> x;
        vector<int> y;

        for (int i = 0; i < M; i++)
        {
            for (int j = 0; j < N; j++)
            {
                if (matrix[i][j] == 0)
                {
                    x.push_back(i);
                    y.push_back(j);
                }
            }
        }

        for (auto &i : x)
        {
            for (int j = 0; j < N; j++)
            {
                matrix[i][j] = 0;
            }
        }

        for (auto &i : y)
        {
            for (int row = 0; row < M; row++)
            {
                matrix[row][i] = 0;
            }
        }
    }
};

标签:下标,matrix,nums,int,算法,intervals,数组,数据结构
From: https://blog.csdn.net/handsomethefirst/article/details/139509628

相关文章

  • Python数据结构——序列(超详细版)
    在计算机程序中会有很多数据,使用数据结构可以管理这些数据,Python中的数据结构主要有序列、集合和字典。常见的数据结构有:数组(array)、集合(set)、列表(list)、队列(queue)、链表(linkedlist)、树(tree)、堆(heap)、栈(stack)和字典(dictionary)。注意:Python中并没有数组结构,因为数组要求元素类......
  • LeetCode 算法:缺失的第一个正数c++
    原题链接......
  • 【Python核心数据结构探秘】:元组与字典的完美协奏曲
    文章目录......
  • 保研复习——数据结构篇
    前言:本笔记是对《王道数据结构》中各章节涉及的基础知识进行整理。本笔记主要用以应对夏令营面试中可能会问到的数据结构方面的问题,比较泛泛而谈,如果您对这些内容感兴趣,建议参考原书。大佬可自行绕路材料来源:王道2025年《数据结构考研复习指导》https://pan.baidu.com/s......
  • 【算法学习】圆方树——处理仙人掌的利器
    圆方树大概分两种,一个是圆方树,一个是广义圆方树。圆方树这可以解决仙人掌上的问题。任意一条边至多只出现在一条简单回路的无向连通图称为仙人掌。很多题解将其作为无向图构建,本文将其构建为外向树,在这个问题中两种构建方式不会影响求解。构建方式记读入的图为原图,构建的......
  • 二叉树相关算法题汇总-go语言实现
    总结先序中序后序遍历就能解决一些算法题。层次遍历使用队列。从左子树、右子树获取答案,然后结合根节点来计算答案。前缀树,比Hashset更稳定。O(1),只不过占内存。trie树。递归。递归,递归。到叶子节点收集答案。然后移除路径。packagemainimport( "fmt" "math......
  • 二维和多维数组概念
    目录1、二维数组2、多维数组1、二维数组在C语言中,当你定义一个二维数组时,你需要指定两个维度:行数和列数。二维数组就是在一维的数组上再嵌套了一层数组上去,使数组获得一种矩阵的假象。    inta[3][4]int是数组中元素的数据类型a 是数组的名称,即标识符。3是数......
  • 程序设计与算法(三)C++:第四章poj代码
    课程:北京大学程序设计与算法(三)   MOOCOJ:OpenJudge014:MyString这个题需要写的函数有点多我们来分析一下。charw1[200],w2[100]; while(cin>>w1>>w2){ MyStrings1(w1),s2=s1;//构造函数题目有了,不用写//复制构造函数没有,需要写 MyStrings3......
  • 代码随想录算法训练营第三十五天 | 1005.K次取反后最大化的数组和 134.加油站 135.分
    1005.K次取反后最大化的数组和题目链接文章讲解视频讲解思路:  按绝对值从大到小排序  遍历数组,遇到负数,如果次数未用完就取反  最后如果剩余次数未用完且为奇数就将数组最后一个元素取反classSolution{staticboolmyCompare(constint&lhs,constint&r......
  • 算法
    背包问题#include<cstdio>#include<cstring>usingnamespacestd;intT,n,sum,w[205],lim;//w[i]:物品i的价值booldp[20005];intmain(){while(1==scanf("%d",&T)){while(T-->0){sum=0;scanf(&q......