算法笔记:二分算法

程序或算法的时间复杂度

  • 一个程序或算法的时间效率,也称“时间复杂度”,有时简称“复杂度”
  • 复杂度常用大的字母O和小写字母n来表示,比如O(n),O(n^{2})等。n代表问题的规模
  • 时间复杂度是用算法运行过程中,某种时间固定的操作需要被执行的次数和n的关系来度量的。在无序数列中查找某个数,复杂度是O(n)
  • 计算复杂度的时候,只统计执行次数最多的(n足够大时)那种固定操作的次数。比如某个算法需要执行加法n^{2}次,除法n次,那么就记其复杂度是O(n^2)的。

计算复制度一般不考虑n的系数 ;

如果复杂度是多个n的函数之和,则只关心随n的增长增长得最快的那个函数;

O(n^3+n^2)=> O(n^3);

O(n^3+2^n)=> O(2^n);

O(n!+ 3^n)  => O(n!);

常数复杂度:O(1)           时间(操作次数)和问题的规模无关;

对数复杂度:O(log(n));

线性复杂度:O(n);

多项式复杂度:O(n^k);

指数复杂度:O(a^n);

阶乘复杂度:O(n!) ;

复杂度有“平均复杂度”和“最坏复杂度”两种,两者可能一致也可能不一致。

几种程序的复杂度

一个问题的复杂度是解决这个问题最好方法的复杂度;

  • 在无序数列中查找某个数(顺序查找) O(n)
  • 平面上有n个点,要求出任意两点之间的距离O(n^2)
  • 插入排序、选择排序、冒泡排序 O(n^2)
  • 快速排序 O(n*log(n))
  • 二分查找 O(log(n))

二分查找适用于有顺序的查找

二分查找函数

  • 写一个函数BinarySearch,在包含的size个元素的、从小到大排列的数组a里查找元素p,找到返回下标,找不到返回-1。要求复杂度O(log(n))
int BinarySearch(int a[], int size, int p)
{
	int L = 0, R = size - 1;
	while (L <= R)
	{
		int mid = L + (R - L) / 2;
		if (a[mid] == p)
			return mid;
		if(a[mid]<p)
			L = mid+1;
		else
			R = mid-1;
	}
	return -1;
}
  • 写一个函数LowerBound,在包含的size个元素的、从小到大排列的数组a里查找比给定整数p小的、下标最大的元素,找到返回下标,找不到返回-1。
int LowerBound(int a[], int size, int p)
{
	int lastPos = -1, L = 0, R = size - 1;
	while (L <= R)
	{
		int mid = L + (R - L) / 2;
		if (a[mid] >= p)
			R = mid - 1;
		else
		{
			L = mid;
			lastPos = mid + 1;
		}
	}
	return lastPos;
}

注意

为防止(L+R)过大溢出,查找区间正中下标不应写为 int mid=(L+R)/2; 应写为int mid = L + (R - L) / 2;

二分查找求方程的根

求方程f(x)=x^3-5x^2+10x-80=0的一个根,若求出的根是a,则要求\left | f(a) \right |\leqslant 10^{-6}

思想:易知该函数在[0,100]单调递增,f(0)<0,f(100)>0,那么在0到100上必有一根,用二分查找即可。

double f(double x)
{
	return x * x * x - 5 * x * x + 10 * x - 80;
}
int main()
{
	double EPS = 1e-6;
	double L = 0, R = 100, root,y;
	root = L + (R - L) / 2;
	y = f(root);
	while (abs(y) > EPS)
	{
		if (y < 0)
			L = root;
		else
			R = root;
		root = L + (R - L) / 2;
		y = f(root);
	}
	printf("%.8f", root);
	return 0;
}

运行结果

 例题1:

 程序

#include <stdio.h>
#include <iostream>
using namespace std;
int BinarySearch(int a[], int size, int p)//二分查找
{
	int L = 0, R = size - 1;
	while (L <= R)
	{
		int mid = L + (R - L) / 2;
		if (a[mid] == p)
			return mid;
		if(a[mid]<p)
			L = mid+1;
		else
			R = mid-1;
	}
	return -1;
}
void QuickSort(int *a, int begin, int end)//快速排序
{
	if (begin > end)
		return;
	int tmp = a[begin];
	int i = begin, j = end;
	while (i != j)
	{
		while (tmp <= a[j]&&i<j)
			j--;
		while (tmp >= a[i]&&i<j)
			i++;
		if (i < j)
		{
			int t = a[j];
			a[j] = a[i];
			a[i] = a[j];
		}
	}
	a[begin] = a[i];
	a[i] = tmp;
	QuickSort(a, begin, i - 1);
	QuickSort(a, i + 1, end);
}
int main()
{
	int a[100000], m, n;
	scanf_s("%d",&n);
	for (int i = 0; i < n; i++)
		scanf_s("%d", &a[i]);
	scanf_s("%d", &m);
	QuickSort(a, 0, n - 1);
	for (int i = 0; i < n - 1; i++)
	{
		 int x = BinarySearch(a, n, m - a[i]);
		if (x != -1)
		{
			printf("%d+%d", a[i], a[x]);
			break;
		}
	}
	return 0;
}

