java有哪些数据结构类型(java数据结构算法设计模式)

java有哪些数据结构类型(java数据结构算法设计模式)

作者 | 张振华.Jack

责编 | 郭芮

出品 | CSDN 博客

本文和大家一起来重温《Java数据结构》经典之作。

Java数据结构

要理解Java数据结构,必须能清楚何为数据结构?

数据结构:

  1. Data_Structure,它是储存数据的一种结构体,在此结构中储存一些数据,而这些数据之间有一定的关系。
  2. 而各数据元素之间的相互关系,又包括三个组成成分,数据的逻辑结构,数据的存储结构和数据运算结构。
  3. 而一个数据结构的设计过程分成抽象层、数据结构层和实现层。

数据结构在Java的语言体系中按逻辑结构可以分为两大类:线性数据结构和非线性数据结构。

一、Java数据结构之:线性数据结构

线性数据结构:常见的有一维数组,线性表,栈,队列,双队列,串。

1、一维数组

在Java里面常用的util有:String ,int ,ArrayList,Vector,CopyOnWriteArrayList等,及可以通过一维数组自己实现不同逻辑结构的Util类,而ArrayList封装了一些的基本操作方法。

ArrayList和Vector的区别是:Vector是线程安全的,方法同步。CopyOnWriteArrayList也是线程安全的但效率要比Vector高很多。

数组这种数据结构典型的操作方法,是根据下标进行操作的,所以insert的的时候可以根据下标插入到具体的某个位置,但是这个时候它后面的元素都得往后面移动一位。所以插入效率比较低,更新,删除效率也比较低,而查询效率非常高,查询效率时间复杂度是1。

2、线性表

线性表是有序的储存结构、链式的储存结构。链表的物理储存空间是不连续的,链表的每一个节点都知道上一个节点、或者下一个节点是谁,通常用Node表示。常见的有顺序链表(LinkedList、Linked***),单项链表(里面只有Node类),双向链表(两个Node类),循环链表(多个Node类)等。

操作方法:插入效率比较高,插入的时候只需要改变节点的前后节点的连接即可。而查询效率就比较低了,如果实现的不好,需要整个链路找下去才能找到应该找的元素。所以常见的方法有:add(index,element),addFirst(element),addLast(element),getFirst,getLast,get(element)等。

常见的Uitil有:LinkedList,LinkedMap等,而这两个JDK底层也做了N多优化,可以有效避免查询效率低的问题,当自己实现的时候需要注意。其实树形结构可以说是非线性的链式储存结构。

3、栈Stack

栈,最主要的是要实现先进后出,后进先出的逻辑结构。来实现一些场景对逻辑顺序的要求。所以常用的方法有push(element)压栈,pop出栈。

java.util.Stack就实现了这用逻辑,而Java的Jvm里面也用的到了此种数据结构,就是线程栈,来保证当前线程的执行顺序。

4、队列

队列,队列是一种特殊的线性数据结构,队列只能允许在队头,队尾进行添加和查询等相关操作。队列又有单项有序队列、双向队列、阻塞队列等。

Queue这种数据结构注定了基本操作方法有:add(E e)加入队列,remove,poll等方法。

队列在Java语言环境中是使用频率相当高的数据结构,所有其实现的类也很多来满足不同场景。

java有哪些数据结构类型(java数据结构算法设计模式)

queue

使用场景也非常多,如线程池、MQ、连接池等。

5、

串:也称字符串,是由N个字符组成的优先序列。在Java里面就是指String,而String里面是由chat来进行储存。

KMP算法: 这个算法一定要牢记,Java数据结构这本书里面针对字符串的查找匹配算法也只介绍了一种。关键点就是:在字符串比对的时候,主串的比较位置不需要回退的问题。

二、Java数据结构之:非线性数据结构

非线性数据结构:常见的有:多维数组,集合,树,图,散列表(hash)。

1、多维数组

一维数组前面咱也提到了,多维数组无非就是String ,int等。Java里面很少提供这样的工具类,而Java里面tree和图底层的native方法用了多维数组来储存。

2、集合

由一个或多个确定的元素所构成的整体叫做集合,在Java里面可以去广义的去理解为实现了Collection接口的类都叫集合。

java有哪些数据结构类型(java数据结构算法设计模式)

Collection

3、

树形结构,作者觉得它是一种特殊的链形数据结构。最少有一个根节点组成,可以有多个子节点。树,显然是由递归算法组成。

树的特点:

  1. 在一个树结构中,有且仅有一个结点没有直接父节点,它就是根节点;
  2. 除了根节点,其他结点有且只有一个直接父节点;
  3. 每个结点可以有任意多个直接子节点。

