二分查找&删除数组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;
}
相关推荐
从以前3 分钟前
【算法题解】Bindian 山丘信号问题(E. Bindian Signaling)
开发语言·python·算法
一根稻草君6 分钟前
利用poi写一个工具类导出逐级合并的单元格的Excel(通用)
java·excel
不白兰6 分钟前
[代码随想录23回溯]回溯的组合问题+分割子串
算法
kirito学长-Java8 分钟前
springboot/ssm网上宠物店系统Java代码编写web宠物用品商城项目
java·spring boot·后端
木头没有瓜23 分钟前
ruoyi 请求参数类型不匹配,参数[giftId]要求类型为:‘java.lang.Long‘,但输入值为:‘orderGiftUnionList
android·java·okhttp
奋斗的老史23 分钟前
Spring Retry + Redis Watch实现高并发乐观锁
java·redis·spring
high201125 分钟前
【Java 基础】-- ArrayList 和 Linkedlist
java·开发语言
老马啸西风32 分钟前
NLP 中文拼写检测纠正论文 C-LLM Learn to CSC Errors Character by Character
java
御风@户外1 小时前
质数生成函数、质数判断备份
算法·acm
Cosmoshhhyyy1 小时前
LeetCode:3083. 字符串及其反转中是否存在同一子字符串(哈希 Java)
java·leetcode·哈希算法