AcWing算法基础课----数据结构(二) 笔记 (Tire树 + 并查集 + 堆)

数据结构

  • 1.Trie树
  • 2.并查集
    • (1)朴素并查集:
    • (2)维护size的并查集:
    • (3)维护到祖宗节点距离的并查集:
  • 3.堆
    • 如何手写一个堆?

1.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];
}

2.并查集

代码短 + 思路精巧

时间复杂度:近乎O(1)

作用:
1.将两个集合合并
2.询问两个元素是否在一个集合当中

基本原理:
树的形式来维护每一个集合
根节点编号即为该集合编号
每个节点存储他的父节点: p[x]表示x的父节点

Q1:如何判断树根 if( p[x] == x)
Q2:如何求x的集合编号:while(p[x]!=x) x=p[x]
Q3:如何合并两个集:px是x的集合编号,py是y的集合编号 ,p[x]=y

优化:
路径压缩、按秩合并

(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)的偏移量

3.堆

如何手写一个堆?

堆是一颗完全二叉树

down(x) 向下调整

up(x) 向上调整

STL堆可直接实现:

1.插入一个数 heap[++size]=x ; up(size);
2.求集合当中的最小值 heap[1]
3.删除最小值:用堆的最后一个元素覆盖堆顶元素 heap[1] = heap[size]; size--; down(1);

STL堆不可直接实现:

4.删除任意元素 heap[k]=heap[size];size--;down(k);up(k)
5.修改任意元素 heap[k]=x; down(k); up(k);

注意上面两个操作中down和up只会进行其中之一

小顶堆:每一个点都小于等于左右儿子

堆的存储:一维数组,1#为根节点 x的左儿子 2x 右儿子 2x+1

模板

// 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,c语言,c++)