算法与数据结构全阶班-左程云版(二)基础阶段之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天前
|
存储 人工智能 C语言
数据结构基础详解(C语言): 栈的括号匹配(实战)与栈的表达式求值&&特殊矩阵的压缩存储
本文首先介绍了栈的应用之一——括号匹配,利用栈的特性实现左右括号的匹配检测。接着详细描述了南京理工大学的一道编程题,要求判断输入字符串中的括号是否正确匹配,并给出了完整的代码示例。此外,还探讨了栈在表达式求值中的应用,包括中缀、后缀和前缀表达式的转换与计算方法。最后,文章介绍了矩阵的压缩存储技术,涵盖对称矩阵、三角矩阵及稀疏矩阵的不同压缩存储策略,提高存储效率。
|
12天前
|
存储 C语言
数据结构基础详解(C语言): 栈与队列的详解附完整代码
栈是一种仅允许在一端进行插入和删除操作的线性表,常用于解决括号匹配、函数调用等问题。栈分为顺序栈和链栈,顺序栈使用数组存储,链栈基于单链表实现。栈的主要操作包括初始化、销毁、入栈、出栈等。栈的应用广泛,如表达式求值、递归等场景。栈的顺序存储结构由数组和栈顶指针构成,链栈则基于单链表的头插法实现。
|
13天前
|
Java
【数据结构】栈和队列的深度探索,从实现到应用详解
本文介绍了栈和队列这两种数据结构。栈是一种后进先出(LIFO)的数据结构,元素只能从栈顶进行插入和删除。栈的基本操作包括压栈、出栈、获取栈顶元素、判断是否为空及获取栈的大小。栈可以通过数组或链表实现,并可用于将递归转化为循环。队列则是一种先进先出(FIFO)的数据结构,元素只能从队尾插入,从队首移除。队列的基本操作包括入队、出队、获取队首元素、判断是否为空及获取队列大小。队列可通过双向链表或数组实现。此外,双端队列(Deque)支持两端插入和删除元素,提供了更丰富的操作。
16 0
【数据结构】栈和队列的深度探索,从实现到应用详解
|
17天前
|
Linux C++ Windows
栈对象返回的问题 RVO / NRVO
具名返回值优化((Name)Return Value Optimization,(N)RVO)是一种优化机制,在函数返回对象时,通过减少临时对象的构造、复制构造及析构调用次数来降低开销。在C++中,通过直接在返回位置构造对象并利用隐藏参数传递地址,可避免不必要的复制操作。然而,Windows和Linux上的RVO与NRVO实现有所不同,且接收栈对象的方式也会影响优化效果。
|
19天前
crash —— 获取内核地址布局、页大小、以及栈布局
crash —— 获取内核地址布局、页大小、以及栈布局
|
20天前
|
存储 程序员 C语言
堆和栈之间有什么区别
【9月更文挑战第1天】堆和栈之间有什么区别
89 0
|
29天前
|
机器学习/深度学习 消息中间件 缓存
栈与队列的实现
栈与队列的实现
37 0
|
16天前
|
算法 BI Serverless
基于鱼群算法的散热片形状优化matlab仿真
本研究利用浴盆曲线模拟空隙外形,并通过鱼群算法(FSA)优化浴盆曲线参数,以获得最佳孔隙度值及对应的R值。FSA通过模拟鱼群的聚群、避障和觅食行为,实现高效全局搜索。具体步骤包括初始化鱼群、计算适应度值、更新位置及判断终止条件。最终确定散热片的最佳形状参数。仿真结果显示该方法能显著提高优化效率。相关代码使用MATLAB 2022a实现。
|
16天前
|
算法 数据可视化
基于SSA奇异谱分析算法的时间序列趋势线提取matlab仿真
奇异谱分析(SSA)是一种基于奇异值分解(SVD)和轨迹矩阵的非线性、非参数时间序列分析方法,适用于提取趋势、周期性和噪声成分。本项目使用MATLAB 2022a版本实现从强干扰序列中提取趋势线,并通过可视化展示了原时间序列与提取的趋势分量。代码实现了滑动窗口下的奇异值分解和分组重构,适用于非线性和非平稳时间序列分析。此方法在气候变化、金融市场和生物医学信号处理等领域有广泛应用。
|
17天前
|
资源调度 算法
基于迭代扩展卡尔曼滤波算法的倒立摆控制系统matlab仿真
本课题研究基于迭代扩展卡尔曼滤波算法的倒立摆控制系统,并对比UKF、EKF、迭代UKF和迭代EKF的控制效果。倒立摆作为典型的非线性系统,适用于评估不同滤波方法的性能。UKF采用无迹变换逼近非线性函数,避免了EKF中的截断误差;EKF则通过泰勒级数展开近似非线性函数;迭代EKF和迭代UKF通过多次迭代提高状态估计精度。系统使用MATLAB 2022a进行仿真和分析,结果显示UKF和迭代UKF在非线性强的系统中表现更佳,但计算复杂度较高;EKF和迭代EKF则更适合维数较高或计算受限的场景。