AcWing
  • 首页
  • 课程
  • 题库
  • 更多
    • 竞赛
    • 题解
    • 分享
    • 问答
    • 应用
    • 校园
  • 关闭
    历史记录
    清除记录
    猜你想搜
    AcWing热点
  • App
  • 登录/注册

常用代码模板2——数据结构

作者: 作者的头像   yxc ,  2019-07-31 21:34:51 ,  所有人可见 ,  阅读 271781


1148


1777

算法基础课相关代码模板

  • 活动链接 —— 算法基础课

单链表 —— 模板题 AcWing 826. 单链表

// head存储链表头,e[]存储节点的值,ne[]存储节点的next指针,idx表示当前用到了哪个节点
int head, e[N], ne[N], idx;

// 初始化
void init()
{
    head = -1;
    idx = 0;
}

// 在链表头插入一个数a
void insert(int a)
{
    e[idx] = a, ne[idx] = head, head = idx ++ ;
}

// 将头结点删除,需要保证头结点存在
void remove()
{
    head = ne[head];
}

双链表 —— 模板题 AcWing 827. 双链表

// e[]表示节点的值,l[]表示节点的左指针,r[]表示节点的右指针,idx表示当前用到了哪个节点
int e[N], l[N], r[N], idx;

// 初始化
void init()
{
    //0是左端点,1是右端点
    r[0] = 1, l[1] = 0;
    idx = 2;
}

// 在节点a的右边插入一个数x
void insert(int a, int x)
{
    e[idx] = x;
    l[idx] = a, r[idx] = r[a];
    l[r[a]] = idx, r[a] = idx ++ ;
}

// 删除节点a
void remove(int a)
{
    l[r[a]] = l[a];
    r[l[a]] = r[a];
}

栈 —— 模板题 AcWing 828. 模拟栈

// tt表示栈顶
int stk[N], tt = 0;

// 向栈顶插入一个数
stk[ ++ tt] = x;

// 从栈顶弹出一个数
tt -- ;

// 栈顶的值
stk[tt];

// 判断栈是否为空,如果 tt > 0,则表示不为空
if (tt > 0)
{

}

队列 —— 模板题 AcWing 829. 模拟队列

1. 普通队列:
// hh 表示队头,tt表示队尾
int q[N], hh = 0, tt = -1;

// 向队尾插入一个数
q[ ++ tt] = x;

// 从队头弹出一个数
hh ++ ;

// 队头的值
q[hh];

// 判断队列是否为空,如果 hh <= tt,则表示不为空
if (hh <= tt)
{

}
2. 循环队列
// hh 表示队头,tt表示队尾的后一个位置
int q[N], hh = 0, tt = 0;

// 向队尾插入一个数
q[tt ++ ] = x;
if (tt == N) tt = 0;

// 从队头弹出一个数
hh ++ ;
if (hh == N) hh = 0;

// 队头的值
q[hh];

// 判断队列是否为空,如果hh != tt,则表示不为空
if (hh != tt)
{

}

单调栈 —— 模板题 AcWing 830. 单调栈

常见模型:找出每个数左边离它最近的比它大/小的数
int tt = 0;
for (int i = 1; i <= n; i ++ )
{
    while (tt && check(stk[tt], i)) tt -- ;
    stk[ ++ tt] = i;
}

单调队列 —— 模板题 AcWing 154. 滑动窗口

常见模型:找出滑动窗口中的最大值/最小值
int hh = 0, tt = -1;
for (int i = 0; i < n; i ++ )
{
    while (hh <= tt && check_out(q[hh])) hh ++ ;  // 判断队头是否滑出窗口
    while (hh <= tt && check(q[tt], i)) tt -- ;
    q[ ++ tt] = i;
}

KMP —— 模板题 AcWing 831. KMP字符串