树的数据结构又分如下几种:

1) 自由树/普通树:对子节点没有任何约束。

java有哪些数据结构类型(java数据结构算法设计模式)

自由树

2) 二叉树:每个节点最多含有两个子节点的树称为二叉树。

2.1) 一般二叉树:每个子节点的父亲节点不一定有两个子节点的二叉树成为一般二叉树。

2.2) 完全二叉树:对于一颗二叉树,假设其深度为d(d>1)。除了第d层外,其它各层的节点数目均已达最大值,且第d层所有节点从左向右连续地紧密排列,这样的二叉树被称为完全二叉树;

2.3) 满二叉树:所有的节点都是二叉的二叉树成为满二叉树。

java有哪些数据结构类型(java数据结构算法设计模式)

二叉树

3) 二叉搜索树/BST:binary search tree,又称二叉排序树、二叉查找树。是有序的。要点:如果不为空,那么其左子树节点的值都小于根节点的值;右子树节点的值都大于根节点的值。

java有哪些数据结构类型(java数据结构算法设计模式)

二叉搜索

3.1) 二叉平衡树:二叉搜索树,是有序的排序树,但左右两边包括子节点不一定平衡,而二叉平衡树是排序树的一种,并且加点条件,就是任意一个节点的两个叉的深度差不多(比如差值的绝对值小于某个常数,或者一个不能比另一个深出去一倍之类的)。这样的树可以保证二分搜索任意元素都是O(log n)的,一般还附带带有插入或者删除某个元素也是O(log n)的的性质。

为了实现,二叉平衡树又延伸出来了一些算法,业界常见的有AVL、和红黑算法,所以又有以下两种树:

3.1.1) AVL树:最早的平衡二叉树之一。应用相对其他数据结构比较少。windows对进程地址空间的管理用到了AVL树。

3.1.2) 红黑树:通过制定了一些红黑标记和左右旋转规则来保证二叉树平衡。

java有哪些数据结构类型(java数据结构算法设计模式)

红黑树

红黑树的5条性质:

  1. 每个结点要么是红的,要么是黑的。
  2. 根结点是黑的。
  3. 每个叶结点(叶结点即指树尾端NIL指针或结点)是黑的。
  4. 如果一个结点是红的,那么它的俩个儿子都是黑的。
  5. 对于任一结点而言,其到叶结点树尾端NIL指针的每一条路径都包含相同数目的黑结点。

4) B-tree:又称B树、B-树。又叫平衡(balance)多路查找树。树中每个结点最多含有m个孩子(m>=2)。它类似普通的平衡二叉树,不同的一点是B-树允许每个节点有更多的子节点。

java有哪些数据结构类型(java数据结构算法设计模式)

B-tree

4) B+tree:又称B+。是B-树的变体,也是一种多路搜索树。

java有哪些数据结构类型(java数据结构算法设计模式)

B+tree

树总结: 树在Java里面应用的也比较多。非排序树,主要用来做数据储存和展示。而排序树,主要用来做算法和运算,HashMap里面的TreeNode就用到了红黑树算法。而B+树在数据库的索引原理里面有典型的应用。

4、Hash

Hash概念:

  • Hash,一般翻译做“散列”,也有直接音译为“哈希”的,就是把任意长度的输入(又叫做预映射, pre-image),变换成固定长度的输出,该输出就是散列值。一般通过Hash算法实现。
  • 所谓的Hash算法都是散列算法,把任意长度的输入,变换成固定长度的输出,该输出就是散列值(如:MD5,SHA1,加解密算法等)。
  • 简单的说就是一种将任意长度的消息压缩到某一固定长度的消息摘要的函数。

Java中的hashCode:

  • 我们都知道所有的class都是Object的子类,既所有的class都会有默认Object.java里面的hashCode的方法,如果自己没有重写,默认情况就是native方法通过对象的内存的+对象的值然后通过hash散列算法计算出来个int的数字。
  • 最大的特性是:不同的对象,不同的值有可能计算出来的hashCode可能是一样的。

Hash表:

Java中数据存储方式最底层的两种结构,一种是数组,另一种就是链表。而Hash表就是综合了这两种数据结构。

如:HashTable,HashMap。这个时候就得提一下HashMap的原理了,默认16个数组储存,通过Hash值取模放到不同的桶里面去。(注意:JDK1.8此处算法又做了改进,数组里面的值会演变成树形结构。)

