首页 > 编程语言 >算法基础课-第二章复盘

算法基础课-第二章复盘

时间:2024-07-19 17:56:45浏览次数:15  
标签:结点 idx px 元素 son 数组 基础课 第二章 复盘

一、栈

(一)单调栈

  • 特点:栈内元素以递增或递减的形式排序。
  • 应用:求解下一个更大元素、下一个更小元素、最大子数组和等问题。
  • 题目示例:

  • 代码逻辑:1、采用递增排序的单调栈,栈顶元素是栈内最大元素。2、循环读入列表元素,在单调栈非空的情况下,如果栈顶元素大于当前元素,将栈顶元素弹出栈,直到栈空或者栈顶元素小于当前元素。3、判断是栈空还是找到了小于当前元素的前序栈中元素。若栈空tt=-1,表明前面没有比当前元素小的。4、将当前元素压入栈中。
def main():
    N=100010;s=[0]*N;tt=-1;ans=[]
    n=int(input())
    nums=list(map(int,input().split()))
    for num in nums:
        while tt>-1 and num<=s[tt]:
            tt-=1
        if tt==-1:
            ans.append(-1)
        else:
            ans.append(s[tt])
        tt+=1;s[tt]=num
    print(" ".join(map(str,ans)))   
main()

(二)表达式求值

  • 数据结构:操作符栈op,操作数栈num,字典priority存放操作符的优先级。
  • 方法:1、循环读入串中字符,利用Python中的isdigit方法判断当前字符是否为数字字符,若是数字字符则循环判断累加数值。2、若非数字字符,则为操作符:①若为左括号,无条件压入。②若为右括号,在操作符栈非空且栈顶元素不是左括号的情况下,不断弹出操作符和操作数进行运算。最后将栈顶的左括号弹出栈(在一切操作合法的前提下)。③若为其他操作符,在栈非空且栈顶元素不是左括号的前提下,如果当前读入的操作符的优先级小于等于操作符栈顶元素的优先级,则不断弹出操作符和操作数进行运算。最后将当前读入的操作符压入操作符栈中。③检查操作符栈是否为空,若非空,弹出操作符和操作数进行计算。④最终操作数栈顶元素就是最终值。
  • 题目

from collections import deque
op=deque();num=deque()
priority={'+':0,'-':0,'*':1,'/':1}
def cal():
    global op,num
    ins=op.pop()
    b=num.pop()
    a=num.pop()
    if ins=='+':
        a+=b
    elif ins=='-':
        a-=b
    elif ins=='*':
        a*=b
    else:
        a=int(a/b)
    num.append(a)
def main():
    char=input();i=0
    while i<len(char):
        if char[i].isdigit():
            res=0
            #比如1+23456,其中23456就需要如下操作得到
            while i<len(char) and char[i].isdigit():
                res=res*10+int(char[i])
                i+=1
            #i回退一位,防止后续字符被越位
            i-=1
            num.append(res)
        else:
            if char[i]=='(':
                op.append(char[i])
            elif char[i]==')':
                while op and op[-1]!='(':
                    cal()
                op.pop()
            else:
                while op and op[-1]!='(' and priority[char[i]]<=priority[op[-1]]:
                    cal()
                op.append(char[i])
        i+=1
    while op:
        cal()
    print(num[-1])
main()

二、滑动窗口

  • 实现:使用动态队列实现
  • 应用:
  1. 最大/最小子数组问题:例如,找出数组中和最大的连续子数组(最大子数组和问题)。
  2. 窗口内元素的统计:比如,给定一个数组和两个整数 k 和 t,找出数组中是否存在长度为 k 的连续子数组,使得子数组内所有元素的和至少为 t。
  3. 满足特定条件的子数组:例如,找出数组中所有连续子数组,使得子数组内所有元素的乘积不超过给定的阈值。
  4. 字符串问题:比如,找出字符串中最长的不含有重复字符的子串。
  5. 数据流问题:在处理数据流时,滑动窗口可以用来计算最近一段时间内的统计数据,如移动平均值。
  6. 时间序列分析:在时间序列数据中,滑动窗口可以用来分析一段时间内的趋势或周期性变化。
  • 题目