运行结果

 例题2:

 程序

#include <iostream>
#include <cstdio>
#include <algorithm>
using namespace std;
long long room[100005];
bool f(long long x, long long N, long long room[], long long cow) 
{
	cow--;
	long long left = 0;
	for (long long i = 0; i < N; i++) 
	{
		if (room[i] - room[left] >= x) 
		{
			left = i;
			cow--;
		}
		if (cow == 0)return true;
	}
	return false;
}
int main()
{
	long long N, cow;
	cin >> N >> cow;
	for (long long i = 0; i < N; i++)
		scanf_s("%lld", &room[i]);
	sort(room, room + N);

	long long left = 1, right = room[N - 1] / cow, mid;
	mid = left + (right - left) / 2;
	while (left < right) 
	{
		if (f(mid, N, room, cow))
			left = mid;
		else 
			right = mid - 1;
		mid = left + (right - left) / 2;

		if (right - left == 1)
		{ 
			if (f(right, N, room, cow))
				mid = right;
			else
				mid = left;
			break;
		}
	}
	printf("%d", mid);
	return 0;
}

运行结果

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 《算法笔记》是由胡凡编写的一本关于算法学习的教材,它最初以C/C++为主要编程语言。这本教材主要面向想要学习算法的初学者,以及希望进一步提升算法能力的人群。 《算法笔记》的内容广泛而且深入,涵盖了很多算法的基础知识和主要思想,如递归、排序、查找、图论、动态规划等。通过学习这本教材,读者可以掌握这些算法的基本原理和实现方法,提高解决实际问题的能力。 该教材有几个特点:首先,它提供了很多例子和习题,帮助读者巩固所学的知识;其次,它介绍了很多常用的数据结构,如数组、链表、栈、队列等,读者可以通过学习这些数据结构更好地理解和应用算法;最后,它还介绍了一些高级主题,如高级数据结构、算法优化等,这些内容对于进一步提升算法水平非常有帮助。 《算法笔记》是一本入门级的教材,因此在阅读时需要一些基本的编程知识和逻辑思维能力。该教材的语言简洁明快,适合作为学习参考书,同时也可以作为算法竞赛的辅助教材。 总而言之,《算法笔记》是一本很好的算法学习教材,它以C/C++为编程语言,全面介绍了算法的基本知识和常用方法,适合想要学习算法的初学者。无论是学术研究还是实际应用,阅读《算法笔记》都能提升算法能力,并为进一步学习和应用算法打下坚实的基础。 ### 回答2: 《算法笔记 胡凡 c/c 快速入门pdf》是一本介绍算法和C/C++编程语言的入门书籍。该书的作者是胡凡,它主要规划了算法的学习路径以及基本的C/C++编程语言。这本书适合初学者快速入门,对于想要系统学习算法和C/C++编程的人来说是一本不错的选择。 这本书的内容非常系统和全面。它以算法和数据结构为基础,包括基本排序、查找、图论等算法的介绍和实践,让读者能够快速掌握这些算法的原理和实现方法。同时,它还介绍了C/C++语言的基础知识和编程技巧,帮助读者理解和运用这些知识。 书中每章都有一些练习题,帮助读者巩固所学的知识。同时,每个章节末尾还提供了一些进阶的题目和参考答案,供读者深入学习和自我检测。这样的设计能够帮助读者更好地理解和掌握所学的内容。 总的来说,《算法笔记 胡凡 c/c 快速入门pdf》是一本很好的算法和C/C++入门书籍。它能够帮助读者快速了解算法和数据结构的基础知识,并学会使用C/C++语言进行编程。无论是对于想要入门算法和C/C++编程的初学者,还是已经有一定基础的读者,这本书都是一个很好的选择。 ### 回答3: 《算法笔记:胡凡C/C++快速入门PDF》是一本很棒的入门算法书籍。这本书主要介绍了常用的数据结构与算法,并通过C/C++语言来实现这些算法。 首先,这本书非常适合算法初学者。它从基础的数据结构开始讲解,如数组、链表、栈和队列,然后逐渐引入更复杂的数据结构,如二叉树、图和堆。此外,书中还介绍了常用的排序和查找算法,如冒泡排序、快速排序、二分查找等。每个算法都配有具体的代码实现和详细的解释,帮助读者理解算法的原理和应用。 其次,这本书的学习资料丰富。书中提供了很多例题和习题,读者可以通过实践来巩固所学的知识。此外,书中还介绍了一些常见的算法优化技巧和设计思路,提供了一些高效解决问题的方法和思考方式。 最后,这本书的编写风格简明易懂。作者通过清晰的语言和简洁的代码,将复杂的算法问题简化为易于理解的部分。不论是对于算法初学者还是对于有一定编程基础的读者,这本书都是一本很好的学习资料。 总而言之,《算法笔记:胡凡C/C++快速入门PDF》是一本很好的入门算法书籍。它适合初学者学习和理解常用的数据结构与算法,通过具体的代码实现和详细的解释帮助读者掌握算法的原理和应用。无论是编程初学者还是有一定基础的读者,都可以从这本书中获得丰富的知识和实践经验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值