第一弹

作者: lee_08b0 | 来源:发表于2020-07-14 15:50 被阅读0次

思考能力是人最重要的核心竞争力,而算法是为数不多能有效训练大脑思考能力的途径之一

什么叫数据结构,什么叫算法

  1. 数据结构:在计算机科学中,数据结构(英语:data structure)是计算机中存储、组织数据的方式。 上面这个是wiki百科中关于数据结构的定义。
  2. 算法:
    • 一个有限指令集,
    • 接受一些输入(有时候不需要输入)
    • 产生输出
    • 一定在有限步骤后终止
    • 每一条指令必须明确

一般来说,广义上讲数据结构,就是指一些数据的存储结构。算法就是操作数据的方法。
以图书馆储藏书籍举例,管理员将书籍分门别类进行存储。按照一定的规律编号,就是书籍这个数据的存储结构。
而在查找一本书的时候,可以一本本找,也可以根据书籍类别编号等信息定位同类型位置在依次查找。这些查找的方法就是算法
而狭义的讲代指某些著名的数据结构和算法,例如队列二分查找动态规划

数据结构和算法的关系

数据结构和算法是相辅相成的,数据结构是为算法服务的,算法腰作用在特定的数据结构之上

算法的优劣

image.png

复杂度分析

  • 空间复杂度S(n):根据算法写成的程序在执行时占用存储单元的长度。(S : space ) SN = C * N
  • 时间复杂度T(n):根据算法写成的程序在执行时耗费时间的长度。(T: time)
    所有代码的执行时间 T(n) 与每行代码的执行次数 n 成正比
    T(n) = O(f(n)) O 表示代码执行时间T(n) 与f(n)表达式成正比。
    这就是大 O 时间复杂度表示法。大 O 时间复杂度实际上并不具体表示代码真正的执行时间,而是表示代码执行时间随数据规模增长的变化趋势,所以,也叫作渐进时间复杂度(asymptotic time complexity),简称时间复杂度
//如下代码,假设每行代码执行时间都一样 为个unit_time
//2、3行代码分别需要1个unit_time 4、5行都执行了n遍 需要2n
//所以此段代码执行时间为(2n + 2)* unit_time
int cal(int n) {
   int sum = 0;
   int i = 1;
   for (; i <= n; ++i) { 
    sum = sum + i;
   } 
   return sum;
 }
/**
*  2、3、4各需要1个unit_time
* 5、6需要2n * unit_time  7、8 执行了n^
* 则总时间为T(n) = (2n^ + 2n + 3) * unit_time
*/
 int cal(int n) {
   int sum = 0;
   int i = 1;
   int j = 1;
   for (; i <= n; ++i) {
     j = 1;
     for (; j <= n; ++j) {
       sum = sum +  i * j;
     }
   }
 }

上述代码,当n趋于无穷大时,公式中低阶、常量、系数三部分对增长趋势的影响无限趋于1,所以可以忽略。T(n) = O(n); T(n) = O(n^2)

时间复杂度分析

  1. 只关注循环执行次数最多的一段代码
    大O这种复杂度表示方法只是表示一种变化趋势,通常可以忽略掉公式中的常量、低阶、系数,只需要记录最大阶的量级就就可以了。在分析一个算法、一段代码的时间复杂度的时候,也只关注循环执行次数最多的那一段代码就可以了
// 此段代码。2、3行都是常量级的执行时间,与n大小无关,所以对于复杂度没有任何影响
//循环影响4、5行代码。所以总的时间复杂度为O(n)
 int cal(int n) {
   int sum = 0;
   int i = 1;
   for (; i <= n; ++i) {
     sum = sum + i;
   }
   return sum;
 }
  1. 加法法则:总复杂度等于量级最大的那段代码的复杂度。
  2. 乘法法则:嵌套代码的复杂度等于嵌套内外代码复杂度的乘积。

常见时间复杂度分析

image.png
对于上述复杂度量级,粗略的分为多项式量级非多项式量级.
把时间复杂度为非多项式量级的算法问题叫作 NP(Non-Deterministic Polynomial,非确定多项式)问题。当数据规模 n 越来越大时,非多项式量级算法的执行时间会急剧增加,求解问题的执行时间会无限增长。所以,非多项式时间复杂度的算法其实是非常低效的算法。
  1. O(1):常量级时间复杂度的一种表示方法,并不是指只执行一行代码。一般情况下,代码执行时间不随n增大而增大,这样的记为O(1)。一般情况下,只要算法中不存在循环语句、递归语句,即使有成千上万行的代码,其时间复杂度也是Ο(1)。
  2. O(logn)/O(nlogn)
//O(log2n)
//2^0  2^1  2^2 .... 2^x = n
//x = log2n
 i=1;
 while (i <= n)  {
   i = i * 2;
 }
//O(log3n)
 i=1;
 while (i <= n)  {
   i = i * 3;
 }

对数之间是可以互相转换的,log3n 就等于 log32 * log2n,所以 O(log3n) = O(C * log2n),其中 C=log32 是一个常量。基于我们前面的一个理论:在采用大 O 标记复杂度的时候,可以忽略系数,即 O(Cf(n)) = O(f(n))。所以,O(log2n) 就等于 O(log3n)。因此,在对数阶时间复杂度的表示方法里,我们忽略对数的“底”,统一表示为 O(logn)。同理,根据乘法法则,如果一段代码的时间复杂度是 O(logn),我们循环执行 n 遍,时间复杂度就是 O(nlogn) 了。而且,O(nlogn) 也是一种非常常见的算法时间复杂度。比如,归并排序、快速排序的时间复杂度都是 O(nlogn)。

  1. m+n / m*n
//从代码中可以看出,m 和 n 是表示两个数据规模。我们无法事先评估 m 
//和 n 谁的量级大,所以我们在表示复杂度的时候,就不能简单地利用
//加法法则,省略掉其中一个。所以,代码的时间复杂度就是 O(m+n)。
//针对这种情况,原来的加法法则就不正确了,需要将加法规则
//改为:T1(m) + T2(n) = O(f(m) + g(n))。
//但是乘法法则继续有效:T1(m)*T2(n) = O(f(m) * f(n))。
int cal(int m, int n) {
  int sum_1 = 0;
  int i = 1;
  for (; i < m; ++i) {
    sum_1 = sum_1 + i;
  }
  int sum_2 = 0;
  int j = 1;
  for (; j < n; ++j) {
    sum_2 = sum_2 + j;
  }
  return sum_1 + sum_2;
}

空间复杂度分析

时间复杂度的全称是渐进时间复杂度,表示算法的执行时间与数据规模之间的增长关系。类比一下,空间复杂度全称就是渐进空间复杂度(asymptotic space complexity),表示算法的存储空间与数据规模之间的增长关系。

相关文章

网友评论

    本文标题:第一弹

    本文链接:https://www.haomeiwen.com/subject/isffhktx.html