数据结构与算法(一):基础数据结构 算法概念、数组、链表、栈、队列

news2025/1/22 17:04:11

判断一个数是否是2的N次方?

N & (N-1) == 0  (N > 0)

算题:

  • 力扣 https://leetcode.cn/
  • POJ http://poj.org/

算法

算法概念

算法代表: 高效率和低存储 内存占用小、CPU占用小、运算速度快

算法的高效率与低存储:内存 + CPU

评价算法的两个指标

  • 时间复杂度: 运行一个程序所花费的时间 O()
  • 空间复杂度: 运行程序所需要的内存 OOM

时间复杂度 O(1,logn,n,nlogn,n^2,n^x)

  • 常数:O(1) 1 表示是常数,所有能确定的数字我们都用O(1),O(10000)=>O(1)
  • 对数: O(logn), O(nlogn)
  • 线性: O(N) n一定是未知的; 如果n是已知的O(1)
  • 线性对数: O(nlogn)
  • 平方: O(n^2)
  • N次方: O(n^n)

如何找时间复杂度

  • 有循坏的地方
  • 有网络请求的地方 (RPC、远程调用、分布式、数据库)

测试时间打印log

在这里插入图片描述

O(1) > O(logn) > O(n) > O(nlogn) > O(n^2) > O(n^n)

越接近O(1)时间复杂度越低

常数:O(1)

int a = 1; // 1次O(1)
for(int i = 0; i < 3; i++) { //这里运行4次
  a = a + 1; //这里运行3次
}

对数: O(logn), O(nlogn)

//对数 2^x=n  x就是我们运行的次数 => 对数 x=log2(n) = log2n = 计算机忽略常数 => logn => O(logn)
int n = Integer.MAX_VALUE;
int i = 1;

//O(logn)
while (i <= n) {
    i = i * 2;
}

//O(nlogn)
for (int j = 0; j < n; j++) {
    while (i <= n) {
        i = i * 3;
    }
}

线性: O(N)

//线性: O(N) n一定是未知的; 如果n是已知的O(1)
for (i = 0; i < n; i++) {
    a = a + 1;
}

平方: O(n^2)

for (i = 0; i < n; i++) {
    for (int j = 0; j < n; j++) {
        a = a + 1; // O(n^2)  n*(n+1)/2 => O(n^2) = 1+2+3+4+..+n=n*(n+1)/2
    }
}

调优

列表排序 冒泡排序 < 快速排序 归并排序 堆排序

经典:链表、排序算法、二叉树、红黑树、B-Tree、B+Tree

进阶:数论、图论

数据结构

数组

面试题:

  • 问题1:给你一个文件包含全国14亿人口年龄数据(0~180),让你统计每一个年龄有多少人?
    限定机器单台2G + 2G内存,不得使用现成的容器,比如map等

int age[] = new int[180];
a[0]++; // 0表示0岁

利用数组下标,也可以利用下标随机定位到数组中的某一个数据

  • 问题2:为什么数值的下标是从0开始的,数组的特点是一段连续的内存地址

int arr[] = new int[5];

申请到内存地址例如是:10001,10002,10003,10004,10005

保存数据:a[0] => 10001 ===> 10001 + 0
保存数据:a[1] => 10002 ===> 10001 + 1
保存数据:a[2] => 10003 ===> 10001 + 2
保存数据:a[3] => 10004 ===> 10001 + 3
保存数据:a[4] => 10005 ===> 10001 + 4
  • 问题3:二维数组的内存地址是怎么样的?
1 2 3
4 5 6  =>  1 2 3 4 5 6 => i*n + j (i是一维数组的长度、j是在列的位置) => 4 => 1*3 + 0 = 3

ArrayList 和 数组 如何选择

  • ArrayList是JDK封装好的,不需要管扩容
  • 数组删除添加慢O(n),修改获取快O(1)
  • 随机访问
  • 下标

如何选择?

  • 如果不知道数据大小选择ArrayList
  • 如果知道数据的大小而且又非常关注性能选择数组;需要注意越界(头和尾)

Java 内存分为堆内存和栈内存

堆内存:存放new对象和数组
栈内存:引用变量

堆和栈都是用来存数据的地方

堆栈区别:

  • 栈的速度更快
  • 栈的内存数据可以共享,主要存一些基本数据类型 int a = 3; 在栈中创建变量a, 然后给a赋值,先不会创建一个3而是先在栈中找有没有3
String s1 = "ja";
String s2 = "va";
String s3 = "java";
String s4 = s1 + s2; // java 里面重装了+, 其实调用了 stringBuild, 会new对象
System.out.println(s3 == s4); //false
System.out.println(s3.equals(s4)); //true

