算法练习day1

前言

大二开始定下实践项目的主方向后,对于数据结构、算法这一块基本就是停滞状态了。

选择游戏方向后,unity和C#的实际项目对于算法方面的要求并不多。实习需要去大厂,并且为了进一步提高自己的能力,决定接下来的日子里苦练算法。

以前练习算法这方面,用的都是C、Java这类。用C#写还是头一次,在练习算法的同时,增进自己对C#的使用熟练度,也是一举两得。

接下来的日子,打算跟着卡哥的《算法随想录》刷题。根据http://t.csdnimg.cn/l8b7i,这篇大哥的博客,每天进行练习。自己练习确实得要很大的毅力和决心,希望自己能够坚持下来,共勉~

二分查找

代码随想录 二分查找

力扣 704.二分查找

(用时:1.5小时)

写的过程中出现了错误,思考了10分钟无法解决后,找了参考代码进行了纠错。错误如下:

  • 错误一:target < arr[mid]写成了target < mid

  • 错误二:写成了递归(也不算是错误,递归也能实现二分查找。但写的混合了递归和非递归,导致写的非常混乱!这也是对递归没有很好的理解。后来理清非递归后,重新写了下递归,一起放在后面了)

  • 错误三:左闭右开时,right = mid写成了right = mid-1

错误复盘

错误一分析:mid是一个下标,在target是一个值,比较的时候应是拿值比较。这里是对mid和target的关系没有更加深刻的理解,当查找到时,两者关系:target=arr[mid]

错误二分析:写成了递归这个错误就emmmm,其实是可以写成递归的。递归之前也一直是迈不过去的坎,导致这里先入为主了。递归是写出来了,但并不是题目格式的要求。。每天的时间有限,第一遍刷更侧重于复习回顾理解,第二遍再拓展深究。

错误三分析:

  • 卡哥视频二分查找有两个重点,一是while的判断条件,二是if里的判断条件。

  • while的判断条件看开闭区间,看left=right是否能取即可。

  • if判断条件也是看开闭区间,这里是看left是否能等于区间左侧、right是否能等于区间右侧

    例如,假设 区间为[i,j) mid=(i+j)/2,left=mid、左闭是可取的,right=mid、右开不可取

    image-20240417185239607

正解

左闭右闭:

/// <summary>
/// 左闭右闭
/// </summary>
/// <param name="arr"></param>
/// <param name="target"></param>
public int Search1(int[] arr, int target)
{
    int mid, left, right;
    left = 0;
    right = arr.Length - 1;
    //左闭右闭,left可以等于right
    while (left <= right)
    {
        mid = (left + right) / 2;
        if (target < arr[mid])
        {
            //右闭,mid可以取右边界,这里不用重复比较右边界,所以减一
            right = mid - 1;
        }
        else if (target > arr[mid])
        {
            //左闭,mid可以取右边界,这里不用重复比较左边界,所以加一
            left = mid + 1;
        }
        else
        {
            return mid;
        }
    }
    return -1;
}

左闭右开:

/// <summary>
/// 左闭右开
/// </summary>
/// <param name="arr"></param>
/// <param name="target"></param>
/// <returns></returns>
public int Search2(int[] arr, int target)
{
    int mid, left, right;
    left = 0;
    right = arr.Length - 1;
​
    while (left < right)
    {
        mid = (left + right) / 2;
​
        if (target < arr[mid])
        {
            //右开,mid不可以取右边界,这里恰好也不用重复比较右边界,所以不变
            right = mid;
        }
        else if (target > arr[mid])
        {
            //左闭,mid可以取右边界,这里不用重复比较左边界,所以加一
            left = mid + 1;
        }
        else
        {
            return mid;
        }
    }
    return -1;
}

尝试写的递归(并不符合题目条件,只是想训练一下自己递归的思路):

/// <summary>
/// 左闭右闭,尝试写的递归,加上了left和right
/// </summary>
/// <param name="arr"></param>
/// <param name="left"></param>
/// <param name="right"></param>
/// <param name="target"></param>
/// <returns></returns>
public int Search3(int[] arr, int left, int right, int target)
{
    int mid;
    mid = (left + right) / 2;
    if (left>right)
    {
        return -1;
    }
    else
    {
        if (target < arr[mid])
        {
            return Search3(arr, left, mid - 1, target);
        }
        else if (target > arr[mid])
        {
            return Search3(arr, mid + 1, right, target);
        }
        else
        {
            return mid;
        }
    }
}

后记

今天实在是有点忙,还有一题没写,第二天一起补上呜呜呜。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值