【NOIP校内模拟】T3 长者(主席树+哈希+二分)

我们先考虑如何比较两两的字符串

我们可以用线段树来维护哈希值 在线段树上根据二分的性质来做即可

又考虑到 每颗线段树是在之前的某颗基础上只修改了一个节点 那显然就想到了主席树

另外说说如何pushup 我们考虑这样一个字符串 abcdefg

假设当前节点左儿子是abc 右儿子是defg

由于我们的哈希值用的是unsigned long long 自然溢出大法 也就是说abc的哈希值是a(base^1)+b(base^2)+c(base^3),而defg的哈希值是d(base^1)+...

因此只需要给cefg乘上一个base^3就好了

代码也很好写

#include<bits/stdc++.h>
#define N 100005
#define M 100005
#define ull unsigned long long
const int base=31;
using namespace std;
int n,m,tot,len[N],sum[N],p[N],lc[N],rc[N],rt[N];
ull pw[M];
char s[N];
void build(int &now,int l,int r)
{
    now=++tot;  len[now]=r-l+1;
    if(l==r)
    {
        sum[now]=s[l];
        return;
    }
    int m=(l+r)>>1;
    build(lc[now],l,m);
    build(rc[now],m+1,r);
    sum[now]=sum[lc[now]]+sum[rc[now]]*pw[len[lc[now]]];
}
inline void copy(int &x,int y)  
{
    x=++tot;
    lc[x]=lc[y];    rc[x]=rc[y];    len[x]=len[y];  sum[x]=sum[y];
}
inline void insert(int x,int &y,int l,int r,int p,int v)
{
    copy(y,x);
    if(l==r)    //把x复制一份 给y 
    {
        sum[y]=v;
        return;
    }
    int m=(l+r)>>1;
    if(p<=m)    insert(lc[x],lc[y],l,m,p,v);
    else insert(rc[x],rc[y],m+1,r,p,v);
    sum[y]=sum[lc[y]]+sum[rc[y]]*pw[len[lc[y]]];
}
inline int cmp(int x,int y,int l,int r)
{
    if(l==r)
    {
        if(sum[x]<sum[y])   return -1;
        else return 1;
    }
    int m=(l+r)>>1;
    if(sum[lc[x]]!=sum[lc[y]])  return cmp(lc[x],lc[y],l,m);
    else return cmp(rc[x],rc[y],m+1,r);
}
inline bool comp(const int &a,const int &b)
{
    if(sum[rt[a]]==sum[rt[b]])  return a<b;
    return cmp(rt[a],rt[b],1,m)<0;
} 
int main()
{
    cin>>n>>m;
    scanf("%s",s+1);
    pw[0]=1;
    for(int i=1;i<=m;i++)   pw[i]=pw[i-1]*base;
    build(rt[1],1,m);
    for(int i=2;i<=n;i++)
    {
        int who,pos;
        cin>>who>>pos;
        scanf("%s",s+1);
        insert(rt[who],rt[i],1,m,pos,s[1]); //把以i为根的线段树 接到被膜拜的人身上 
    }
    for(int i=1;i<=n;i++)   p[i]=i;
    sort(p+1,p+n+1,comp);
    for(int i=1;i<=n;i++)   cout<<p[i]<<" ";
    return 0;
}

转载于:https://www.cnblogs.com/Patrickpwq/articles/9799343.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Go语言(也称为Golang)是由Google开发的一种静态强类型、编译型的编程语言。它旨在成为一门简单、高效、安全和并发的编程语言,特别适用于构建高性能的服务器和分布式系统。以下是Go语言的一些主要特点和优势: 简洁性:Go语言的语法简单直观,易于学习和使用。它避免了复杂的语法特性,如继承、重载等,转而采用组合和接口来实现代码的复用和扩展。 高性能:Go语言具有出色的性能,可以媲美C和C++。它使用静态类型系统和编译型语言的优势,能够生成高效的机器码。 并发性:Go语言内置了对并发的支持,通过轻量级的goroutine和channel机制,可以轻松实现并发编程。这使得Go语言在构建高性能的服务器和分布式系统时具有天然的优势。 安全性:Go语言具有强大的类型系统和内存管理机制,能够减少运行时错误和内存泄漏等问题。它还支持编译时检查,可以在编译阶段就发现潜在的问题。 标准库:Go语言的标准库非常丰富,包含了大量的实用功能和工具,如网络编程、文件操作、加密解密等。这使得开发者可以更加专注于业务逻辑的实现,而无需花费太多时间在底层功能的实现上。 跨平台:Go语言支持多种操作系统和平台,包括Windows、Linux、macOS等。它使用统一的构建系统(如Go Modules),可以轻松地跨平台编译和运行代码。 开源和社区支持:Go语言是开源的,具有庞大的社区支持和丰富的资源。开发者可以通过社区获取帮助、分享经验和学习资料。 总之,Go语言是一种简单、高效、安全、并发的编程语言,特别适用于构建高性能的服务器和分布式系统。如果你正在寻找一种易于学习和使用的编程语言,并且需要处理大量的并发请求和数据,那么Go语言可能是一个不错的选择。
你好!感谢你的提问。根据你的要求,我将为你解答关于NOIP2017普及t3问题的内容。 在这道题目中,我们需要解决一个关于棋盘的问题。题目描述了一个n×m的棋盘,每个格子上有一个非负整数。开始时,小明站在左上角的格子上,他可以向右或向下移动一步,每次移动的代价为目标格子上的数值。我们需要找到一条从左上角到右下角的路径,使得移动的总代价最小。 解决这个问题的一个常见的方法是使用动态规划(Dynamic Programming)。我们可以创建一个二维数组dp,其中dp[i][j]表示从起点到达坐标为(i, j)的格子时的最小代价。然后,我们可以按照从左上角到右下角的顺序依次计算dp数组的值。 具体的计算方法如下: 1. 首先,我们可以初始化dp数组的第一行和第一列,即dp[0][j]和dp[i][0],它们表示从起点到达第一行和第一列的格子时的最小代价。初始化的方法是累加前面的格子的代价。 2. 接下来,我们可以使用一个双重循环,从(1, 1)开始遍历整个棋盘。对于每个格子(i, j),我们可以选择从上方格子(i-1, j)或左方格子(i, j-1)中选择一个代价较小的路径,并加上当前格子的代价。即dp[i][j] = min(dp[i-1][j], dp[i][j-1]) + grid[i][j]。 3. 最后,当我们计算完dp数组的所有值后,dp[n-1][m-1]即为从起点到达右下角的格子时的最小代价。 这样,我们就可以得到从左上角到右下角的最小代价。希望能对你的问题有所帮助!如果你还有其他问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值