谁是天才(续)

本文介绍了一种利用线段树解决特定字符串操作问题的方法,包括找到字典序最小的字符串后,如何通过查询确定原字符串中对应位置的字典序最大字符。

Description

张大牛:“我是天才!”
大肥熊:“你为什么是天才?”
张大牛:“你随便给我一个单词(大小写字母组成)长度为N,去掉M个字符后,我能知道字典序最小的字符串是什么样子的”
大肥熊:“换过来,现在假设这个字典序最小的字符串中第ai个字符在原串中的位置为pos,那么原串中区间[pos-ki,pos+ki]中字典序最大的字符是什么?”
张大牛又被难倒了。现在这个难倒天才的题目交到你手上了。

注:区间[ pos-ki , pos+ki ]超过原字符串的边界,按照原字符串的边界处理。原串的整个区间为[1,N],如果pos可以取多个值,按照最靠前的位置计算。

Input

输入文件名为(genius.in)。
第一行M,T。M如题意,T为T组询问。
第二行为一个由大、小写字母组成的字符串。串的长度N(N>M>0)。
接下来T行每行两个正整数ai,ki(1<=ai<=N-M),(1<=ki<=N)。

Output

输出文件名为(genius.out)。
T行,每行对询问给出一个答案。

Sample Input

20 2
abcdefghijklmnopqrstuvwxyz
5 20
3 3

Sample Output

y
f

Hint

对于100%数据,N<=10^5   T<=10^5

Key To Problem

乍一看,貌似是RMQ的题,但是RMQ是什么我不会写啊!所以我用的线段树水过。
维护区间最大和区间最小两个线段树,用区间最小的线段树来求字典序最小的子串,记录子串在字符串中的位置,之后对于每个查询就可以用区间最大的线段树搞出来。

Code

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define N 100010
#define ls rt<<1
#define rs rt<<1|1
#define lson l,mid,ls
#define rson mid+1,r,rs
using namespace std;
struct node
{
    int s,x;
};
node mx[N<<2],mn[N<<2];
int n,m,q;
char s[N];
int a[N];
int f[N];

void PushUp(int rt)
{
    if(mx[ls].s>=mx[rs].s)
        mx[rt]=mx[ls];
    else
        mx[rt]=mx[rs];
    if(mn[ls].s<=mn[rs].s)
        mn[rt]=mn[ls];
    else
        mn[rt]=mn[rs];
}

void build(int l,int r,int rt)
{
    if(l==r)
    {
        mn[rt].s=mx[rt].s=a[l];
        mn[rt].x=mx[rt].x=l;
        return ;
    }
    int mid=(l+r)>>1;
    build(lson);
    build(rson);
    PushUp(rt);
}

node query_mx(int L,int R,int l,int r,int rt)
{
    if(l>=L&&r<=R)
        return mx[rt];
    int mid=(l+r)>>1;
    node a,b,s;
    if(mid>=L)
        a=query_mx(L,R,lson);
    if(mid<R)
        b=query_mx(L,R,rson);
    if(mid<L)
        return b;
    if(mid>=R)
        return a;
    if(a.s>=b.s)
        s=a;
    else
        s=b;
    return s;
}

node query_mn(int L,int R,int l,int r,int rt)
{
    if(l>=L&&r<=R)
        return mn[rt];
    int mid=(l+r)>>1;
    node a,b,s;
    if(mid>=L)
        a=query_mn(L,R,lson);
    if(mid<R)
        b=query_mn(L,R,rson);
    if(mid<L)
        return b;
    if(mid>=R)
        return a;
    if(a.s<=b.s)
        s=a;
    else
        s=b;
    return s;
}

int main()
{
//  freopen("genius.in","r",stdin);
//  freopen("genius.out","w",stdout);
    scanf("%d%d%s",&m,&q,s+1);
    n=strlen(s+1);
    m=n-m;
    for(int i=1;i<=n;i++)
    {
        if(s[i]>='A'&&s[i]<='Z')
            a[i]=s[i]-'A'+1;
        else
            a[i]=s[i]-'a'+27;
    }
    build(1,n,1);
    int t=n-m,s=1,al=0;
    while(++t&&al<m)
    {
        node x=query_mn(s,t,1,n,1);
        f[++al]=x.x;
        s=x.x+1;
    }
    for(int i=1;i<=q;i++)
    {
        int x,y;
        scanf("%d%d",&x,&y);
        int l=max(f[x]-y,1);
        int r=min(f[x]+y,n);
        int z=query_mx(l,r,1,n,1).s;
        if(z<=26)
            printf("%c\n",z-1+'A');
        else
            printf("%c\n",z-27+'a');
    }
    return 0;
}
RocketMQ 中,**分片(Sharding)** 是实现高并发、高可用和可展性的重要机制之一。其核心作用是将消息的存储和消费进行分布式管理,以支持大规模消息处理场景。 ### 分片的作用 1. **提升系统吞吐量** 通过将一个 Topic 的消息分布到多个 Broker 上,每个 Broker 负责一部分消息的存储和转发,从而实现横向展,提高整体系统的吞吐能力。 2. **支持负载均衡** 在消息生产与消费过程中,分片机制使得消息可以均匀分布在多个 Broker 上,生产者和消费者可以并行地处理多个分片,实现负载均衡[^5]。 3. **增强系统可用性与容错性** 每个分片可以配置主从结构(Master-Slave),实现数据复制与故障切换,确保在某个 Broker 故障时仍能保证消息的高可用[^4]。 ### 分片的工作机制 1. **Topic 与 Message Queue 的关系** 在 RocketMQ 中,每个 Topic 会被划分为多个 **Message Queue**(也称为队列或分片),这些队列分布在不同的 Broker 上。例如,一个 Topic 可能有 4 个队列,分别分布在两个 Broker 上,每个 Broker 管理两个队列。 2. **生产者的分片选择** 当生产者发送消息时,会根据一定的策略(如轮询、哈希等)选择一个合适的 Message Queue 进行投递。这一过程称为**生产者负载均衡**。生产者会定期从 NameServer 获取 Topic 的队列分布信息,以保证选择的准确性[^5]。 3. **消费者的分片分配** 消费者组(ConsumerGroup)中的每个消费者实例会负责一部分 Message Queue 的消费任务。这一过程称为**消费者负载均衡**,由 Broker 协调完成,确保每个队列只被一个消费者实例消费,从而避免重复消费和竞争问题。 4. **消息的物理存储** RocketMQ 将所有消息写入统一的 **CommitLog** 文件中,然后通过 **ConsumeQueue** 文件记录每个 Topic 的分片索引信息,实现逻辑分片与物理存储的分离。这种机制保证了写入的高效性和读取的灵活性[^4]。 ### 分片配置与管理 - **创建 Topic 时指定分片数量** 在创建 Topic 时,可以通过命令行或配置文件指定其分片数量(即 Message Queue 数量)。 - **动态容** 可以在不中断服务的情况下,向集群中新增 Broker,并为已有 Topic 增加分片,以应对不断增长的消息量。 ### 示例代码:查看 Topic 分片信息 ```bash # 查看 Topic 的队列分布信息 mqadmin topicRoute -n localhost:9876 -t MyTopic ``` 该命令将输出 Topic `MyTopic` 的路由信息,包括各个 Message Queue 所在的 Broker 地址。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值