算法与数据结构全阶班-左程云版(二)基础阶段之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

总结

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

相关文章
|
10月前
|
存储 算法 Java
算法系列之递归反转单链表
递归反转链表的基本思路是将当前节点的next指针指向前一个节点,然后递归地对下一个节点进行同样的操作。递归的核心思想是将问题分解为更小的子问题,直到达到基本情况(通常是链表末尾)。
336 5
算法系列之递归反转单链表
|
10月前
|
存储 机器学习/深度学习 算法
C 408—《数据结构》算法题基础篇—链表(下)
408考研——《数据结构》算法题基础篇之链表(下)。
404 30
|
10月前
|
存储 算法 C语言
C 408—《数据结构》算法题基础篇—链表(上)
408考研——《数据结构》算法题基础篇之链表(上)。
503 25
|
11月前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】单链表的基本运算(头歌实践教学平台习题)【合集】
本内容介绍了单链表的基本运算任务,涵盖线性表的基本概念、初始化、销毁、判定是否为空表、求长度、输出、求元素值、按元素值查找、插入和删除数据元素等操作。通过C++代码示例详细解释了顺序表和链表的实现方法,并提供了测试说明、通 - **任务描述**:实现单链表的基本运算。 - **相关知识**:包括线性表的概念、初始化、销毁、判断空表、求长度、输出、求元素值、查找、插入和删除等操作。 - **测试说明**:平台会对你编写的代码进行测试,提供测试输入和预期输出。 - **通关代码**:给出了完整的C++代码实现。 - **测试结果**:展示了测试通过后的预期输出结果。 开始你的任务吧,祝你成功!
558 5
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
1061 9
|
存储 算法
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
310 59
|
6月前
|
编译器 C语言 C++
栈区的非法访问导致的死循环(x64)
这段内容主要分析了一段C语言代码在VS2022中形成死循环的原因,涉及栈区内存布局和数组越界问题。代码中`arr[15]`越界访问,修改了变量`i`的值,导致`for`循环条件始终为真,形成死循环。原因是VS2022栈区从低地址到高地址分配内存,`arr`数组与`i`相邻,`arr[15]`恰好覆盖`i`的地址。而在VS2019中,栈区先分配高地址再分配低地址,因此相同代码表现不同。这说明编译器对栈区内存分配顺序的实现差异会导致程序行为不一致,需避免数组越界以确保代码健壮性。
140 0
栈区的非法访问导致的死循环(x64)
232.用栈实现队列,225. 用队列实现栈
在232题中,通过两个栈(`stIn`和`stOut`)模拟队列的先入先出(FIFO)行为。`push`操作将元素压入`stIn`,`pop`和`peek`操作则通过将`stIn`的元素转移到`stOut`来实现队列的顺序访问。 225题则是利用单个队列(`que`)模拟栈的后入先出(LIFO)特性。通过多次调整队列头部元素的位置,确保弹出顺序符合栈的要求。`top`操作直接返回队列尾部元素,`empty`判断队列是否为空。 两题均仅使用基础数据结构操作,展示了栈与队列之间的转换逻辑。
|
11月前
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
558 77
|
10月前
|
算法 调度 C++
STL——栈和队列和优先队列
通过以上对栈、队列和优先队列的详细解释和示例,希望能帮助读者更好地理解和应用这些重要的数据结构。
261 11

热门文章

最新文章