// s[]是长文本,p[]是模式串,n是s的长度,m是p的长度
求模式串的Next数组:
for (int i = 2, j = 0; i <= m; i ++ )
{
    while (j && p[i] != p[j + 1]) j = ne[j];
    if (p[i] == p[j + 1]) j ++ ;
    ne[i] = j;
}

// 匹配
for (int i = 1, j = 0; i <= n; i ++ )
{
    while (j && s[i] != p[j + 1]) j = ne[j];
    if (s[i] == p[j + 1]) j ++ ;
    if (j == m)
    {
        j = ne[j];
        // 匹配成功后的逻辑
    }
}

Trie树 —— 模板题 AcWing 835. Trie字符串统计

int son[N][26], cnt[N], idx;
// 0号点既是根节点,又是空节点
// son[][]存储树中每个节点的子节点
// cnt[]存储以每个节点结尾的单词数量

// 插入一个字符串
void insert(char *str)
{
    int p = 0;
    for (int i = 0; str[i]; i ++ )
    {
        int u = str[i] - 'a';
        if (!son[p][u]) son[p][u] = ++ idx;
        p = son[p][u];
    }
    cnt[p] ++ ;
}

// 查询字符串出现的次数
int query(char *str)
{
    int p = 0;
    for (int i = 0; str[i]; i ++ )
    {
        int u = str[i] - 'a';
        if (!son[p][u]) return 0;
        p = son[p][u];
    }
    return cnt[p];
}

并查集 —— 模板题 AcWing 836. 合并集合, AcWing 837. 连通块中点的数量

(1)朴素并查集:

    int p[N]; //存储每个点的祖宗节点

    // 返回x的祖宗节点
    int find(int x)
    {
        if (p[x] != x) p[x] = find(p[x]);
        return p[x];
    }

    // 初始化,假定节点编号是1~n
    for (int i = 1; i <= n; i ++ ) p[i] = i;

    // 合并a和b所在的两个集合:
    p[find(a)] = find(b);


(2)维护size的并查集:

    int p[N], size[N];
    //p[]存储每个点的祖宗节点, size[]只有祖宗节点的有意义,表示祖宗节点所在集合中的点的数量

    // 返回x的祖宗节点
    int find(int x)
    {
        if (p[x] != x) p[x] = find(p[x]);
        return p[x];
    }

    // 初始化,假定节点编号是1~n
    for (int i = 1; i <= n; i ++ )
    {
        p[i] = i;
        size[i] = 1;
    }

    // 合并a和b所在的两个集合:
    size[find(b)] += size[find(a)];
    p[find(a)] = find(b);


(3)维护到祖宗节点距离的并查集:

    int p[N], d[N];
    //p[]存储每个点的祖宗节点, d[x]存储x到p[x]的距离

    // 返回x的祖宗节点
    int find(int x)
    {
        if (p[x] != x)
        {
            int u = find(p[x]);
            d[x] += d[p[x]];
            p[x] = u;
        }
        return p[x];
    }

    // 初始化,假定节点编号是1~n
    for (int i = 1; i <= n; i ++ )
    {
        p[i] = i;
        d[i] = 0;
    }

    // 合并a和b所在的两个集合:
    p[find(a)] = find(b);
    d[find(a)] = distance; // 根据具体问题,初始化find(a)的偏移量

堆 —— 模板题 AcWing 838. 堆排序, AcWing 839. 模拟堆

// h[N]存储堆中的值, h[1]是堆顶,x的左儿子是2x, 右儿子是2x + 1
// ph[k]存储第k个插入的点在堆中的位置
// hp[k]存储堆中下标是k的点是第几个插入的
int h[N], ph[N], hp[N], size;

// 交换两个点,及其映射关系
void heap_swap(int a, int b)
{
    swap(ph[hp[a]],ph[hp[b]]);
    swap(hp[a], hp[b]);
    swap(h[a], h[b]);
}

