zoukankan      html  css  js  c++  java
  • 数据结构题大赏

    偷走北大讲义

    T1:紧急集合

    【题目链接】

    涉及主要算法:LCA;

    SOLUTION:

    求两两之间的LCA,取MIN就是最小的qwq;

    求三个结点到一个结点距离之和最小的结点以及距离和

    求出两两lca,其中有两个相同,答案则为另一个,画画图就可以理解

    贴一篇blog

    T2:中位数:

    【题目链接】

    维护两个堆,一个大根堆,一个小根堆;

    对于序列:每新输入一个数,我们将之加入大根堆中,但大根堆不是无限加入的,我们设当前输入了k个数,那么大根堆中的元素至多有k/2 +1个,当大根堆中个数超过限制后,将堆顶移到小根堆,这样就可以保证大根堆堆顶永远是中位数;

    next

    合并果子:

    【题目链接】

    这个题就很水了;

    haffman tree:每次将最小的两个合并,一定是最优的;

    深度越深的节点,统计的次数越多;

    因此越小的应该越先被合并;

    题解:【堆】【洛谷例题】p1090 p1334 p1177

     滑雪等级Ski Course Rating:

    【题目链接】

    并查集问题:

    1. 将所有点,把边连出来
    2. 把所有边全删去(拿出来),按长度(高度差)从小到大排个序;
    3. 从小的边开始加,合并的同时统计这个集合大小,某一刻时,集合个数>=t时,最后一条边权就是难度等级D,然后ans+=D*集合个数???可能会有好多集合??

    突然安利oj:

    巴扎黑oj

    树的同构:

    【题目链接】

     一种暴力的想法:

     人为定义一个根,对于每个根,算出一个Hash来,如果有两棵树的Hash集合完全相同,那么这两棵树是同构的;

    如何计算树形Hash:

    把所有儿子的哈希值从大到小排个序,然后按字符串的方法来,计算当前节点的hash值;

    一种很巧妙的算法:

    重心:找到一个点为根,最大的一棵子树节点数最少;

    每一棵无根树,重心个数不会超过二;

    枚举每个重心,以重心为根求出这棵有根树的最小表示,然后取字典序最大的即可。

    对于有根树的最小表示,可以看成括号序列,每次把子树的括号序列按字典序排序后依次串连起来即可。

    有些许懵,嗯;

    有关括号序列:

    POJ 2299 Ultra-QuickSort

    求逆序对数;

    请用树状数组做;

    用树状数组维护每个数之前有多少个大于它的数;

    暴力O(n^2)

    0 1 2 3 4 5 6 7 8 9……

    开一个数据范围大小的数组用来维护当前每个数组出现了几次,然后我们需要求一个>当前数的数的个数,可以发现这是数组后缀,然后可以转化为总数- ≤当前数的个数;

    举个亚子:

    假设数据范围是9,我们开一个10个的数组,分别代表0~9,假设这个序列是19260817

     那么在插入0之前,序列:

    0 1 2 3 4 5 6 7 8 9

    0 1 1 0 0 0 0 0 0 1

    此时我们插入0,我们需要寻找的是现在0之前插入的比0大的数,我们发现这是一个后缀问题,可以利用神仙操作将其转化为前缀。求出所有小于等于0的数,显然是0个,然后用当前插入的数的个数-0就是0的后缀;

    然后可以用树状数组维护前缀和;

    树状数组:动态将某个数+1,动态维护前缀和;

    离散化:sort =>unique => 二分查找;

    有关二维偏序:

    xi≤xj

    yi≤yj

    ybt蓝书水题?!

    按照y的升序(from small to big)处理;

    建立一个树状数组,分别代表x为i的星星有多少个;

    然后动态维护树状数组,对于每组数据,求出所有a≤当前x的数的个数(显然树状数组前缀和维护就好了);

    即为答案level;

    你看这个m超级小,一定有什么玄学?!

    才华小哥哥上去分享,一语点不破梦中人;

    开m个树状数组

    每个树状数组都是0/1树状数组;

    然后第i个树状数组第j个下标,表示a[j]%m==i?1:0;

    下标为i的树状数组+x,把i+x%m置为1,把i%m的树状数组置为0;

    思路1:把新插入的元素y插到set中,找到y的迭代器p,比较*(p++)和*(p--)与y的差值较小的就是波动√

    思路2:维护一个线段树,求区间最大值和区间最小值

    设每天的营业额在线段树数组中的下标为他本身的值

    然后每次输入一个数时我们先在这个数的左边的区间查找最大值

    再在右边的区间查找最小值

    (这意味着我们要找比这个数小的最大值和比这个数大的最小值)

    然后比较两者取差的绝对值最小的加到Ans里即可

    看着就像线段树板子2;

    单区间乘:

    tag=1;开一个tag2表示加法lazy-tag

    每次*c,tag*=c;sum*=c;tag*=c;

    定义矩阵F=1 1  G=1

               1 0    1

    Σi=l~r f(ai);

    Σi=l~rFai*G=(Σi=l~rFai)*G;

    对于区间l~r加上一个数x,相当于变成了*Fx

    对于查询l~r,就是线段树查询出这个数。

     

    这个题需要到叶子节点才能返回;,当某个节点为0或1时就不再更新了;

    同样当某个区间全部为0或1时我们就不需要在将这个区间的数递归开根了,直接返回;

    开一个数组s[5]={s0,s1,s2,s3,s4},分别代表集合l中%5余0,1,2 3,4,之和;

    0 1 2 6 7 8 9

    s:8,10,2,6,7  

    Mex:最小的没有出现过的非负整数;

    求出每个数组的前缀mex;

    前缀mex:

    19260817

    00003333

    拿走一个数x,在出现下一个x之前,所有前缀mex>x的都会被改变为x(区间修改去min)

    然后添加一个数也随意?

    枚举等差中项:

    如果一个数出现过,位置为1,否则为0;每进入一个新数,把它枚举为等差数列中项,看现在以等差中项为对称轴的两侧是否对称(用hash来比较),说明这个等差中项不能产生等差数列,如果不对称,一定存在等差数列,直接输出yes;

  • 相关阅读:
    [转] Linux 最大进程数, unable to create new native thread问题
    [转] Maven 从命令行获取项目的版本号
    [转]【JVM】调优笔记2-----JVM在JDK1.8以后的新特性以及VisualVM的安装使用
    DISCUZ 自定义模板
    Linux系统性能统计工具Sar和实时系统性能监控脚本
    shell脚本常规技巧
    Java中文编码小结
    json-smart 使用示例(推荐fastjson)
    HBase Java简单示例
    Ehcache BigMemory: 摆脱GC困扰
  • 原文地址:https://www.cnblogs.com/zhuier-xquan/p/11191492.html
Copyright © 2011-2022 走看看