算法与数据结构全阶班-左程云版(二)基础阶段之2.链表、栈、队列、递归行为、哈希表和有序表(下)

简介: 本文主要介绍了一些常用的数据结构,包括链表、栈、队列、递归、哈希表和有序表。

用栈实现队列:

也是用两个栈来实现,包括push栈和pop栈,如下:

2345_image_file_copy_109.jpg

遵循的原则:

pop栈为空时,才能将数据导入到pop栈中;

push栈导数据到pop栈时,一次导完。

实现如下:

static class TwoStackQueue {
    private final Stack<Integer> stackPush;
    private final Stack<Integer> stackPop;
    public TwoStackQueue() {
        stackPush = new Stack<>();
        stackPop = new Stack<>();
    }
    // push栈向pop栈导入数据
    private void pushToPop() {
        if (stackPop.isEmpty()) {
            while (!stackPush.isEmpty()) {
                stackPop.push(stackPush.pop());
            }
        }
    }
    public void add(int num) {
        stackPush.push(num);
        pushToPop();
    }
    public int poll() {
        if (stackPush.isEmpty() && stackPop.isEmpty()) {
            throw new RuntimeException("Queue is empty!");
        }
        pushToPop();
        return stackPop.pop();
    }
    public int peek() {
        if (stackPush.isEmpty() && stackPop.isEmpty()) {
            throw new RuntimeException("Queue is empty!");
        }
        pushToPop();
        return stackPop.peek();
    }
}

面试:

一般情况下,图的宽度优先遍历使用的是队列,深度优先遍历使用的是栈;

如果要让用栈实现宽度优先遍历、队列实现深度优先遍历,则需要用两个栈实现队列、两个队列实现栈。

3.递归

怎么从思想上理解递归;

怎么从实际实现的角度出发理解递归。

递归的思想是将一个大的任务分解成小的任务,小的任务再进一步拆解,直到达到某一个边界,最后经过某种逻辑进行整合,得到整个问题的求解。

递归调用时,使用了系统栈,调用的函数放入栈,执行完返回时就会从栈顶pop出;

任何递归都可以改为迭代。

有一类递归的时间复杂度可以确定,时间复杂度原始形式如下:

2345_image_file_copy_111.jpg

即条件为:子问题的规模是一致的

时间复杂度公式如下:

2345_image_file_copy_112.jpg

2345_image_file_copy_113.jpg

例题:求数组arr[L…R]中的最大值,怎么用递归方法实现。

分析思路:

1)将[L…R]范围分成左右两半。左:[L…Mid]右[Mid+1…R];

2)左部分求最大值,右部分求最大值;

3)[L…R]范围上的最大值,是max{左部分最大值,右部分最大值}。

注意:2)是个递归过程,当范围上只有一个数,就可以不用再递归了。

思路如下:

2345_image_file_copy_114.jpg

这里的子问题规模是原问题的一半,所以可以求出时间复杂度,时间复杂度如下:

2345_image_file_copy_115.jpg2345_image_file_copy_116.jpg

实现如下:

public class MaxWithRecursion {
    public static int getMax(int[] arr) {
        return process(arr, 0, arr.length - 1);
    }
    // arr[L..R]范围上求最大值
    public static int process(int[] arr, int L, int R) {
        // arr[L..R]范围上只有1个数,直接返回
        if (L == R) {
            return arr[L];
        }
        int mid = L + ((R - L) >> 1);           // 中点
        int leftMax = process(arr, L, mid);
        int rightMax = process(arr, mid + 1, R);
        return Math.max(leftMax, rightMax);
    }
}

4.哈希表和有序表

哈希表

1)哈希表在使用层面上可以理解为—种集合结构;

2)如果只有key、没有伴随数据value,可以使用HashSet结构;

3)如果既有key,又有伴随数据value,可以使用HashMap结构;

4)有无伴随数据,是HashMap和HashSet唯一的区别,实际结构是一回事;

5)使用哈希表增(put)、删(remove)、改(put)和查(get)的操作,可以认为时间复杂度为O(1),但是常数时间比较大;

6)放入哈希表的元素,如果是基础类型,内部按值传递,内存占用是这个元素的大小;

7)放入哈希表的元素,如果不是基础类型(是自定义类型),内部按引用传递,内存占用是8字节(也就是引用地址的大小)。

使用如下:

static class Node {
    public int value;
    public LinkedList.Node next;
    public Node(int data) {
        value = data;
    }
}
public static void testHashMap() {
    // 键值对
    HashMap<Integer, String> map = new HashMap<>();
    // 新增
    map.put(1, "one");
    map.put(2, "two");
    map.put(3, "three");
    map.put(4, "four");
    map.put(5, "five");
    System.out.println(map.containsKey(1));
    System.out.println(map.containsKey(10));
    System.out.println(map.containsValue("one"));
    System.out.println(map.containsValue("one"));
    System.out.println(map.get(1));
    System.out.println(map.get(10));
    // 修改
    map.put(1, "一");
    System.out.println(map.get(1));
    map.remove(5);
    System.out.println(map.get(5));
    System.out.println("--------------------");
    // 集合元素
    HashSet<String> set = new HashSet<>();
    set.add("one");
    System.out.println(set.contains("one"));
    set.remove("one");
    System.out.println(set.contains("one"));
    System.out.println("--------------------");
    // 引用传递和值传递
    int a = 10000000;
    int b = 10000000;
    System.out.println(a == b);         // int属于基本数据类型,使用值传递,所以比较时只比较值,不比较地址
    Integer c = 10000000;
    Integer d = 10000000;
    System.out.println(c == d);         // Integer属于引用数据类型,使用引用传递,所以比较时比较地址,不比较值,要比较值使用equals方法
    System.out.println(c.equals(d));
    Integer e = 128;
    Integer f = 128;
    System.out.println(e == f);
    e = 127;
    f = 127;
    System.out.println(e == f);         // -128到127之间,使用常量池,转化为值传递,不在这个区间才使用引用传递
    // 新增
    map.put(10000, "10000");
    System.out.println(map.get(10000));
    // 修改
    map.put(10000, "一万");
    System.out.println(map.get(10000)); // 放入哈希表的元素,如果是基础类型,内部按值传递
    HashMap<Node, String> map2 = new HashMap<>();
    Node node1 = new Node(1);
    Node node2 = new Node(1);
    map2.put(node1, "node1");
    map2.put(node2, "node1 too");
    System.out.println(map2.get(node1));
    System.out.println(map2.get(node2));
    node2 = node1;
    System.out.println(map2.get(node2));
}

输出:

true
false
true
true
one
null
null
--------------------
true
false
--------------------
true
false
true
false
true
10000
一万
node1
node1 too
node1

有序表:

有序表指集合内的元素按照某种规则进行排序;

使用哈希表增(put)、删(remove)、改(put)和查(get)的操作,时间复杂度为O(logN)

AVL、SB树、红黑树和跳表都可以实现有序表,通过调整实现平衡。

使用有序表TreeMap如下:

public static void testTreeMap() {
    TreeMap<Integer, String> map = new TreeMap<>();
    map.put(3, "three");
    map.put(7, "seven");
    map.put(2, "two");
    map.put(1, "one");
    map.put(5, "five");
    map.put(9, "nine");
    map.put(6, "six");
    System.out.println(map.containsKey(1));
    System.out.println(map.containsKey(10));
    System.out.println(map.get(1));
    System.out.println(map.get(10));
    // 修改
    map.put(5, "五");
    System.out.println(map.get(5));
    // 删除
    map.remove(5);
    System.out.println(map.get(5));
    // 查看顺序
    System.out.println(map.firstKey());
    System.out.println(map.lastKey());
    System.out.println(map);
    // 小于等于5的最近的值
    System.out.println(map.floorKey(5));
    // 大于等于5的最近的值
    System.out.println(map.ceilingKey(5));
}

输出:

true
false
one
null
null
1
9
{1=one, 2=two, 3=three, 6=six, 7=seven, 9=nine}
3
6

总结

有很多常用的数据结构,每种数据结构都有适用的场景,可以根据自己的需要进行选择。

相关文章
|
26天前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
122 9
|
17天前
|
存储 算法
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
22 1
|
2月前
|
存储 人工智能 算法
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
这篇文章详细介绍了Dijkstra和Floyd算法,这两种算法分别用于解决单源和多源最短路径问题,并且提供了Java语言的实现代码。
83 3
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
|
4天前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
25 5
|
20天前
|
存储 算法 Java
数据结构的栈
栈作为一种简单而高效的数据结构,在计算机科学和软件开发中有着广泛的应用。通过合理地使用栈,可以有效地解决许多与数据存储和操作相关的问题。
|
23天前
|
存储 JavaScript 前端开发
执行上下文和执行栈
执行上下文是JavaScript运行代码时的环境,每个执行上下文都有自己的变量对象、作用域链和this值。执行栈用于管理函数调用,每当调用一个函数,就会在栈中添加一个新的执行上下文。
|
25天前
|
存储
系统调用处理程序在内核栈中保存了哪些上下文信息?
【10月更文挑战第29天】系统调用处理程序在内核栈中保存的这些上下文信息对于保证系统调用的正确执行和用户程序的正常恢复至关重要。通过准确地保存和恢复这些信息,操作系统能够实现用户模式和内核模式之间的无缝切换,为用户程序提供稳定、可靠的系统服务。
47 4
|
29天前
|
算法 安全 NoSQL
2024重生之回溯数据结构与算法系列学习之栈和队列精题汇总(10)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第3章之IKUN和I原达人之数据结构与算法系列学习栈与队列精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2月前
|
存储 算法 Java
Set接口及其主要实现类(如HashSet、TreeSet)如何通过特定数据结构和算法确保元素唯一性
Java Set因其“无重复”特性在集合框架中独树一帜。本文解析了Set接口及其主要实现类(如HashSet、TreeSet)如何通过特定数据结构和算法确保元素唯一性,并提供了最佳实践建议,包括选择合适的Set实现类和正确实现自定义对象的hashCode()与equals()方法。
35 4
|
2月前
数据结构(栈与列队)
数据结构(栈与列队)
20 1