哈希表具有较快(常量级)的查询速度,及相对较快的增删速度,所以很适合在海量数据的环境中使用。一般实现哈希表的方法采用“拉链法”,我们可以理解为“链表的数组”。

java有哪些数据结构类型(java数据结构算法设计模式)

哈希表

一致性Hash:

我们查看一下HashMap的原理,其实发现Hash很好的解决了单体应用情况下的数据查找和插入的速度问题。但是毕竟单体应用的储存空间是有限的,所有在分布式环境下,应运而生了一致性Hash算法。

用意和hashCode的用意一样,只不过它是取模放在不同的IP机器上而已。具体算法可以找一下相关资料。

而一致性Hash需要注意的就是默认分配的桶比较多些,而当其中一台机器挂了,影响的面比较小一些。

需要注意的是,相同的内容算出来的hash一定是一样的。既:幂等性。

java有哪些数据结构类型(java数据结构算法设计模式)

一致性Hash

Java基本算法

理解了Java数据结构,还必须要掌握一些常见的基本算法。理解算法之前必须要先理解的几个算法的概念:

空间复杂度:一句来理解就是,此算法在规模为n的情况下额外消耗的储存空间。

时间复杂度:一句来理解就是,此算法在规模为n的情况下,一个算法中的语句执行次数称为语句频度或时间频度。

稳定性:主要是来描述算法,每次执行完,得到的结果都是一样的,但是可以不同的顺序输入,可能消耗的时间复杂度和空间复杂度不一样。

一、二分查找算法

二分查找又称折半查找,优点是比较次数少,查找速度快,平均性能好,占用系统内存较少;其缺点是要求待查表为有序表,且插入删除困难。这个是基础,最简单的查找算法了。