链表

链表面试题:

  1. 如何设计一个LRU缓存淘汰算法(链表)
  2. 约瑟夫问题(循环链表):N个人围成一圈,从第一个开始报数,第M个将被淘汰,最后剩下一个人,其余人都将被杀掉。例如N=6,M=5,被杀掉的顺序是5,4,6,2,3,1

链表特点

  • 不需要连续的内存空间
  • 有指针引用
  • 常见链表结构:单链表、双链表、循环链表

单链表 LinkList

查询 O(n)
插入删除 O(1)

双链表

B+树 Mysql 叶子节点就是双向链表
跳表和Mysql B+树很像

循环链表

循环链表是一种特殊的单链表,他和单链表唯一区别是尾结点
单链表的尾结点是空地址
循环链表的尾结点是头节点

链表和数组对比

查询

数组

  • O(1)

链表

  • O(n)

插入/删除

数组插入

  • 尾部O(1)
  • 头部O(n)

链表插入

  • 头部O(1)
  • 尾部O(1)
  • 中间O(1*2)

重要区别:

  1. 数组简单易用,在实现上使用的是连续的内存空间,可以借助CPU的缓存机制,预读数组中的数据,所以访问效率更高。
  2. 链表在内存中并不是连续存储,所以对CPU缓存不友好,没办法有效预读。
  3. 数组的缺点是大小固定,一经声明就要占用整块连续内存空间。如果声明的数组过大,系统可能没有足够的连续内存空间分配给它,
    导致“内存不足(out ofmemory)”。如果声明的数组过小,则可能出现不够用的情况。
  4. 动态扩容:数组需再申请一个更大的内存空间,把原数组拷贝进去,非常费时。链表本身没有大小的限制,天然地支持动态扩容。

栈—后进先出–LILO

面试题:

  1. 如何设计一个括号匹配的功能?栈
  2. 如何设计一个浏览器前进和后退功能?(两个栈一个前进、一个后退)
  3. 如何实现四则数字运算公式比如 3+2*5-3 ?(两个栈一个数字、一个符号)
  4. 代码函数调用顺序

栈特点

栈是一个限定仅在表尾插入和删除操作的线性表,被称为栈顶、另一端为栈底

向一个栈插入新元素称为进栈、入栈、压栈
删除元素称为出栈、退栈

栈其实是一种特殊的链表或数组,数组链表暴露太多接口,容易出错

public class KuoHaoStack {
    public static boolean isOk(String s) {
        MyStack<Character> brackets = new ArrayStack<>(20);
        char c[] = s.toCharArray();
        Character top = null;
        for (char x : c) {
            switch (x) {
                case '{':
                case '(':
                case '[':
                    brackets.push(x);
                    break;
                case '}':
                    top = brackets.pop();
                    if (top == null) return false;
                    if('{' == top) {
                        break;
                    } else {
                        return false;
                    }
                case ')':
                    top = brackets.pop();
                    if (top == null) return false;
                    if('(' == top) {
                        break;
                    } else {
                        return false;
                    }
                case ']':
                    top = brackets.pop();
                    if (top == null) return false;
                    if('[' == top) {
                        break;
                    } else {
                        return false;
                    }
                default:
                    break;
            }
        }
        return brackets.isEmpty();
    }

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        while (scanner.hasNext()) {
            System.out.println("S的输出结果:" + isOk(scanner.next()));
        }
    }
}

队列–先进先出–FIFO

问题

  • 线程池里面当任务满时,此时又来一个新任务,线程池是如何处理的?具体有哪些策略?这些策略又是如何实现的呢?
  1. 排队:阻塞队列。有空闲的时候再拿,不就是那个take和put,如果是在公平的情况下,那肯定就是先进先出。这就是今天讲的队列。这时候我们就有两种方式,一个是无限的排队队列。(链表,千万别用。LinkedBlockingQueue,JDK的),还有一种就是有界(用数组来实现的),只处理我们开的空间大小,多了的继续抛出去。Integer.MAX=?2^32-1=21亿多,但是注意的是这个队列大小,别搞小了。就不够,大了就浪费。在一些小型系统,你知道数据请求量是不大的,可以用。
  2. 丢弃:不处理了,直接抛出去。

队列是一种特殊的线性表,特殊之处在于它只允许在表的前端进行删除操作,在表的后端进行插入操作

队列分类

  1. 顺序(单向)队列:(Queue) 只能在一端插入数据,另一端删除数据
    在这里插入图片描述
  2. 循环(双向)队列(Deque):每一端都可以进行插入数据和删除数据操作
    判断是否已满
  • 方式一:添加size
  • 方式二:(tail+1)%n == head

