前言
大二开始定下实践项目的主方向后,对于数据结构、算法这一块基本就是停滞状态了。
选择游戏方向后,unity和C#的实际项目对于算法方面的要求并不多。实习需要去大厂,并且为了进一步提高自己的能力,决定接下来的日子里苦练算法。
以前练习算法这方面,用的都是C、Java这类。用C#写还是头一次,在练习算法的同时,增进自己对C#的使用熟练度,也是一举两得。
接下来的日子,打算跟着卡哥的《算法随想录》刷题。根据http://t.csdnimg.cn/l8b7i,这篇大哥的博客,每天进行练习。自己练习确实得要很大的毅力和决心,希望自己能够坚持下来,共勉~
二分查找
(用时: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、右开不可取
正解
左闭右闭:
/// <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;
}
}
}
后记
今天实在是有点忙,还有一题没写,第二天一起补上呜呜呜。