方法:比如要输出每个窗口内的最小值,就创建一个递增的动态队列,每个队列的队头元素就是每个窗口内的最小值的索引(需要维护边界,保证每个队列的队头是在窗口的左窗台以内)。

代码:

def main():
    n,k=map(int,input().split())
    nums=list(map(int,input().split()))
    q=[0]*1001000
    hh=0;tt=-1;i=0
    while i<n:
        while hh<=tt and q[hh]<i-k+1:
            hh+=1
        while hh<=tt and nums[q[tt]]>=nums[i]:
            tt-=1
        tt+=1;q[tt]=i
        if i-k+1>=0:
            print(nums[q[hh]],end=" ")
        i+=1
    print()
    hh=0;tt=-1;i=0
    while i<n:
        while hh<=tt and q[hh]<i-k+1:
            hh+=1
        while hh<=tt and nums[q[tt]]<=nums[i]:
            tt-=1
        tt+=1;q[tt]=i
        if i-k+1>=0:
            print(nums[q[hh]],end=" ")
        i+=1
main()

三、KMP

  • 应用:字符串匹配,返回匹配的下标值
  • 主要思想:
  1. 预处理阶段(构建next数组)

    • 构建next数组,存储模式字符串的前缀和后缀的最长公共元素长度。前缀是指字符串开头到某个位置的子串,后缀是指字符串末尾到某个位置的子串。
    • 能够根据next数组的信息,决定模式字符串应该向右滑动多少位。
  2. 字符串匹配阶段

    • 算法会使用next数组来指导模式字符串在文本字符串中的滑动。当模式字符串的某个字符与文本字符串的对应字符不匹配时,算法会查看部分匹配表,根据表中存储的信息,将模式字符串向右滑动适当的位数,而不是简单地滑动一位。
    • 通过这种方式,KMP算法能够减少不必要的比较,提高字符串搜索的效率。
def main():
    N=100100;ne=[0]*N
    n=int(input())
    #第一个字符对应的下标变成1
    p=[x for x in input()];p=[0]+p
    m=int(input())
    s=[x for x in input()];s=[0]+s
    i=2;j=0
    while i<=n:
        while j and p[i]!=p[j+1]:
            j=ne[j]
        if p[i]==p[j+1]:
            j+=1
        ne[i]=j
        i+=1
    i=1;j=0
    while i<=m:
        while j and s[i]!=p[j+1]:
            j=ne[j]
        if s[i]==p[j+1]:
            j+=1
        if j==n:
            print(i-j,end=" ")
            j=ne[j]
        i+=1
main()

四、Trie树

  • 数据结构:子节点指针数组son,子结点计数数组cnt,结点标记idx。
  • 应用:字符串检索。
  • 性质:
  1. 根节点不包含字符:除根节点外,每一个节点都只包含一个字符。‌
  2. 路径表示字符串:从根节点到某一节点,路径上经过的字符连接起来,为该节点对应的字符串。
  3. 子节点字符唯一性:每个节点的所有子节点包含的字符都不相同。‌

(一)字符串统计

题目:

思想:从根结点(标记为0)开始追溯,p表示当前结点。对于插入操作,循环读入字符,如果当前结点p的子节点指针数组son中并没有指向当前字符的指针(即son[p][u]=0,其中u可以由当前字符ASCII码减去'a'的ASCII码得到),那么创建结点(idx+=1,son[p][u]=idx,通过idx来标识结点)。如果已存在指针,p指向当前字符的结点。

代码:

N=100100;son=[[0]*26 for _ in range(N)]
cnt=[0]*N;idx=0
def insert(char):
    global son,cnt,idx
    p=0
    for ch in char:
        u=ord(ch)-ord('a')
        if not son[p][u]:
            idx+=1;son[p][u]=idx
        p=son[p][u]
    cnt[p]+=1