public static void main(String[] args) {int srcArray = {3,5,11,17,21,23,28,30,32,50,64,78,81,95,101};System.out.println(binSearch(srcArray, 28));}/*** 二分查找普通循环实现** @param srcArray 有序数组* @param key 查找元素* @return*/public static int binSearch(int srcArray[], int key) {int mid = srcArray.length / 2;// System.out.println("=:"+mid);if (key == srcArray[mid]) {return mid;}//二分核心逻辑int start = 0;int end = srcArray.length - 1;while (start <= end) {// System.out.println(start+"="+end);mid = (end - start) / 2 + start;if (key < srcArray[mid]) {end = mid - 1;} else if (key > srcArray[mid]) {start = mid + 1;} else {return mid;}}return -1;}

二分查找算法如果没有用到递归方法的话,只会影响CPU。对内存模型来说影响不大。时间复杂度log2n,2的开方。空间复杂度是2。一定要牢记这个算法。应用的地方也是非常广泛,平衡树里面大量采用。

二、递归算法

递归简单理解就是方法自身调用自身。

public static void main(String[] args) {int srcArray = {3,5,11,17,21,23,28,30,32,50,64,78,81,95,101};System.out.println(binSearch(srcArray, 0,15,28));}/*** 二分查找递归实现** @param srcArray 有序数组* @param start 数组低地址下标* @param end 数组高地址下标* @param key 查找元素* @return 查找元素不存在返回-1*/public static int binSearch(int srcArray[], int start, int end, int key) {int mid = (end - start) / 2 + start;if (srcArray[mid] == key) {return mid;}if (start >= end) {return -1;} else if (key > srcArray[mid]) {return binSearch(srcArray, mid + 1, end, key);} else if (key < srcArray[mid]) {return binSearch(srcArray, start, mid - 1, key);}return -1;}

递归几乎会经常用到,需要注意的一点是:递归不光影响的CPU。JVM里面的线程栈空间也会变大。所以当递归的调用链长的时候需要-Xss设置线程栈的大小。

三、八大排序算法

  • 直接插入排序(Insertion Sort)
  • 希尔排序(Shell Sort)
  • 选择排序(Selection Sort)
  • 堆排序(Heap Sort)
  • 冒泡排序(Bubble Sort)
  • 快速排序(Quick Sort)
  • 归并排序(Merging Sort)
  • 基数排序(Radix Sort)

八大算法,网上的资料就比较多了。吐血推荐参考资料:git hub 八大排序算法详解(
https://itimetraveler.github.io/index.html)。此大神比作者讲解的还详细,作者就不在这里,描述重复的东西了,作者带领大家把重点的两个强调一下,此两个是必须要掌握的。

1、冒泡排序

基本思想:

冒泡排序(Bubble Sort)是一种简单的排序算法。它重复地走访过要排序的数列,一次比较两个元素,如果他们的顺序错误就把他们交换过来。走访数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成。这个算法的名字由来是因为越小的元素会经由交换慢慢“浮”到数列的顶端。

以下是冒泡排序算法复杂度:

平均时间复杂度最好情况最坏情况空间复杂度
O(n²)O(n)O(n²)O(1)

冒泡排序是最容易实现的排序, 最坏的情况是每次都需要交换, 共需遍历并交换将近n²/2次, 时间复杂度为O(n²). 最佳的情况是内循环遍历一次后发现排序是对的, 因此退出循环, 时间复杂度为O(n). 平均来讲, 时间复杂度为O(n²). 由于冒泡排序中只有缓存的temp变量需要内存空间, 因此空间复杂度为常量O(1).

Tips:由于冒泡排序只在相邻元素大小不符合要求时才调换他们的位置, 它并不改变相同元素之间的相对顺序, 因此它是稳定的排序算法。

/*** 冒泡排序** ①. 比较相邻的元素。如果第一个比第二个大,就交换他们两个。* ②. 对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对。这步做完后,最后的元素会是最大的数。* ③. 针对所有的元素重复以上的步骤,除了最后一个。* ④. 持续每次对越来越少的元素重复上面的步骤①~③,直到没有任何一对数字需要比较。* @param arr 待排序数组*/public static void bubbleSort(int[] arr){for (int i = arr.length; i > 0; i--) { //外层循环移动游标for(int j = 0; j < i && (j+1) < i; j++){ //内层循环遍历游标及之后(或之前)的元素if(arr[j] > arr[j+1]){int temp = arr[j];arr[j] = arr[j+1];arr[j+1] = temp;System.out.println("Sorting: " + Arrays.toString(arr));}}}}

2、快速排序

java有哪些数据结构类型(java数据结构算法设计模式)

快速排序

快速排序使用分治策略来把一个序列(list)分为两个子序列(sub-lists)。步骤为:

①从数列中挑出一个元素,称为”基准”(pivot)。

②重新排序数列,所有比基准值小的元素摆放在基准前面,所有比基准值大的元素摆在基准后面(相同的数可以到任一边)。在这个分区结束之后,该基准就处于数列的中间位置。这个称为分区(partition)操作。

③递归地(recursively)把小于基准值元素的子数列和大于基准值元素的子数列排序。

递归到最底部时,数列的大小是零或一,也就是已经排序好了。这个算法一定会结束,因为在每次的迭代(iteration)中,它至少会把一个元素摆到它最后的位置去。

代码实现:

用伪代码描述如下:

①i = L; j = R; 将基准数挖出形成第一个坑a[i]。

②j–,由后向前找比它小的数,找到后挖出此数填前一个坑a[i]中。

③i++,由前向后找比它大的数,找到后也挖出此数填到前一个坑a[j]中。

④再重复执行②,③二步,直到i==j,将基准数填入a[i]中。

快速排序采用“分而治之、各个击破”的观念,此为原地(In-place)分区版本。

java有哪些数据结构类型(java数据结构算法设计模式)

快速排序 In-place

/*** 快速排序(递归)** ①. 从数列中挑出一个元素,称为"基准"(pivot)。* ②. 重新排序数列,所有比基准值小的元素摆放在基准前面,所有比基准值大的元素摆在基准后面(相同的数可以到任一边)。在这个分区结束之后,该基准就处于数列的中间位置。这个称为分区(partition)操作。* ③. 递归地(recursively)把小于基准值元素的子数列和大于基准值元素的子数列排序。* @param arr 待排序数组* @param low 左边界* @param high 右边界*/public static void quickSort(int[] arr, int low, int high){if(arr.length <= 0) return;if(low >= high) return;int left = low;int right = high;int temp = arr[left]; //挖坑1:保存基准的值while (left < right){while(left < right && arr[right] >= temp){ //坑2:从后向前找到比基准小的元素,插入到基准位置坑1中right--;}arr[left] = arr[right];while(left < right && arr[left] <= temp){ //坑3:从前往后找到比基准大的元素,放到刚才挖的坑2中left++;}arr[right] = arr[left];}arr[left] = temp; //基准值填补到坑3中,准备分治递归快排System.out.println("Sorting: " + Arrays.toString(arr));quickSort(arr, low, left-1);quickSort(arr, left+1, high);}

以下是快速排序算法复杂度:

平均时间复杂度最好情况最坏情况空间复杂度
O(nlog₂n)O(nlog₂n)O(n²)O(1)(原地分区递归版)

快速排序排序效率非常高。虽然它运行最糟糕时将达到O(n²)的时间复杂度, 但通常平均来看, 它的时间复杂为O(nlogn), 比同样为O(nlogn)时间复杂度的归并排序还要快. 快速排序似乎更偏爱乱序的数列, 越是乱序的数列, 它相比其他排序而言, 相对效率更高。

最后,作者希望让大家对《Java数据结构》整体有个全面的了解,知道什么是数据结构,离我们工作中有多远,而不是一个深不可测的神秘物件。里面的细节,篇幅有限可能不能描述完,但是只要同学们的方向没有搞错,那只要针对每个点再详细的看看即可。

面试和工作,这些都是离不开的,当同学们有个完整的认识之后,一定要在工作中留心,留意每个用到的地方。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人,不代表IT壹频道。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。转载请注明出处:https://www.1pindao.com/baike/5052.html

(0)
上一篇 2022年 11月 15日 13:39:37
下一篇 2022年 11月 15日 13:51:11

相关推荐

  • 新生儿男女比例2021中国(新生儿男女比例真实数据最新)

    众所周知,这几年我国的出生率不太理想,新生儿从2017年的1723万下降到2020年的1200万,连续4年下降,人口警报拉响。国家也意识到这个问题,于是在2021年5月宣布全面放开三孩,鼓励大家多生育孩子。 但是效果似乎并不明显,前几日,国家统计局发布数据,2021年出生人口1062万人,比2020年少生138万,人口出生率为7.52‰,创下新低,完成五连降…

    2022年 12月 18日 百科知识
    1.3K
  • 租车平台哪个好(租车自驾超全面避坑攻略)

    本内容来源于@什么值得买APP,观点仅代表作者本人 |作者:讨我喜欢 刚进入2022年,工作就没了。正好趁这个时间旅游了一大圈,全程租车自驾,两个月也跑了小一万公里。今天给大家分享下租车自驾的超全面避坑攻略,新手一定要看哟! 上面是这两个月租的三辆车,从左到右分别是一嗨云南一个月、携程海南一个月和这两天还在租的浙江十来天。一路上除了跨省大交通没自驾,其他都是…

    2022年 11月 26日 百科知识
    233
  • 抖音音乐人怎么申请(2022认证抖音音乐人详细教程)

    最近几天快手平台对音乐人进行了升级,很多人快手小伙伴着急了,因为他们的音乐标识被取消,找了半天原因,最终确定是平台在升级,意思就是对于不活跃的用户只保留音乐人的权利不保留音乐人保识,需要根据音乐人后续表现是否重新决定是否给予重新点亮,快手音乐人从之前的翻唱就可以入住到现在的需要看音乐人的表现来入住,于是很多小伙伴准备转战抖音平台,那么抖音音乐人怎么申请审核才…

    2022年 11月 23日 百科知识
    228
  • 品牌传播策略有哪些(3招打造品牌传播策略)

    在我们确定目标之后,就要建立具体的策略信息了,这一块在我们日常中是非常重要的一块。传播策略是干什么的呢?其实就干3个事。 说什么,就是要建立内容策略; 如何说,就是要确定传播方式和手段,也就是未来你在选择渠道的时候你要考虑在这个时候哪些方式更加适合你; 谁来说,就是确定信息来源,这个在数字时代就特别有意义,因为在数字时代,你传播的这汇总信息未必是你企业单独创…

    2022年 12月 24日 百科知识
    124
  • 如何听声音辨别歌曲(手机听歌识曲方法教程)

    大家好,欢迎来到科技熊! 最近接到好多朋友的留言,比如说我们在刷手机的时候听到一些音乐或者视频里的音乐非常的好,想收藏起来,但不知道是什么歌曲。那么今天给大家分享的是我们手机实际有一个听歌识曲的功能。 比如说我们在看视频时,里边有一些音乐,手机就可以自动去识别是什么音乐,可以说非常实用,下面我们一起来看一下该如何去操作。 准备工作 首先我们做一下准备工作,当…

    2022年 12月 19日 百科知识
    197
  • 女生发kiss是什么意思(网络热梗kisskiss的含义及出处)

    “Kiss Kiss”是什么梗? “Kiss Kiss”是什么梗?最近小伙伴经常在网络上看到过kiss kiss这个梗吧,Kiss本来作为英文单词翻译过来的意思是一种亲密的接触亲吻的意思,但是在网络上又变成了一种新的意思,今天小编给大家带来“Kiss Kiss”梗的含义及出处,感兴趣的小伙伴们一起来看看吧! “Kiss Kiss”梗含义: 这个梗在网络表示暂…

    2023年 1月 13日 百科知识
    142