代码随想录算法训练营Day1 —— 704. 二分查找、27.移除元素

理论部分:

数组理论基础:

C++中的部分容器的底层原理仍然是数组,数组是存放在连续内存空间上的相同类型数据的集合

一维数组:

  • 数组的下标从0开始。
  • 内存空间地址连续,当删除、添加元素时需要对其他元素的地址进行更改。
  • 不能删除只能覆盖。

二维数组:

二维数组也是连续分布的,先行后列

常用函数方法(慢慢总结):

//获取数组长度
len = nums.size();
//获取数组的头尾指针
nums.begin();
nums.end();
//排序指针(默认从小到大)
sort(nums.begin(),nums.end());

算法部分:

704.二分查找:

重点在于对于区间开闭的理解一致性,常参照左闭右闭、左闭右开

左闭右闭:

左闭右开:

代码部分:

//左闭右闭
class Solution {
public:
    int search(vector<int>& nums, int target) {
         int left=0;
         int right=nums.size()-1;
         while(left<=rihgt){
             int middle = (left+right)/2;
             if (nums[middle]>target){
                 right = middle - 1;
             }
             else if(nums[middle]<target){
                 left = middle + 1;
             }
             else {
             return middle;
             }
         }
         return -1;
    }
};

//左闭右开
class Solution {
public:
    int search(vector<int>& nums, int target) {
         int left=0;
         int right=nums.size();
         while(left<right){
             int middle = (left+right)/2;
             if (nums[middle]>target){
                 right = middle ;
             }
             else if(nums[middle]<target){
                 left = middle + 1;
             }
             else {
             return middle;
             }
         }
         return -1;
    }
};

题目反思:

我是典型的一听就会一写就费,之前对于<=的符号没有清晰的认知,导致很多基本都是蒙的,以后需要注意区分类型,照着思路多写总没错!

27.移除元素:

暴力搜索(时间复杂度:O(n^2)、空间复杂度:O(1)):

第一次自己直接写的代码不能正常运行不知道什么鬼...

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

看完注释再决定重写一次(还是不能正常运行):

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

报错是越界了,分析原因是因为nums[j+1]中的j取值范围是<size,导致size+1对于数组而言越界,修改好后完美运行!!!(我真牛逼)

以下是暴力搜索的正解:

//自己改正后的暴力搜索
class Solution {
public:
    int removeElement(vector<int>& nums, int val) {
        int size = nums.size();
        for (int i = 0 ; i < size ; i++){
            if(nums[i]==val){
                for(int j = i ; j < size-1 ; j++){
                    nums[j]=nums[j+1];
                }
                i--;
                size--;
            }
        }
        return size;
    }
};

//卡子哥自带暴力搜索
class Solution {
public:
    int removeElement(vector<int>& nums, int val) {
        int size = nums.size();
        for (int i = 0 ; i < size ; i++){
            if(nums[i]==val){
                for(int j = i + 1 ; j < size ; j++){
                    nums[j-1]=nums[j];
                }
                i--;
                size--;
            }
        }
        return size;
    }
};

双指针法(时间复杂度:O(n)、空间复杂度:O(1)):

双指针法(快慢指针法): 该方法并不改变元素相对位置,通过一个快指针和慢指针在一个for循环下完成两个for循环的工作。

定义快慢指针

  • 快指针:寻找新数组的元素 ,新数组就是不含有目标元素的数组
  • 慢指针:指向更新 新数组下标的位置
//同向指针,核心思想是将不相等的值存储,相同的值不赋值
class Solution {
public:
    int removeElement(vector<int>& nums, int val) {
        int slowIndex = 0;
        for(int longIndex=0 ; longIndex < nums.size() ; longIndex++){
            if (nums[longIndex] != val){
                nums[slowIndex]=nums[longIndex];
                slowIndex++;
            }
        }
        return slowIndex;
    }
};

//相向双指针方法,基于元素顺序可以改变的题目描述改变了元素相对位置,确保了移动最少元素(参考)
class Solution {
public:
    int removeElement(vector<int>& nums, int val) {
        int leftIndex = 0;
        int rightIndex = nums.size() - 1;
        while (leftIndex <= rightIndex) {
            // 找左边等于val的元素
            while (leftIndex <= rightIndex && nums[leftIndex] != val){
                ++leftIndex;
            }
            // 找右边不等于val的元素
            while (leftIndex <= rightIndex && nums[rightIndex] == val) {
                -- rightIndex;
            }
            // 将右边不等于val的元素覆盖左边等于val的元素
            if (leftIndex < rightIndex) {
                nums[leftIndex++] = nums[rightIndex--];
            }
        }
        return leftIndex;   // leftIndex一定指向了最终数组末尾的下一个元素
    }
};

题目反思:忘记了数组不能直接删除某个值,只能覆盖!!!因为数组是连续内存的!!!同时要注意数组索引不要越界,今天收获很多,明天继续加油!!!

  • 5
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
代码随想录算法训练营是一个优质的学习和讨论平台,提供了丰富的算法训练内容和讨论交流机会。在训练营中,学员们可以通过观看视频讲解来学习算法知识,并根据讲解内容进行刷题练习。此外,训练营还提供了刷题建议,例如先看视频、了解自己所使用的编程语言、使用日志等方法来提高刷题效果和语言掌握程度。 训练营中的讨论内容非常丰富,涵盖了各种算法知识点和解题方法。例如,在第14天的训练营中,讲解了二叉树的理论基础、递归遍历、迭代遍历和统一遍历的内容。此外,在讨论中还分享了相关的博客文章和配图,帮助学员更好地理解和掌握二叉树的遍历方法。 训练营还提供了每日的讨论知识点,例如在第15天的讨论中,介绍了层序遍历的方法和使用队列来模拟一层一层遍历的效果。在第16天的讨论中,重点讨论了如何进行调试(debug)的方法,认为掌握调试技巧可以帮助学员更好地解决问题和写出正确的算法代码。 总之,代码随想录算法训练营是一个提供优质学习和讨论环境的平台,可以帮助学员系统地学习算法知识,并提供了丰富的讨论内容和刷题建议来提高算法编程能力。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [代码随想录算法训练营每日精华](https://blog.csdn.net/weixin_38556197/article/details/128462133)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值