def query(char):
    global son,cnt,idx
    p=0;flag=True
    for ch in char:
        u=ord(ch)-ord('a')
        if not son[p][u]:
            flag=False;break
        p=son[p][u]
    if flag:
        print(cnt[p])
    else:
        print(0)
def main():
    global son,cnt,idx
    n=int(input())
    for _ in range(n):
        op,char=input().split()
        if op=='I':
            insert(char)
        else:
            query(char)
main()

(二)最大异或对

题目:

思路: 1、先用Trie树存储每个树的二进制表达,从最高位开始存储。2、再逐个比较得到的每个数的最大异或值,其中最大的数即为输出:初始化res=0,p指向根结点(p=0)。从最高位开始计算,如果存在相应位置i上与该位不同的值(比如0和1),则res加上1左移i位(1<<i),当前结点指向异或结点。

代码:

N=3000000;s=[[0]*2 for _ in range(N)];idx=0
def insert(num):
    global s,idx
    p=0
    for i in range(30,-1,-1):
        x=num>>i&1
        if not s[p][x]:
            idx+=1;s[p][x]=idx
        p=s[p][x]
def query(num):
    global s,idx
    p=0;res=0
    for i in range(30,-1,-1):
        x=num>>i&1
        if s[p][1-x]:
            res+=1<<i
            p=s[p][1-x]
        else:
            p=s[p][x]
    return res
def main():
    n=int(input())
    nums=list(map(int,input().split()))
    for num in nums:
        insert(num)
    res=0
    for num in nums:
        res=max(res,query(num))
    print(res)
main()

五、并查集

  • 数据结构:祖宗结点数组p
  • 核心思想:1、初始化:每个结点的祖宗结点都是自身,以祖宗结点代表一个集合。2、并集:将其中一个集合的祖宗结点改成另外一个结点的祖宗结点。
  • 核心代码:
def find(x):
    if x!=p[x]:
       p[x]=find(p[x])
    return p[x]
  • 题目:食物链