void down(int u)
{
    int t = u;
    if (u * 2 <= size && h[u * 2] < h[t]) t = u * 2;
    if (u * 2 + 1 <= size && h[u * 2 + 1] < h[t]) t = u * 2 + 1;
    if (u != t)
    {
        heap_swap(u, t);
        down(t);
    }
}

void up(int u)
{
    while (u / 2 && h[u] < h[u / 2])
    {
        heap_swap(u, u / 2);
        u >>= 1;
    }
}

// O(n)建堆
for (int i = n / 2; i; i -- ) down(i);

一般哈希 —— 模板题 AcWing 840. 模拟散列表

(1) 拉链法
    int h[N], e[N], ne[N], idx;

    // 向哈希表中插入一个数
    void insert(int x)
    {
        int k = (x % N + N) % N;
        e[idx] = x;
        ne[idx] = h[k];
        h[k] = idx ++ ;
    }

    // 在哈希表中查询某个数是否存在
    bool find(int x)
    {
        int k = (x % N + N) % N;
        for (int i = h[k]; i != -1; i = ne[i])
            if (e[i] == x)
                return true;

        return false;
    }

(2) 开放寻址法
    int h[N];

    // 如果x在哈希表中,返回x的下标;如果x不在哈希表中,返回x应该插入的位置
    int find(int x)
    {
        int t = (x % N + N) % N;
        while (h[t] != null && h[t] != x)
        {
            t ++ ;
            if (t == N) t = 0;
        }
        return t;
    }

字符串哈希 —— 模板题 AcWing 841. 字符串哈希

核心思想:将字符串看成P进制数,P的经验值是131或13331,取这两个值的冲突概率低
小技巧:取模的数用2^64,这样直接用unsigned long long存储,溢出的结果就是取模的结果

typedef unsigned long long ULL;
ULL h[N], p[N]; // h[k]存储字符串前k个字母的哈希值, p[k]存储 P^k mod 2^64

// 初始化
p[0] = 1;
for (int i = 1; i <= n; i ++ )
{
    h[i] = h[i - 1] * P + str[i];
    p[i] = p[i - 1] * P;
}

// 计算子串 str[l ~ r] 的哈希值
ULL get(int l, int r)
{
    return h[r] - h[l - 1] * p[r - l + 1];
}

C++ STL简介

vector, 变长数组,倍增的思想
    size()  返回元素个数
    empty()  返回是否为空
    clear()  清空
    front()/back()
    push_back()/pop_back()
    begin()/end()
    []
    支持比较运算,按字典序

pair<int, int>
    first, 第一个元素
    second, 第二个元素
    支持比较运算,以first为第一关键字,以second为第二关键字(字典序)

string,字符串
    size()/length()  返回字符串长度
    empty()
    clear()
    substr(起始下标,(子串长度))  返回子串
    c_str()  返回字符串所在字符数组的起始地址

queue, 队列
    size()
    empty()
    push()  向队尾插入一个元素
    front()  返回队头元素
    back()  返回队尾元素
    pop()  弹出队头元素

priority_queue, 优先队列,默认是大根堆
    size()
    empty()
    push()  插入一个元素
    top()  返回堆顶元素
    pop()  弹出堆顶元素
    定义成小根堆的方式:priority_queue<int, vector<int>, greater<int>> q;

stack, 栈
    size()
    empty()
    push()  向栈顶插入一个元素
    top()  返回栈顶元素
    pop()  弹出栈顶元素

deque, 双端队列
    size()
    empty()
    clear()
    front()/back()
    push_back()/pop_back()
    push_front()/pop_front()
    begin()/end()
    []

