精华内容
下载资源
问答
  • 数据结构预算法

    2017-08-15 22:09:38
    学习算法
  • python-数据结构预算法

    2021-02-07 19:59:08
    数据结构预算法
  • 数据结构预算法分析

    2012-04-25 13:48:23
    数据结构预算法分析 20世纪顶尖30部计算机书籍之一
  • 数据结构预算法Python语言描述裘宗燕,书链接:https://book.douban.com/subject/26702568/ (PS,该版本为扫描版本,有目录,。字迹不是特别清晰。。仅供参考)
  • 数据结构预算法 很好的演示教程。有FLASH 的动画演示,更加形象
  • 本课件是北大教授张铭的数据结构预算法的课件
  • 版权声明:本文为博主原创文章,遵循 CC 4.0 by-sa 版权协议,转载请附上原文出处链接和本声明。 ...数据结构指的是“一组数据的存储结构”,算法指的是“操作数据的一组方法...最常用的数据结构预算法: 数据结构:数组...

    版权声明:本文为博主原创文章,遵循 CC 4.0 by-sa 版权协议,转载请附上原文出处链接和本声明。
    本文链接:https://blog.csdn.net/ityqing/article/details/82838524

    数据结构指的是“一组数据的存储结构”,算法指的是“操作数据的一组方法”。
    数据结构是为算法服务的,算法是要作用再特定的数据结构上的。

    最常用的数据结构预算法:

    数据结构:数组、链表、栈、队列、散列表、二叉树‘、堆、跳表、图、Tire树
    算法: 递归、排序、二分查找、搜索、哈希算法、贪心算法、分治算法、回溯算法、动态规划、字符串匹配算法
    1·算法的复杂度
    1.1大O复杂度表示法
     公式:

    T(n)表示代码执行的时间; n表示数据规模的大小; f(n) 表示每行代码执行的次数总和。因为这是一个公式, 所以用f(n)来表示。公式中的O,表示代码的执行时间T(n)与f(n)表达式成正比。

    所以,第一个例子中的T(n) = O(2n+2),第二个例子中的T(m) = 0(2n2 +2n+3)。这就是大O时间复杂度表示法。大O时间复杂度实际上并不具体表示代码真正的执行时间,而是表示代码执行时间随数据规模增长的变化趋势,所以,也叫作渐进时间复杂度(asymptotic time complexity),简称时间复杂度。

    当n很大时,你可以把它想象成10000、100000。 而公式中的低阶、常量、系数三部分并不左右增长趋势,所以都可以忽略。我们只需要记录-个最大量级就可以了,如果用大O表示法表示刚讲的那两段代码的时间复杂度,就可以记为: T(n) = O(n); T(n)= 0(n2)。

    1.2.复杂度分析法则
    1)单段代码看高频:比如循环。
    2)多段代码取最大:比如一段代码中有单循环和多重循环,那么取多重循环的复杂度。
    3)嵌套代码求乘积:比如递归、多重循环等
    4)多个规模求加法:比如方法有两个参数控制两个循环的次数,那么这时就取二者复杂度相加。

    1.3 时间复杂度分析

    只关注循环执行次数最多的一段代码
    加法法则:总复杂度等于量级最大的那段代码的复杂度
    乘法法则:嵌套代码的复杂度等于嵌套内外代码复杂度的乘积
    1.4 几种常见时间复杂度实例分析

    多项式阶:随着数据规模的增长,算法的执行时间和空间占用,按照多项式的比例增长。包括,
    O(1)(常数阶)、O(logn)(对数阶)、O(n)(线性阶)、O(nlogn)(线性对数阶)、O(n2)(平方阶)、O(n3)(立方阶)
    非多项式阶:随着数据规模的增长,算法的执行时间和空间占用暴增,这类算法性能极差。包括,
    O(2^n)(指数阶)、O(n!)(阶乘阶)

    O(1) :
    常量级时间复杂度,只要代码的执行时间不随 n 的增大而增长,这样代码的时间复杂度我们都记作 O(1)。

    O(logn)、O(nlogn)
    i=1;
    while(i<=n) {
    i = i*2;
    }

    x=log2n,所以,这段代码的时间复杂度就是 O(log2n)

    O(m+n)、O(m*n)

    int cal(int m, int n) {
    intsum_1=e;
    inti=1;
    for(;iくm;++i){
    sum_1=sum1+i;
    }
    int sum_2=0;
    int j=1;
    for (;jくn;+j){
    sum_2=sum_2+j;
    }
    return sum_1 + sum_2;
    }
    从代码中可以看出,m和n是表示两个数据规模。我们无法事先评估m和n谁的量级大,所以我们在表示复杂度的时候,就不能简单地利用加法法则,省略掉其中一个。所以,上面代码的时间复 杂度就是0(m+n)。

    针对这种情况,原来的加法法则就不正确了,我们需要将加法规则改为: T1(m) + T2(m) = O(f(m) + g(n))。但是乘法法则继续有效: T1(m)*T2(n) = O(f(m) * f(n))。

    1.5 空间复杂度分析

    表示算法的存储空间与数据规模之间的增长关系。

    void print(int n) {
    inti=0;
    int[] a = new int[n];
    for (i; i <n; ++i) {
    a[i] =i* i;
    }
    for(i=n-1;i>=0;–i){
    print out a[i]
    }
    }
    跟时间复杂度分析一样,我们可以看到,第2行代码中,我们申请了一个空间存储变量i,但是它是常最阶的,跟数据规模n没有关系,所以我们可以忽略。第3行申请了一个大小为n的int类型数组,除此之外,剩下的代码都没有占用更多的空间,所以整段代码的空间复杂度就是O(n)。

    我们常见的空间复杂度就是O(1)、O(n)、 O(n2), 像O(logn)、O(nlogn) 这样的对数阶复杂度平时都用不到。而且,空间复杂度分析比时间复杂度分析要简单很多。所以,对于空间复杂度,掌握刚我说的这些内容已经足够了。

    1.6 复杂度增长趋势图:

    最好情况时间复杂度、最坏时间复杂度、平均情況时间复杂度、均摊时间复杂度。

    一、复杂度分析的4个概念
    1.最坏情况时间复杂度:代码在最理想情况下执行的时间复杂度。
    2.最好情况时间复杂度:代码在最坏情况下执行的时间复杂度。
    3.平均时间复杂度:用代码在所有情况下执行的次数的加权平均值表示。
    4.均摊时间复杂度:在代码执行的所有复杂度情况中绝大部分是低级别的复杂度,个别情况是高级别复杂度且发生具有时序关系时,可以将个别高级别复杂度均摊到低级别复杂度上。基本上均摊结果就等于低级别复杂度。

    二、为什么要引入这4个概念?
    1.同一段代码在不同情况下时间复杂度会出现量级差异,为了更全面,更准确的描述代码的时间复杂度,所以引入这4个概念。
    2.代码复杂度在不同情况下出现量级差别时才需要区别这四种复杂度。大多数情况下,是不需要区别分析它们的。

    三、如何分析平均、均摊时间复杂度?
    1.平均时间复杂度
    代码在不同情况下复杂度出现量级差别,则用代码所有可能情况下执行次数的加权平均值表示。
    2.均摊时间复杂度
    两个条件满足时使用:1)代码在绝大多数情况下是低级别复杂度,只有极少数情况是高级别复杂度;2)低级别和高级别复杂度出现具有时序规律。均摊结果一般都等于低级别复杂度。

    数组
    线性表:   线性表就是数据排成像一条线一样的结构.每个现行表上的数据最多只有前和后两个方向.常见的线性表结构:数组,链表、队列、栈等。

    什么是数组:

    数组(Array)是一种线性表数据结构。它用一组连续的内存空间,来存储一组具有相同类型的数据。
     连续的内存空间和相同类型的数据(随机访问的前提)
    优点:两限制使得具有随机访问的特性缺点:删除,插入数据效率低
    数组怎么根据下标随机访问的?
    通过寻址公式:a[i]_address = base_address + i * data_type_size
    其中data type size表示数组中每个元素的大小,base_address 是首元素地址。

    为何数组插入和删除低效

    插入:
    若有一元素想往int[n]的第k个位置插入数据,需要在k-n的位置往后移。
    最好情况时间复杂度 O(1)

    如果数组中的数据不是有序的,也就是无规律的情况下,可以直接把第k个位置上的数据移到最后,然后将插入的数据直接放在第k个位置上。

    最坏情况复杂度为O(n)

    平均负责度为O(n)

    1. 低效的插入和删除
      1) 插入:从最好O(1) 最坏O(n) 平均O(n)
      2) 插入:数组若无序,插入新的元素时,可以将第K个位置元素移动到数组末尾,把心的元素,插入到第k个位置,此处复杂度为O(1)。作者举例说明
      3) 删除:从最好O(1) 最坏O(n) 平均O(n)
      4) 多次删除集中在一起,提高删除效率
      记录下已经被删除的数据,每次的删除操作并不是搬移数据,只是记录数据已经被删除,当数组没有更多的存储空间时,再触发一次真正的删除操作。即JVM标记清除垃圾回收算法。

    链表
    什么是链表
    1.和数组一样,链表也是一种线性表。
    2.从内存结构来看,链表的内存结构是不连续的内存空间,是将一组零散的内存块串联起来,从而进行数据存储的数据结构。

    3.链表中的每一个内存块被称为节点Node。节点除了存储数据外,还需记录链上下一个节点的地址,即后继指针next。

    链表的特点
    1.插入、删除数据效率高O(1)级别(只需更改指针指向即可),随机访问效率低O(n)级别(需要从链头至链尾进行遍历)。

    2.和数组相比,内存空间消耗更大,因为每个存储数据的节点都需要额外的空间存储后继指针。

    常用链表
    1.单链表

    1)每个节点只包含一个指针,即后继指针。
    2)单链表有两个特殊的节点,即首节点和尾节点。为什么特殊?用首节点地址表示整条链表,尾节点的后继指针指向空地址null。
    3)性能特点:插入和删除节点的时间复杂度为O(1),查找的时间复杂度为O(n)。

    2.循环链表

    1)除了尾节点的后继指针指向首节点的地址外均与单链表一致。
    2)适用于存储有循环特点的数据,比如约瑟夫问题。

    3.双向链表

    1)节点除了存储数据外,还有两个指针分别指向前一个节点地址(前驱指针prev)和下一个节点地址(后继指针next)。
    2)首节点的前驱指针prev和尾节点的后继指针均指向空地址。
    3)性能特点:
    和单链表相比,存储相同的数据,需要消耗更多的存储空间。
    插入、删除操作比单链表效率更高O(1)级别。以删除操作为例,删除操作分为2种情况:给定数据值删除对应节点和给定节点地址删除节点。对于前一种情况,单链表和双向链表都需要从头到尾进行遍历从而找到对应节点进行删除,时间复杂度为O(n)。对于第二种情况,要进行删除操作必须找到前驱节点,单链表需要从头到尾进行遍历直到p->next = q,时间复杂度为O(n),而双向链表可以直接找到前驱节点,时间复杂度为O(1)。
    对于一个有序链表,双向链表的按值查询效率要比单链表高一些。因为我们可以记录上次查找的位置p,每一次查询时,根据要查找的值与p的大小关系,决定是往前还是往后查找,所以平均只需要查找一半的数据。

    4.双向循环链表:

    首节点的前驱指针指向尾节点,尾节点的后继指针指向首节点。

    选择数组还是链表?
    1.插入、删除和随机访问的时间复杂度
    数组:插入、删除的时间复杂度是O(n),随机访问的时间复杂度是O(1)。
    链表:插入、删除的时间复杂度是O(1),随机访问的时间复杂端是O(n)。

    2.数组缺点
    1)若申请内存空间很大,比如100M,但若内存空间没有100M的连续空间时,则会申请失败,尽管内存可用空间超过100M。
    2)大小固定,若存储空间不足,需进行扩容,一旦扩容就要进行数据复制,而这时非常费时的。
    3.链表缺点
    1)内存空间消耗更大,因为需要额外的空间存储指针信息。
    2)对链表进行频繁的插入和删除操作,会导致频繁的内存申请和释放,容易造成内存碎片,如果是Java语言,还可能会造成频繁的GC(自动垃圾回收器)操作。
    4.如何选择?
    数组简单易用,在实现上使用连续的内存空间,可以借助CPU的缓冲机制预读数组中的数据,所以访问效率更高,而链表在内存中并不是连续存储,所以对CPU缓存不友好,没办法预读。
    如果代码对内存的使用非常苛刻,那数组就更适合。

    应用
    1.如何分别用链表和数组实现LRU缓冲淘汰策略?
    1)什么是缓存?
    缓存是一种提高数据读取性能的技术,在硬件设计、软件开发中都有着非广泛的应用,比如常见的CPU缓存、数据库缓存、浏览器缓存等等。
    2)为什么使用缓存?即缓存的特点
    缓存的大小是有限的,当缓存被用满时,哪些数据应该被清理出去,哪些数据应该被保留?就需要用到缓存淘汰策略。
    3)什么是缓存淘汰策略?
    指的是当缓存被用满时清理数据的优先顺序。
    4)有哪些缓存淘汰策略?
    常见的3种包括先进先出策略FIFO(First In,First Out)、最少使用策略LFU(Least Frenquently Used)、最近最少使用策略LRU(Least Recently Used)。
    5)链表实现LRU缓存淘汰策略
    当访问的数据没有存储在缓存的链表中时,直接将数据插入链表表头,时间复杂度为O(1);当访问的数据存在于存储的链表中时,将该数据对应的节点,插入到链表表头,时间复杂度为O(n)。如果缓存被占满,则从链表尾部的数据开始清理,时间复杂度为O(1)。
    6)数组实现LRU缓存淘汰策略
    方式一:首位置保存最新访问数据,末尾位置优先清理
    当访问的数据未存在于缓存的数组中时,直接将数据插入数组第一个元素位置,此时数组所有元素需要向后移动1个位置,时间复杂度为O(n);当访问的数据存在于缓存的数组中时,查找到数据并将其插入数组的第一个位置,此时亦需移动数组元素,时间复杂度为O(n)。缓存用满时,则清理掉末尾的数据,时间复杂度为O(1)。
    方式二:首位置优先清理,末尾位置保存最新访问数据
    当访问的数据未存在于缓存的数组中时,直接将数据添加进数组作为当前最有一个元素时间复杂度为O(1);当访问的数据存在于缓存的数组中时,查找到数据并将其插入当前数组最后一个元素的位置,此时亦需移动数组元素,时间复杂度为O(n)。缓存用满时,则清理掉数组首位置的元素,且剩余数组元素需整体前移一位,时间复杂度为O(n)。(优化:清理的时候可以考虑一次性清理一定数量,从而降低清理次数,提高性能。)
    2.如何通过单链表实现“判断某个字符串是否为水仙花字符串”?(比如 上海自来水来自海上)
    1)前提:字符串以单个字符的形式存储在单链表中。
    2)遍历链表,判断字符个数是否为奇数,若为偶数,则不是。
    3)将链表中的字符倒序存储一份在另一个链表中。
    4)同步遍历2个链表,比较对应的字符是否相等,若相等,则是水仙花字串,否则,不是。
    六、设计思想
    时空替换思想:“用空间换时间” 与 “用时间换空间”
    当内存空间充足的时候,如果我们更加追求代码的执行速度,我们就可以选择空间复杂度相对较高,时间复杂度小相对较低的算法和数据结构,缓存就是空间换时间的例子。如果内存比较紧缺,比如代码跑在手机或者单片机上,这时,就要反过来用时间换空间的思路。

    队列

    什么是队列:

    队列是一种受限的线性表数据结构,只支持两个操作:

    入栈push()和出栈pop0,队列跟非常相似,支持的操作也 ,很有限,最基本的操作也是两个:入队enqueue(),放一个数据到队列尾部;出队dequeue0),从队列头部取一个元素。

    特点:

    1 . 队列跟栈一样,也是一种抽象的数据结构。

    1. 具有先进先出的特性,支持在队尾插入元素,在队头删除元素。

    实现:

    队列可以用数组来实现,也可以用链表来实现。

    用数组实现的栈叫作顺序栈,用链表实现的栈叫作链式栈。

    同样,用数组实现的队列叫作顺序队列,用链表实现的队列叫作链式队列。

    基于数组的队列:

    代码代码

    实现思路:

    实现队列需要两个指针:一个是head指针,指向队头;一个是tail指针,指向队尾。你可以结合下面这幅图来理解。当a,b,c,d依次入队之后,队列中的head指针指向下标为0的位置, tail指针指向下标为4的位置。

    当我们调用两次出队操作之后,队列中head指针指向下标为2的位置, tail指针仍然指向下标为4的位置.

    随着不停地进行入队、出队操作, head和tail都会持续往后移动。当tail移 . ,动到最右边,即使数组中还有空闲空间,也无法继续往队列中添加数据了。这个问题该如何解决呢?

    在出队时可以不用搬移数据。如果没有空闲空间了,我们只需要在入队时,再集中触 ,发一次数据的搬移操作。

    当队列的tail指针移动到数组的最右边后,如果有新的数据入队,我们可以将 head到tail之间的数据,整体搬移到数组中0到tail-head的位置。

    基于链表的实现:

    需要两个指针: head指针和tail指针,它们分别指向链表的第一个结,点和最后一个结点。

    如图所示,入队时, tail->next= new node, tail = tail->next:出队时, head = head->next.

    循环队列:

    我们刚才用数组来实现队列的时候,在tail==n时,会有数据搬移操作,这样入队操作性能就会受到影响。那有没有办法能够避免数据搬移呢?我们来看看循环队列的解决思路。循环队列,顾名思义,它长得像一个环。原本数组是有头有尾的,是一条直线。现在我们把首尾相,连,板成了一个环。我画了一张图,你可以直观地感受一下。

    我们可以看到,图中这个队列的大小为8,当前head-4, tail-7,当有一个新的元素a入队时, .我们放入下标为7的位置。但这个时候,我们并不把tail更新为8,而是将其在环中后移一位,到下标为0的位置。当再有一个元素b入队时,我们将b放入下标为0的位置,然后tail加1更新为1,所以,在a, b依次入队之后,循环队列中的元素就变成了下面的样子:

    队列为空的判断条件是head == tail,但队列满的判断条件就稍微有点复杂了。我画了一张队列满的图,你可以看一下,试着总结一下规律,

    就像我图中画的队满的情况, tail=3, head-4, n=8,所以总结一下规律就是: (3+1)%8-4,多画几张队满的图,你就会发现,当队满时, (tail+1)%n=head…你有没有发现,当队列满时,图中的tail指向的位置实际上是没有存储数据的。所以,循环队列会浪费一个数组的存储空间。

    解决浪费一个存储空间的思路:定义一个记录队列大小的值size,当这个值与数组大小相等时,表示队列已满,当tail达到最底时,size不等于数组大小时,tail就指向数组第一个位置。当出队时,size—,入队时size++

    阻塞队列和并发队列(应用比较广泛)

    阻塞队列其实就是在队列基础上增加了阻塞操作。

    简单来说,就是在队列为空的时候,从队头取数 , 据会被阻塞。因为此时还没有数据可取,直到队列中有了数据才能返回;如果队列已经满了,那么插入数据的操作就会被阻塞,直到队列中有空闲位置后再插入数据,然后再返回。

    你应该已经发现了,上述的定义就是一个"生产者-消费者模型" !是的,我们可以使用阻塞队列,轻松实现一个"生产者-消费者模型" !这种基干阴寒队列实现的"生产者-消费者模型" ,可以有效地协调生产和消费的速度。当"生产 , 者"生产数据的速度过快, “消费者"来不及消费时,存储数据的队列很快就会满了。这个时候,生产者就阻塞等待,直到"消费者"消费了数据, “生产者"才会被唤醒继续"生产而且不仅如此,基于阻塞队列,我们还可以通过协调"生产者"和"消费者"的个数,来提高数据,的处理效率。比如前面的例子,我们可以多配置几个"消费者” ,来应对一个"生产者”

    小结:

    队列最大的特点就是先进先出,主要的两个操作是入队和出队。

    它既可以用数组来实现,也可以用链表来实现。用数组实现的叫顺序队列,用链表实现的叫链式队列。

    长在数组实现队列的时候,会有数据搬移操作,要想解决数据搬移的问题,我们就,需要像环一样的循环队列。要想写出没有bug的循环队列实现代码,关键要确定好队空和队满的,判定条件。

    阻塞队列、并发队列,底层都还是队列这种数据结构,只不过在之上附加了很多其他功能。阻塞队列就是入队、出队操作可以阴寒,并发队列就是队列的操作多线程安全。

    递归

    一、什么是递归?

    1.递归是一种非常高效、简洁的编码技巧,一种应用非常广泛的算法,比如DFS深度优先搜索、前中后序二叉树遍历等都是使用递归。
    2.方法或函数调用自身的方式称为递归调用,调用称为递,返回称为归。
    3.基本上,所有的递归问题都可以用递推公式来表示,比如
    f(n) = f(n-1) + 1; 
    f(n) = f(n-1) + f(n-2);
    f(n)=n*f(n-1);

    二、为什么使用递归?递归的优缺点?

    1.优点:代码的表达力很强,写起来简洁。
    2.缺点:空间复杂度高、有堆栈溢出风险、存在重复计算、过多的函数调用会耗时较多等问题。

    三、什么样的问题可以用递归解决呢?

    一个问题只要同时满足以下3个条件,就可以用递归来解决:
    1.问题的解可以分解为几个子问题的解。何为子问题?就是数据规模更小的问题。
    2.问题与子问题,除了数据规模不同,求解思路完全一样
    3.存在递归终止条件

    四、如何实现递归?

    1.递归代码编写
    写递归代码的关键就是找到如何将大问题分解为小问题的规律,并且基于此写出递推公式,然后再推敲终止条件,最后将递推公式和终止条件翻译成代码。
    2.递归代码理解
    对于递归代码,若试图想清楚整个递和归的过程,实际上是进入了一个思维误区。
    那该如何理解递归代码呢?如果一个问题A可以分解为若干个子问题B、C、D,你可以假设子问题B、C、D已经解决。而且,你只需要思考问题A与子问题B、C、D两层之间的关系即可,不需要一层层往下思考子问题与子子问题,子子问题与子子子问题之间的关系。屏蔽掉递归细节,这样子理解起来就简单多了。
    因此,理解递归代码,就把它抽象成一个递推公式,不用想一层层的调用关系,不要试图用人脑去分解递归的每个步骤。

    递归的关键是终止条件
    五、递归常见问题及解决方案

    1.警惕堆栈溢出:可以声明一个全局变量来控制递归的深度,从而避免堆栈溢出。
    2.警惕重复计算:通过某种数据结构来保存已经求解过的值,从而避免重复计算。

    六、如何将递归改写为非递归代码?

    笼统的讲,所有的递归代码都可以改写为迭代循环的非递归写法。如何做?抽象出递推公式、初始值和边界条件,然后用迭代循环实现。

    排序

    一、排序方法与复杂度归类
    (1)几种最经典、最常用的排序方法:冒泡排序、插入排序、选择排序、快速排序、归并排序、计数排序、基数排序、桶排序。
    (2)复杂度归类
    冒泡排序、插入排序、选择排序 O(n^2)
    快速排序、归并排序 O(nlogn)
    计数排序、基数排序、桶排序 O(n)

    二、如何分析一个“排序算法”?
    <1>算法的执行效率

    1. 最好、最坏、平均情况时间复杂度。
    2. 时间复杂度的系数、常数和低阶。
    3. 比较次数,交换(或移动)次数。
      <2>排序算法的稳定性
    4. 稳定性概念:如果待排序的序列中存在值相等的元素,经过排序之后,相等元素之间原有的先后顺序不变。
    5. 稳定性重要性:可针对对象的多种属性进行有优先级的排序。
    6. 举例:给电商交易系统中的“订单”排序,按照金额大小对订单数据排序,对于相同金额的订单以下单时间早晚排序。用稳定排序算法可简洁地解决。先按照下单时间给订单排序,排序完成后用稳定排序算法按照订单金额重新排序。
      <3>排序算法的内存损耗
      原地排序算法:特指空间复杂度是O(1)的排序算法。

    常见的排序算法:

    冒泡排序

    冒泡排序只会操作相邻的两个数据。每次冒泡操作都会对相邻的两个元素进行比较,看是否满足大小关系要求,如果不满足就让它俩互换。

    代码:

    public int[] bubbleSort(int[] a) {
    	int n = a.length;
    	if (n<=1) return a;
    	
    	//提前退出冒泡循环的标志
    	boolean flag = false;
    	
    	for (int i = 0; i < n; i++) {
    		for (int j = 0; j < n-i-1; j++) {
    			if (a[j]>a[j+1]) {//
    				int temp = a[j];
    				a[j] = a[j+1];
    				a[j+1] = temp;
    				
    				flag = true;//表示有数据交换
    			}
    		}
    		if (!flag) break; //没有数据交换(说明已排好序无需再进行冒泡),提前退出
    	}
    	
    	return a;
    }
    

    四、插入排序

    插入排序将数组数据分成已排序区间和未排序区间。初始已排序区间只有一个元素,即数组第一个元素。在未排序区间取出一个元素插入到已排序区间的合适位置,直到未排序区间为空。

    代码:

    public int[] insertionSort(int[] a) {
    	int n = a.length;
    	if (n<=1) return a;
    	
    	for (int i = 1; i < n; i++) {
    		int value = a[i];
    		int j = i-1;
    		for (; j >=0; j--) {
    			if (a[j] > value) {
    				a[j+1] = a[j];//移动数据
    			}else {
    				break;
    			}
    		}
    		a[j+1] = value;//插入数据
    	}
    	
    	return a;
    }
    

    五、选择排序

    选择排序将数组分成已排序区间和未排序区间。初始已排序区间为空。每次从未排序区间中选出最小的元素插入已排序区间的末尾,直到未排序区间为空。
    代码:

    public int[] selectionSort(int[] a) {
    int n = a.length;

    	for (int i = 0; i < a.length - 1; i++) {
    		for (int j = i+1; j < a.length; j++) {
    			//交换
    			if (a[i] > a[j]) {
    				int temp = a[i];
    				a[i] = a[j];
    				a[j] = temp;
    			}
    		}
    	}
    	
    	return a;
    }
    

    归并排序

    如果要排序一个数组,我们先把数组从中间分成前后两部分,然后对前后两部分分别排序,再将排好序的两部分合并在一起,这样整个数组就都有序了。

    实现思路:

    merge-sort(p…r)表示,给下标从p到r之间的数组排序。我们将这个排序问题转化为了两个子问 ,题, merge_sort(p…q)和merge-sort(q+1…r),其中下标q等于p和r的中间位置,也就是, (p+r)/2,当下标从p到q和从q+1到r这两个子数组都排好序之后,我们再将两个有序的子数组合并在一起,这样下标从p到r之间的数据就也排好序了。

    代码:

    // 归并排序算法, a是数组,n表示数组大小
    public static void mergeSort(int[] a, int n) {
    mergeSortInternally(a, 0, n-1);
    }

    // 递归调用函数
    private static void mergeSortInternally(int[] a, int p, int r) {
    // 递归终止条件
    if (p >= r) return;

    // 取p到r之间的中间位置q
    int q = (p+r)/2;
    // 分治递归
    mergeSortInternally(a, p, q);
    mergeSortInternally(a, q+1, r);
    
    // 将A[p...q]和A[q+1...r]合并为A[p...r]
    merge(a, p, q, r);
    

    }

    private static void merge(int[] a, int p, int q, int r) {
    int i = p;
    int j = q+1;
    int k = 0; // 初始化变量i, j, k
    int[] tmp = new int[r-p+1]; // 申请一个大小跟a[p…r]一样的临时数组

    // 1 排序
    while (i<=q && j<=r) {
      if (a[i] <= a[j]) {
        tmp[k++] = a[i++]; // i++等于i:=i+1
      } else {
        tmp[k++] = a[j++];
      }
    }
    
    // 2 判断哪个子数组中有剩余的数据
    int start = i;
    int end = q;
    if (j <= r) {
      start = j;
      end = r;
    }
    
    // 3 将剩余的数据拷贝到临时数组tmp
    while (start <= end) {
      tmp[k++] = a[start++];
    }
    
    // 4 将tmp中的数组拷贝回a[p...r]
    for (i = 0; i <= r-p; ++i) {
      a[p+i] = tmp[i];
    }
    

    }
    merge是这样执行的:

    代码分析:

    快速排序

    快排的思想:    如果要排序数组中下标从p到r之间的一组数据,我们选择p到r之间的任意一个数据作为pivot (分区点) 。-我们遍历p到r之间的数据,将小于pivot的放到左边,将大于pivot的放到右边,将pivot放到中间。经过这一步骤之后,数组p到r之间的数据就被分成了三个部分,前面p到q-1之间都是小于pivot的,中间是pivot,后面的q+1到r之间是大于pivot的。

    快排利用的分而治之的思想

    线性排序:
    时间复杂度O(n)

    我们把时间复杂度是线性的排序算法叫作线性排序(Linear sort)常见的线性算法有: 桶排序、计数排序、基数排序

    特点:

    非基于比较的排序算法

    桶排序

    桶排序,顾名思义,会用到“桶" ,核心思想是将要排序的数据分到几个有序的桶里,每个桶里的数据再单独进行排序。桶内排完序之后,再把每个桶里的数据按照顺序依次取出,组成的序列就是有序的了。

    对排序的数据要求苛刻:

    1, 要排序的数据需要很容易就能划分成m个桶,并且,桶与桶之间有着天然的大小顺序。

    2 ,数据在各个桶之间的分布是比较均匀的。

    3 ,桶排序比较适合用在外部排序中。所谓的外部排序就是数据存储在外部磁盘中,数据量比较大,内存有限,无法将数据全部加载到内存中。

    计数排序

    计数排序只能用在数据范围不大的场景中,如果数据范围k比要排序的数据n大很多,就不适合用计数排序了。

    计数排序只能给非负整数排序,如果要排序的数据是其他类型的,要将其在不改变相对大小的情况下,转化为非负整数。

    代码:

    // 计数排序,a是数组,n是数组大小。假设数组中存储的都是非负整数。
    public static void countingSort(int[] a) {
    int n = a.length;
    if (n <= 1) return;

    // 查找数组中数据的范围
    int max = a[0];
    for (int i = 1; i < n; ++i) {
      if (max < a[i]) {
        max = a[i];
      }
    }
    
    // 申请一个计数数组c,下标大小[0,max]
    int[] c = new int[max + 1];
    for (int i = 0; i < max + 1; ++i) {
      c[i] = 0;
    }
    
    // 计算每个元素的个数,放入c中
    for (int i = 0; i < n; ++i) {
      c[a[i]]++;
    }
    
    // 依次累加
    for (int i = 1; i < max + 1; ++i) {
      c[i] = c[i-1] + c[i];
    }
    
    // 临时数组r,存储排序之后的结果
    int[] r = new int[n];
    // 计算排序的关键步骤了,有点难理解
    for (int i = n - 1; i >= 0; --i) {
      int index = c[a[i]]-1;
      r[index] = a[i];
      c[a[i]]--;
    }
    
    // 将结果拷贝会a数组
    for (int i = 0; i < n; ++i) {
      a[i] = r[i];
    }
    

    }
    散列表
    什么是散列表:
    散列表用的是数组支持按照下标随机访问数据的特性,所以散列表其实就是数组的一种扩展,由数组演化而来。可以说,如果没有数组,就没有散列表。

    原理:
    散列表用的就是数组支持按照下标随机访问的时候,时间复杂度是0(1)的特性。我们通过散列函数把元素的键值映射为下标,然后将数据存储在数组中对应下标的位置。当我们按照键值查询元素时,我们用同样的散列函数,将键值转化数组标标,从对应的数组下标的位置取数据。

    散列函数的设计要求:
    散列函数计算得到的散列值是一个非负整数;.
    如果key1 = key2,那hash(key1) == hash(key2);
    如果key1 != key2,那hash(key1)  !=  hash(key2),
    散列函数的设计不能太复杂,散列函数生成值要尽可能随机并且均匀分布

    如果不符合3 那么就出现了散列冲突,散列冲突是无法避免的

    解决散列冲突的方法有两种: 
    开放寻址法(open addressing)和链表法(chaining)

    开放寻址法:如果出现了散列冲突,我们就重新探测一个空闲位置,将其插入。

    装在因子:  散列表中一定比例的空闲槽位。公式: 散列表的装载因子 = 填入表中的元素个数 / 散列表的长度

    装载因子越大,说明空闲位置越少,冲突越多,散列表的性能会下降。

    链表法:

    链表法是一种更加常用的散列冲突解决办法,相比开放寻址法,它要简单很多。我们来看这个图,在散列表中,每个"桶(bucket) "或者"槽(slot) "会对应一条链表,所有散列值相同的元素我们都放到相同槽位对应的链表中。

    ————————————————
    版权声明:本文为CSDN博主「吃完喝完嚼益达」的原创文章,遵循CC 4.0 by-sa版权协议,转载请附上原文出处链接及本声明。
    原文链接:https://blog.csdn.net/ityqing/article/details/82838524

    展开全文
  • Java数据结构预算法中级共29节课程序=数据结构+算法,如果要写出优秀的程序,那么数据结构和算法是必须掌握的技能之一。本套课程使用的编程语言为java,将以数据存储结构与相应的算法实现为主线,并对算法的运算效率...

    586859320bad30449bf4a5990b9300ba.png

    Java数据结构预算法

    中级

    共29节课

    程序=数据结构+算法,如果要写出优秀的程序,那么数据结构和算法是必须掌握的技能之一。本套课程使用的编程语言为java,将以数据存储结构与相应的算法实现为主线,并对算法的运算效率做出分析,对典型的线形结构、树形结构、图形结构有详细介绍和代码实现,以及对常见的查找和排序算法做深入探究,每一个知识点都会配套有大量的习题讲解,以加深对所学知识的理解和应用。课程设计以数据结构为主,注重实战应用,强调逻辑思维能力的培养。>。

    20f53ec5c50c21d53051ca5451a329b0.png

    课程简介

    程序=数据结构+算法,如果要写出优秀的程序,那么数据结构和算法是必须掌握的技能之一。本套课程使用的编程语言为java,将以数据存储结构与相应的算法实现为主线,并对算法的运算效率做出分析,对典型的线形结构、树形结构、图形结构有详细介绍和代码实现,以及对常见的查找和排序算法做深入探究,每一个知识点都会配套有大量的习题讲解,以加深对所学知识的理解和应用。课程设计以数据结构为主,注重实战应用,强调逻辑思维能力的培养。

    适用人群

    掌握一门编程语言,最好是java语言,有一定的入门级编程基础。

    课程亮点

    本套课程将会系统的学习有关数据结构和算法的相关知识,从基本的数据结构栈、队列、链表开始,再到算法分析,从程序的时间复杂度和空间复杂度分析算法的性能,然后是经典的算法实现,如选择排序、插入排序、归并排序、快速排序等,最后到高级的数据结构如树、图等的实现。整个课程的讲解思路是先提出问题,然后分析问题并使用暴力算法解题,然后考察算法的各种改进并验证改进的效果,最后再讲解配套的习题,意在使用更优秀的方法解决大规模问题。

    在每一个知识点的讲解过程中配套了大量的图片加深对算法过程以及数据结构的理解,共配图154副。

    89f13428de80cb08afbd313899ed528c.gif

    课程内容

    第一章:

    数据结构与算法概述;

    算法分析;

    冒泡排序;

    选择排序;

    插入排序;

    希尔排序;

    归并排序;

    第二章:

    快速排序;

    排序稳定性分析;

    顺序表;

    链表;

    第三章:

    栈;

    队列;

    符号表;

    二叉查找树;

    第四章:

    堆;

    优先队列;

    2-3查找树;

    红黑树;

    第五章:

    B-树;

    B+树;

    并查集;

    无向图;

    第六章:

    有向图;

    拓扑排序;

    加权无向图;

    最小生成树;

    加权有向图;

    最短路径;

    2c5c5fa6dcbc726a8d1fee1b426bb6b2.png

    展开全文
  • java数据结构预算法

    2015-12-10 23:19:24
    java数据结构与分析指南教程
  • 数据结构预算法分析(Java)。Java语言描述的数据结构,容易看懂...
  • 31 例如 抽象数据类型 复数 的定义 数据对象 D {e1,e2 e1,e2 RealSet } 数据关系 R1 { ,e2> | e1 是复数的实数部分 | e2 是复数的虚数部分 } ADT Complex { 32 基本操作 AssignComplex( &Z, v1, v2 ) 操作结果 构造...
  • 数据结构预算法简述

    2019-07-11 11:24:44
    数据结构是指数据与数据之间的逻辑关系,算法是基于数据结构之上。 1.1 数据结构分为数据物理结构和数据逻辑结构,物理存储结构是指数据在硬盘存放的顺序,逻辑结构是指数据与数据之间关系。 逻辑结构:数据...

    1:简述数据机构与算法:

         数据结构是指数据与数据之间的逻辑关系,算法是基于数据结构之上。 

    1.1 数据结构分为数据物理结构和数据逻辑结构,物理存储结构是指数据在硬盘存放的顺序,逻辑结构是指数据与数据之间关系。

          

    • 逻辑结构:数据对象中的数据元素之间的逻辑关系

           集合结构:集合结构中的数据元素除了同属一个集合外,没有其他关系。
           线性结构:线性结构中的数据元素之间是一对一的关系。
           树形结构:树形结构中的数据元素之间是一对多的关系。
           图形结构:图形结构中的数据元素之间是多对多的关系。

    • 物理机构:数据的逻辑结构在计算机中的储存形式

           顺序储存结构:把数据元素储存在连续的存储单元。
           链式储存结构:把数据元素储存在任意的存储单元(可以是连续或者不连续)。

    1.2 算法的特性:

    1. 有穷性(Finiteness)。算法的有穷性是指算法必须能在执行有限个步骤之后终止;

    2. 确切性(Definiteness)。算法的每一步骤必须有确切的定义;

    3. 输入项(Input)。一个算法有0个或多个输入,以刻画运算对象的初始情况,所谓0个输入是指算法本身定出了初始条件;

    4. 输出项(Output)。一个算法有一个或多个输出,以反映对输入数据加工后的结果。没有输出的算法是毫无意义的;

    5. 可行性(Effectiveness)。算法中执行的任何计算步骤都是可以被分解为基本的可执行的操作步,即每个计算步都可以在有限时间内完成(也称之为有效性)。

    1.2 算法的基本要求:

    1. 正确性。   算法的正确性是指,得算对啊算出来不是自己想要的结果干哈。

    2. 可读性。   写出来的别的得看的懂哈,在如今的互联网时代 随时跳槽离职入职。
    3. 健壮性。   写出来的算法至少平稳运行吧。
    4. 时间复杂度。 执行时间至少在控制范围内吧。
    5. 空间复杂度。 需要的资源至少在控制范围内吧。

       

    展开全文
  • 很经典的教材汪诗林 翻译 数据结构预算法应用C++教材源代码,很难得得到
  • 数据结构预算法Java版

    2015-01-06 13:59:16
    数据结构电子通讯录系统数据数据库创建脚本 \MyAddressList \DB \MyAddressList.bak 电子通讯录系统数据数据库备份 \MyAddressList \Images\ 电子通讯录系统图片文件夹 \MyAddressList \Styles\...
  • 7.1.2 基本术语 1结点包含一个数据元素及若干指向其子树的分支 2结点的度结点拥有子树数目称为结点的度如图7-1-1的b中A的度为3C的度为1M的度为0 3叶子或终端结点度为零的结点 4分支结点或非终端结点除根结点外度不为...
  • 对查找表进行的操作 1查找某个特定的数据元素是否存在 2检索某个特定数据元素的属性 3在查找表中插入一个数据元素 4在查找表中删除一个数据元素 3.静态查找在查找过程中仅查找某个特定元素存在或查找其属性称为静态...
  • 1;2;3;4;5;6;7;8;9;10;... 例 求10个正整数中的最大数max的算法 描述算法的方法有很多流程图自然语言计算机语言用计算机语言表达的算法就是程序 ;若采用自然语言描述,则如下列步骤所示: (1)给10个元素a[0]-
  • 归并排序归并排序是采用分治的一个非常典型的应用。归并排序的思想就是先递归分解数组,再合并数组。将数组分解最小之后,然后合并两个有序数组,基本思路是比较两个数组的最前面的数,谁小就先取谁,取了后相应的...

    归并排序归并排序是采用分治法的一个非常典型的应用。归并排序的思想就是先递归分解数组,再合并数组。将数组分解最小之后,然后合并两个有序数组,基本思路是比较两个数组的最前面的数,谁小就先取谁,取了后相应的指针就往后移一位。然后再比较,直至一个数组为空,最后把另一个数组的剩余部分复制过来即可。
    def merge_sort(alist):
    if len(alist) <= 1:
    return alist
    # 二分分解
    num = len(alist)/2
    left = merge_sort(alist[:num])
    right = merge_sort(alist[num:])
    # 合并
    return merge(left,right)

    def merge(left, right):
    ‘’‘合并操作,将两个有序数组left[]和right[]合并成一个大的有序数组’’’
    #left与right的下标指针
    l, r = 0, 0
    result = []
    while l<len(left) and r<len(right):
    if left[l] < right[r]:
    result.append(left[l])
    l += 1
    else:
    result.append(right[r])
    r += 1
    result += left[l:]
    result += right[r:]
    return result

    alist = [54,26,93,17,77,31,44,55,20]
    sorted_alist = mergeSort(alist)
    print(sorted_alist)
    时间复杂度最优时间复杂度:O(nlogn)最坏时间复杂度:O(nlogn)稳定性:稳定
    搜索搜索是在一个项目集合中找到一个特定项目的算法过程。搜索通常的答案是真的或假的,因为该项目是否存在。 搜索的几种常见方法:顺序查找、二分法查找、二叉树查找、哈希查找二分法查找二分查找又称折半查找,优点是比较次数少,查找速度快,平均性能好;其缺点是要求待查表为有序表,且插入删除困难。因此,折半查找方法适用于不经常变动而查找频繁的有序列表。首先,假设表中元素是按升序排列,将表中间位置记录的关键字与查找关键字比较,如果两者相等,则查找成功;否则利用中间位置记录将表分成前、后两个子表,如果中间位置记录的关键字大于查找关键字,则进一步查找前一子表,否则进一步查找后一子表。重复以上过程,直到找到满足条件的记录,使查找成功,或直到子表不存在为止,此时查找不成功。
    二分法查找实现(非递归实现)def binary_search(alist, item):
    first = 0
    last = len(alist)-1
    while first<=last:
    midpoint = (first + last)/2
    if alist[midpoint] == item:
    return True
    elif item < alist[midpoint]:
    last = midpoint-1
    else:
    first = midpoint+1
    return False
    testlist = [0, 1, 2, 8, 13, 17, 19, 32, 42,]
    print(binary_search(testlist, 3))
    print(binary_search(testlist, 13))
    (递归实现)def binary_search(alist, item):
    if len(alist) == 0:
    return False
    else:
    midpoint = len(alist)//2
    if alist[midpoint]==item:
    return True
    else:
    if item<alist[midpoint]:
    return binary_search(alist[:midpoint],item)
    else:
    return binary_search(alist[midpoint+1:],item)

    testlist = [0, 1, 2, 8, 13, 17, 19, 32, 42,]
    print(binary_search(testlist, 3))
    print(binary_search(testlist, 13))
    时间复杂度最优时间复杂度:O(1)最坏时间复杂度:O(logn)

    展开全文
  • 大牛的数据结构与算法教程音频还有pdf,特此上传,这是一部分,还有另外的一部分
  • 大牛的数据结构与算法教程音频还有pdf,特此上传,这是一部分,还有另外的一部分
  • 数据结构与算法系列 目录 最近抽空整理了"数据结构和算法"的相关文章。在整理过程中,对于每种数据结构和算法分别给出"C"、"C++"和"Java"这三种语言的实现;实现语言虽不同...
  • 题目: 解题算法: class Solution { public: vector<vector<int>> subsets(vector<int>& nums) { int l=nums.size(); int n=1<<l; vector<...
  • 数据(data)结构(structure) 程序=数据结构+算法 数据结构: 线性 :数组 链表 队列 栈 非线性:二位数组 多维数组 广义表 树结构 图结构 稀疏数组和队列(sparsearray)
  • 1.数据结构和算法解决是“如何让计算机更快时间、更省空间的解决问题”。 2.因此需从执行时间和占用空间两个维度来评估数据结构和算法的性能。 3.分别用时间复杂度和空间复杂度两个概念来描述性能问题,二者统称为...
  • 记录一下,该有的注释都有了 package ... import java.util.Arrays; /** * 冒泡排序 * ...public class Bubble { ... public static void main(String[] args) { ... System.out.println( Arrays.toString
  • 数据结构与算法第一课,c++单项链表,用struct结构体做的小程序

空空如也

空空如也

1 2 3 4 5 ... 10
收藏数 190
精华内容 76
关键字:

数据结构预算法

数据结构 订阅