二分查找&删除数组target元素

二分查找

链接: 704.二分查找

思考过程:

复制代码
// 思路是二分法
// 看着有点熟悉,却一时想不起怎么解
// 有点忘记二分的使用条件:有序加不重复
// 循环方式实现查找
// 递归方式怎么解?

解题代码

c++ 复制代码
int LCFuntion::search(vector<int> &data, int target)
{
    int left = 0, right = data.size() - 1;
    while (left <= right) {
        int mid = (right - left) / 2 + left;
        if (data[mid] == target) {
            return mid;
        }
        else if (data[mid] > target) {
            right = mid - 1;
        }
        else if (data[mid] < target) {
            left = mid + 1;
        }
    }
    return -1;
}

要点总结:注意区分区间的左右开闭处理

移除数组内target元素

链接: 27. 移除元素

// 移除数组内target元素,且保持剩下元素连续

// 想法是遍历+移动元素

// 数组的下标写错了data.size(),导致数组越绝访问崩溃

// 函数崩溃,猜测是数组越界问题,但是没发现问题在哪?

// 在不同编译器下,for (auto i = data.size() - 1; i >= 0; i--)

// 中的i 被推导为不同类型,mingw下是未定义类型,msvc下是unsigned __int64

// 它们都会导致越界问题,因为是无符号类型,永远不可能小于0,所以不能滥用auto

// 时间复杂度O(n^2)

解题代码如下:

c++ 复制代码
int LCFuntion::removeElement(vector<int> &data, int target)
{
    int moveSize = data.size();
    for (int i = data.size() - 1; i >= 0; i--) {
        //        for (auto i = data.size() - 1; i >= 0; i--) {
        qDebug() << "index" << i << typeid(i).name();
        if (data[i] == target) {
            // 后面元素前移
            int j = i;
            while (j < moveSize - 1) {
                data[j] = data[j + 1];
                j++;
            }
            moveSize--;
        }
    }
    return moveSize;
}

我的解法是暴力遍历移动元素,时间复杂度较高

经过优化,可以做到O(n)的时间复杂度

优化后的代码:

C++ 复制代码
// 双指针的优化版本
// 使用交换的技术来实现
// 当双指针相遇,就代表已经移动到后面了
// 但是改变了数组元素相对位置
// 里循环的 left <= right 条件判断要在前面 不然可能会越界
// 边界条件处理不到位 是否要 <=
int LCFuntion::removeElement2(vector<int> &nums, int val)
{
    int right = nums.size() - 1;
    int left  = 0;
    while (left <= right) {
        // find val
        while (left <= right && nums[left] != val) left++;
        // find not val
        while (left <= right && nums[right] == val) right--;
        // swap
        if (left < right) {
            std::swap(nums[left++], nums[right--]);
        }
    }
    return left;
}
相关推荐
仰泳的熊猫33 分钟前
题目2308:蓝桥杯2019年第十届省赛真题-旋转
数据结构·c++·算法·蓝桥杯
独自破碎E33 分钟前
前后端分离+微服务架构下的用户认证
java·面试·架构
hssfscv42 分钟前
力扣练习训练2(java)——二叉树的中序遍历、对称二叉树、二叉树的最大深度、买卖股票的最佳时机
java·数据结构·算法
Byron__1 小时前
HashMap面试知识点
java·面试·hash
诺浅1 小时前
聊聊@DSTransactional的坑
java·多数据源·dstransavtional
菜鸟‍1 小时前
【后端项目】苍穹外卖day01-开发环境搭建
java·开发语言·spring boot
lzksword1 小时前
C++ Builder XE OpenDialog1打开多文件并显示xls与xlsx二种格式文件
java·前端·c++
青槿吖1 小时前
【保姆级教程】Spring事务控制通关指南:XML+注解双版本,避坑指南全奉上
xml·java·开发语言·数据库·sql·spring·mybatis
y = xⁿ2 小时前
【LeetCodehot100】二叉树大合集 T94:二叉树的中序遍历 T104:二叉树的最大深度 T226:翻转二叉树 T101:对称二叉树
后端·算法·深度优先
不想看见4042 小时前
Search a 2D Matrix II数组--力扣101算法题解笔记
数据结构·算法