set, map, multiset, multimap, 基于平衡二叉树(红黑树),动态维护有序序列
    size()
    empty()
    clear()
    begin()/end()
    ++, -- 返回前驱和后继,时间复杂度 O(logn)

    set/multiset
        insert()  插入一个数
        find()  查找一个数
        count()  返回某一个数的个数
        erase()
            (1) 输入是一个数x,删除所有x   O(k + logn)
            (2) 输入一个迭代器,删除这个迭代器
        lower_bound()/upper_bound()
            lower_bound(x)  返回大于等于x的最小的数的迭代器
            upper_bound(x)  返回大于x的最小的数的迭代器
    map/multimap
        insert()  插入的数是一个pair
        erase()  输入的参数是pair或者迭代器
        find()
        []  注意multimap不支持此操作。 时间复杂度是 O(logn)
        lower_bound()/upper_bound()

unordered_set, unordered_map, unordered_multiset, unordered_multimap, 哈希表
    和上面类似,增删改查的时间复杂度是 O(1)
    不支持 lower_bound()/upper_bound(), 迭代器的++,--

bitset, 圧位
    bitset<10000> s;
    ~, &, |, ^
    >>, <<
    ==, !=
    []

    count()  返回有多少个1

    any()  判断是否至少有一个1
    none()  判断是否全为0

    set()  把所有位置成1
    set(k, v)  将第k位变成v
    reset()  把所有位变成0
    flip()  等价于~
    flip(k) 把第k位取反

161 评论


用户头像
Conan15   2021-10-17 16:09      114    踩      回复

y总的STL比我的简洁多了!
###### 因为我的都是一堆废话

用户头像
Conan15   2023-05-12 21:12      35    踩      回复

没想到当时随随便便发的感想拿了22赞

至今感想还是一样qaq

用户头像
Conan15   5个月前    回复了 Conan15 的评论      10    踩      回复

再次考古 QwQ,我现在仍然是废物。

用户头像
星想   5个月前    回复了 Conan15 的评论         踩      回复

95赞

用户头像
早上起来胡辣汤就着算法题会不会更下饭点   4个月前    回复了 Conan15 的评论         踩      回复

因为你将会是屠榜选手,orz,or2


用户头像
王蒙   2022-05-31 11:20      24    踩      回复

数组模拟一切。后悔没有早看到此贴。


用户头像
LeRoi   2022-07-30 21:32      13    踩      回复

查找祖宗—非递归

int find(int x){
    int r=x;
    while(p[r]!=r) r=p[r];//查找祖宗
    int i=x,j;
    while(i!=r){
        j=p[i];//临时存i的父亲节点
        p[i]=r;//更改i的父亲节点为祖宗
        i=j;//对i的父亲下手
    }
    return r;
}
用户头像
lhn2001   2022-08-04 17:26      1    踩      回复

对父亲下手还行哈哈哈,感觉还是递归写法好看,优雅永不过时

用户头像
LeRoi   2022-08-04 17:30    回复了 lhn2001 的评论      2    踩      回复

数据规模大的话可以用

用户头像
lhn2001   2022-08-08 16:55    回复了 LeRoi 的评论         踩      回复

递归写法优势在于处理数据规模大的情况吗?

用户头像
LeRoi   2022-08-08 22:04    回复了 lhn2001 的评论         踩      回复

我说的非递归

用户头像
lhn2001   2022-08-09 10:44    回复了 LeRoi 的评论      1    踩      回复

哦

用户头像
手撕提高课   2022-09-12 10:37    回复了 lhn2001 的评论      3    踩      回复

真实


用户头像
三丰方少   2022-03-07 20:55      7    踩      回复

很多讲的太抽象,其实算法应该分逻辑和物理,但是这里对变量的解释很少

用户头像
宇哥   2022-04-06 22:18      1    踩      回复

主要是讲怎么应用到题目里吧,数据结构那些东西看书就行了

用户头像
grq   2022-04-15 09:57      1    踩      回复

是的。那些内容我都是课下用笔自己模拟一遍

用户头像
yuki123   2022-06-04 16:15         踩      回复

我也感觉听不太懂

用户头像
Kaxiya   2023-02-07 20:20      2    踩      回复

这个课只是教怎么做题

用户头像
yechen   7个月前    回复了 Kaxiya 的评论      3    踩      回复