在这里插入图片描述

public class CircleArrayQueue<Item> {

    private Item data[];
    private int head = 0;
    private int tail = 0;
    private int n = 0; //数组最大空间
    private int size; //当前队列已存个数

    public CircleArrayQueue(int cap) {
        data = (Item[]) new Object[cap];
        n = cap;
    }

    public void push(Item item) {
        if ((tail + 1)%n == head) return; //关键点

        data[tail] = item;
        tail = (tail + 1) % n; //关键点
    }

    public Item pop() {
        if (isEmpty()) return null;

        Item item = data[head];
        head = (head + 1) % n;
        return item;
    }

    public boolean isEmpty() {
        return head == tail;
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/450523.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Oracle 定时任务job实际应用

Oracle 定时任务job实际应用 一、Oracle定时任务简介二、dbms_job涉及到的知识点三、初始化相关参数job_queue_processes四、实际创建一个定时任务&#xff08;一分钟执行一次&#xff09;&#xff0c;实现定时一分钟往表中插入数据4.1 创建需要定时插入数据的目标表4.2 创建定…

如何为Google Play的应用制作宣传视频

在用户打开我们的应用页面时&#xff0c;最先看到的是宣传视频&#xff0c;这是吸引潜在用户注意力的绝好机会&#xff0c;所以这对于 Google Play 来说是一件大事。 宣传视频和屏幕截图一起&#xff0c;都是引导用户去使用我们应用程序的第一步&#xff0c;能够让他们一打开应…

sibelius西贝柳斯2023中文版是什么打谱软件?如何下载

Sibelius是一款专业的音乐制谱软件&#xff0c;被广泛用于各类音乐创作、教育、表演等领域。通过Sibelius&#xff0c;用户可以快速、准确地制作各种类型的音乐谱面&#xff0c;同时支持多种音乐符号和效果的编辑、自定义和输出&#xff0c;可谓是音乐领域的必备工具之一。Sibe…

SQL Server tempdb 闩锁争用

当你反复创建和删除 TempDb 对象&#xff08;临时表、表变量等&#xff09;时&#xff0c;你可能会在 tempdb 中看到页面的闩锁争用。当你注意到tempdb 上的 PAGELATCH_* 争用&#xff08;sysprocesses 中的等待资源以 2: 开头&#xff09;时&#xff0c;请检查闩锁等待是否在 …

【语音之家】AI产业沙龙 —— 三星语言智能团队ICASSP2023论文分享会

由CCF语音对话与听觉专委会 、中国人工智能产业发展联盟&#xff08;AIIA&#xff09;评估组、三星电子中国研究院、语音之家、希尔贝壳共同主办的【语音之家】AI产业沙龙——三星语言智能团队ICASSP2023论文分享会&#xff0c;将于2023年4月25日18:30-20:20线上直播。 沙龙简介…

ERP系统有哪些功能模块?

一、ERP系统是什么 现在市面上的管理软件有很多&#xff0c;不少企业都会去选择一些操作简单便捷的软件&#xff0c;优化工作流程&#xff0c;提高工作效率&#xff0c;其中ERP系统就是常见的一种&#xff0c;ERP是企业资源计划(Enterprise Resource Planning)的简称&#xff…

深入了解Lock同步锁的优化

大家好&#xff0c;我是易安。 今天我们来简单谈谈在JDK1.5之后&#xff0c;Java提供的Lock同步锁。 相对于需要JVM隐式获取和释放锁的Synchronized同步锁&#xff0c;Lock同步锁&#xff08;以下简称Lock锁&#xff09;需要的是显示获取和释放锁&#xff0c;这就为获取和释放锁…

防止机械/移动硬盘休眠 - NoSleepHD

防止机械/移动硬盘休眠 - NoSleepHD 前言解决方案计算机硬盘移动硬盘 前言 机械硬盘休眠后唤醒需要一定时间&#xff0c;且频繁的启动和停止并不有利于硬盘的寿命&#xff0c;因此可根据自身需求防止机械硬盘休眠&#xff0c;下文以Win10系统为例介绍解决方案。 值得一提的是…

Java核心技术 卷1-总结-9

Java核心技术 卷1-总结-9 使用异常机制的技巧为什么要使用泛型程序设计定义简单泛型类泛型方法类型变量的限定 泛型类型的继承规则 使用异常机制的技巧 1.异常处理不能代替简单的测试。 使用异常的基本规则是&#xff1a;只在异常情况下使用异常机制。 2.不要过分地细化异常。…

第三章(3):深入理解Spacy库基本使用方法

第三章&#xff08;3&#xff09;&#xff1a;深入理解Spacy库基本使用方法 本章主要介绍了Spacy库的基本使用方法&#xff0c;包括安装、加载语言模型、分句、分词、词性标注、停用词识别、命名实体识别、依存分析和词性还原等内容。重点介绍了每个步骤的具体实现方式和应用场…

【TortoiseGit】安装和配置

转自 【TortoiseGit】TortoiseGit安装和配置详细说明_No8g攻城狮的博客-CSDN博客 一、TortoiseGit 简介 TortoiseGit 是基于 TortoiseSVN 的 Git 版本的 Windows Shell 界面。它是开源的&#xff0c;可以完全使用免费软件构建。 TortoiseGit 支持你执行常规任务&#xff0c;…

出道即封神的ChatGPT,现在怎么样了?ChatGPT想干掉测试人员,做梦去吧

从互联网的普及到智能手机&#xff0c;都让广袤的世界触手而及&#xff0c;如今身在浪潮中的我们&#xff0c;已深知其力。 前阵子爆火的ChatGPT&#xff0c;不少人保持观望态度。现如今&#xff0c;国内关于ChatGPT的各大社群讨论&#xff0c;似乎沉寂了不少&#xff0c;现在…

Mosquitto vs NanoMQ | 2023 MQTT Broker 对比

引言 Mosquitto 和 NanoMQ 都是用 C/C 开发的快速轻量的开源 MQTT Broker&#xff0c;完全支持 MQTT 3.1.1 和 5.0。 虽然 Mosquitto 和 NanoMQ 都具有轻量级和低资源消耗的特点&#xff0c;但它们的架构设计却截然不同。Mosquitto 采用单线程模式&#xff0c;而 NanoMQ 则基…

数据结构:单向链表(无头非循环)

朋友们、伙计们&#xff0c;我们又见面了&#xff0c;本期来给大家解读一下数据结构方面有关链表的相关知识点&#xff0c;如果看完之后对你有一定的启发&#xff0c;那么请留下你的三连&#xff0c;祝大家心想事成&#xff01; C语言专栏&#xff1a;C语言&#xff1a;从入门到…

【云原生】prometheus监控告警之安装部署alertmanager实战

前言 &#x1f3e0;个人主页&#xff1a;我是沐风晓月 &#x1f9d1;个人简介&#xff1a;大家好&#xff0c;我是沐风晓月&#xff0c;阿里云社区博客专家&#x1f609;&#x1f609; &#x1f495; 座右铭&#xff1a; 先努力成长自己&#xff0c;再帮助更多的人 &#xff0c…

Python基础知识:绝对/相对路径等

1 Python处理相对/绝对路径 由于本人在导入数据时&#xff0c;十分喜欢相对路径&#xff08;在数据的上一级文件中&#xff0c;新建文件夹保存处理整个代码处理过程&#xff09;&#xff0c;因此&#xff0c;将首先简单介绍下Python中相对/绝对路径的处理。 1.1 绝对路径 一…

状态压缩DP-蒙德里安的梦想

题意 求把 NM 的棋盘分割成若干个 12 的长方形&#xff0c;有多少种方案。 例如当 N2&#xff0c;M4 时&#xff0c;共有 5 种方案。当 N2&#xff0c;M3 时&#xff0c;共有 3 种方案。 如下图所示&#xff1a; 输入格式 输入包含多组测试用例。 每组测试用例占一行&#xff0…

Jupyter notebook安装教程

文章目录 前言一、安装步骤1、安装 Python 编译器2、安装 jupyter3、运行 Jupyter notebook 二、 更改打开文件位置和快捷启动方式1、更改打开文件位置2、创建快捷启动方式 前言 Jupyter Notebook 是以网页的形式打开&#xff0c;可以在网页页面中直接编写代码和运行代码&…

20230421 | 203. 移除链表元素、707. 设计链表、206. 反转链表

1、203. 移除链表元素 方法1&#xff1a;不添加虚拟节点方式&#xff0c;但是要注意处理删除头部的数据 时间复杂度 O(n) 空间复杂度 O(1) /*** Definition for singly-linked list.* public class ListNode {* int val;* ListNode next;* ListNode() {}* List…

婚恋交友app开发中需要注意的安全问题

前言 随着移动设备的普及&#xff0c;婚恋交友app已经成为了人们生活中重要的一部分。但是&#xff0c;这些应用的开发者需要确保应用的安全性&#xff0c;以保护用户的隐私和数据免受攻击。本文将介绍在婚恋交友app开发中需要注意的安全问题。 在当今数字化时代&#xff0c;…