方法:1、利用并查集,将有关系的动物(吃与被吃,同类关系)纳入一个集合中,利用不同动物与祖宗结点之间的距离来标识关系(如果1吃2,则(d[1]-d[2]-1)%3=0;如果1和2是同类那么(d[1]-d[2])%3=0)。2、距离说明:①初始化距离全为0。②构建距离关系:(1)如果x与y是同类,目前尚未在同一个集合中,则将x的祖宗结点px改成y的祖宗结点py,即p[px]=py。已知d[x]是x到px的距离,d[y]是y到py的距离。由于将px与py相连,x与y是同类,d[px]=d[y]-d[x](实际上是(d[y]-(d[px]+d[x]))%3=0,移项后可得到该公式。(2)如果x吃y,同理更改祖宗结点后,d[px]=d[y]-d[x]+1③判断距离关系(真话假话判断)

 

代码:

N=50010;p=[0]*N;d=[0]*N
def find(x):
    global p,d
    if x!=p[x]:
        root=find(p[x])
        d[x]+=d[p[x]]
        p[x]=root
    return p[x]
def main():
    global p,d
    n,k=map(int,input().split())
    res=0
    for i in range(1,n+1):
        p[i]=i
    for _ in range(k):
        op,x,y=map(int,input().split())
        if x>n or y>n:
            res+=1
            continue
        if op==1:
            px=find(x);py=find(y)
            if px==py:
                if (d[x]-d[y])%3!=0:
                    res+=1
            else:
                p[px]=py
                d[px]=d[y]-d[x]
        elif op==2:
            if x==y:
                res+=1
                continue
            else:
                px=find(x);py=find(y)
                if px==py:
                    if (d[x]-d[y]-1)%3!=0:
                        res+=1
                else:
                    p[px]=py
                    d[px]=d[y]-d[x]+1
    print(res)
main()

标签:结点,idx,px,元素,son,数组,基础课,第二章,复盘
From: https://blog.csdn.net/Willowii/article/details/140418689

相关文章

  • 第二章 操作系统的运行机制
    中央处理器一:CPU的构成与基本的工作方式1、CPU组成(1)CPU由运算器、控制器、一系列寄存器、高速缓存组成运算器:实现指令中的算术和逻辑运算,是计算机系统的和兴控制器:负责控制程序的运行流程、包括取指令、维护CPI的状态寄存器:存取数据和指令(在CPU内部)高速缓存:位于CPU和物理内......
  • 第二章 编译FFmpeg并开启H.264编码
    目录前言1.下载x2642.编译x2643.编译FFmpeg3.1可能出现的问题和解决方法3.1.1ERROR:x264notfoundusingpkg-config解决方法:3.1.2libx264isgpland--enable-gplisnotspecified.解决方法:4.检查编译结果这里我默认大家已经看过第一章FFmpeg初体验:在Centos7.9下编......
  • 王道数据结构课后习题详细分析 第二章线性表 2.1线性表的定义和基本操作
    单项选择题————————————————————————————————————————解析:正确答案:C————————————————————————————————————————解析:A:集合中的元素没有前后驱关系,错误;C:序列中整数不是有限个,错......
  • 计组笔记第二章——数据的表示和运算
    本章问题:数据如何在计算机中表示?运算器如何实现数据的算术、逻辑运算?2.1.1进位计数制十进制计数法古印度人发明阿拉伯数字。基于乘法思想。十进制表示方式:整数情况下:\[K_nK_{n-1}...K_2K_1K_0=K_n\times10^n+K_{n-1}\times10^{n-1}+...+K_2\times10^2......
  • 计算机组成复习——第二章机器指令知识点总结
    ......
  • 【操作系统原理】第二章课后习题
    前言课本:操作系统原理(第五版)[费翔林,骆斌编著]习题:主要习题内容是第一章到第六章,具体内容如下表章节内容链接第一章思考题1,3,7、应用题7,12(1)~(4)https://blog.csdn.net/Zchengjisihan/article/details/136493304?spm=1001.2014.3001.5501第二章思考题1,3,10......
  • 20240712NOIP模拟赛复盘
    20240712NOIP模拟赛复盘总结T1:其实不难,但是认为自己推出来依旧很难。但是暴力分\(15\)pts应该是好拿的。T2:推了一个正解,但是因为一些细节问题写挂了。以后应该先把暴力分全部拿完再写正解,写代码时也需要注意细节。T3:赛时口胡出了正解,但是边界没有考虑完全,导致样例没过,最后......
  • 《三体开源传》第二章 科技图谱
    科技树:科技树是一种结构图,它将技术按照发展顺序排列成树状,展示从基础技术到高级应用的演进路径,通常用于指导科技研究或游戏中的技能进阶。每项技术的解锁往往需要满足特定前置条件,形象地描绘了技术进步的依赖关系和层次。(来自:GPT-4)随着汪淼敲下“Enter”键的那一刻,一张围绕着......
  • 衡庐浅析·C语言程序设计·第二章·运算符及其优先级关系(练习题一)
        本文适用于大学的期中期末考试、专升本(专接本、专插本)考试、408等考研预科。如有相关题目疑问或建议欢迎在评论区进行互动。    转载请标明出处。不知道大家有没有消化完第二章的内容。在这里我们将列出一些关于运算符及其优先级关系的课后练习题,方便大家......
  • 系统架构设计师教程 第二章 计算机系统基础知识-2.3计算机软件
    系统架构设计师教程第二章计算机系统基础知识-2.3计算机软件2.3计算机软件2.3.1计算机软件概述2.3.2操作系统2.3.2.1操作系统的组成2.3.2.2操作系统的作用2.3.2.3操作系统的特征2.3.2.4操作系统的分类2.3.3数据库2.3.3.1关系数据库2.3.3......