其实真的二刷什么的就会发现y总逻辑很清晰,每一句话都讲清楚了,只是当时水平还没起来所以理解吃力甚至没听清楚是解释


用户头像
我是管小亮   2023-08-20 21:36      6    踩      回复

(更新中)【Java版本】常用代码模板2——数据结构 + 模板题参考实现

用户头像
WA_automat   7个月前         踩      回复

太需要了,写题一直找不到java板子%%%%%


用户头像
郭子羽   2023-02-06 22:56      5    踩      回复

# NB!!!666!!!


用户头像
梦星   2023-03-31 17:30      2    踩      回复

有stl的话为什么还要讲用数组实现前面的数据结构

用户头像
追风筝的人_08   2023-05-29 12:09         踩      回复

手写的快

用户头像
Mebius_5   9个月前         踩      回复

stl遇到大数据直接tle

用户头像
Rsadj88   3个月前         踩      回复

之前说过,这个数组实现的会快一点,因为只用申请一次内存。但是stl每次增加一个节点都要new一次,所以会非常耗时间。


用户头像
李易峰c   9个月前      1    踩      回复

常看常新


用户头像
Lucyna   10个月前      1    踩      回复

现在不买课是不能做题了吗?

用户头像
km.seryyrye   2个月前         踩      回复

好像是


用户头像
陆修远   2022-05-16 08:36      1    踩      回复

$y$总能不能贴一份链式前向星的代码?谢谢

用户头像
户用禁封   2022-05-18 19:09         踩      回复

我大号这个是吗?

https://www.acwing.com/blog/content/10540/

用户头像
陆修远   2022-05-18 20:35    回复了 户用禁封 的评论         踩      回复

感谢大佬

用户头像
户用禁封   2022-05-18 20:37    回复了 陆修远 的评论         踩      回复

OrzOrz


用户头像
Conan15   2021-10-17 16:08      1    踩      回复

### new B!


用户头像
mzhccc   2020-02-17 09:21      1    踩      回复

如果计算队列中元素的个数,普通队列可以直接用hh-tt计算,循环队列只能用变量来存储。还是有循环队列的计算公式。

用户头像
yxc   2020-02-24 16:47      1    踩      回复

如果tt > hh,个数就是tt - hh;如果hh == tt,个数是0;如果hh > tt,个数就是n + tt - hh,n是队列数组的长度。

用户头像
mzhccc   2020-02-24 23:32    回复了 yxc 的评论         踩      回复

蟹蟹

用户头像
尚且   2023-03-12 09:19      1    踩      回复

循环队列无论tt大于或者小于hh,都可以写成
(tt-hh+n)%n


用户头像
霓虹小孩   1个月前         踩      回复

强大


用户头像
我来定义AC   3个月前         踩      回复

[]是什么意思


用户头像
琪琪怪怪   7个月前         踩      回复
cout<<"tql";

用户头像
_admin   10个月前         踩      回复

111


用户头像
而陆   10个月前         踩      回复

# 牛逼


用户头像
@_4   2023-03-23 22:42         踩      回复

大佬,谁有循环单链表和循环双链表的模板


用户头像
dannywang   2023-02-07 19:59         踩      回复

循环队列那个模板,如果hh等于tt,是不是也可能表示队列装满了啊?

用户头像
大头虾   2023-02-21 22:06         踩      回复

所以循环队列一般都会留一个空位置。比如说size是N,那么最多只装N-1个元素


用户头像
小mo   2023-01-28 20:22         踩      回复

好耶ヽ(✿゚▽゚)ノ


你确定删除吗?
1024
x

© 2018-2024 AcWing 版权所有  |  京ICP备2021015969号-2
用户协议  |  隐私政策  |  常见问题  |  联系我们
AcWing
请输入登录信息
更多登录方式: 微信图标 qq图标 qq图标
请输入绑定的邮箱地址
请输入注册信息