题目大意: 给定一个长度不超过10^5的字符串(小写英文字母),和不超过5000个操作。

每个操作 L R K 表示给区间[L,R]的字符串排序,K=1为升序,K=0为降序。

最后输出最终的字符串

首先这么想想,对于一段区间的排序,排完序的样子和排序之前每个字母的位置并没有关系,而是和每一个字母出现的次数有关。所以我们对于每一次操作,统计出区间中每一个字母出现了多少次,然后按字典序排序就行。更确切的说,就是这个区间中的哪一个部分都改成某一个字母,区间修改。

既然是区间修改,那么就可以用线段树实现。不过这样的话,打lazy标记就显得不是很方便。为此,我们可以开26个线段树,每一个字母开一个长度为n的权值线段树,如果第i为是这个字母,我们就把这一位改成1,然后统计区间中这个字母有多少个,就相当于求区间和了。至于修改,那就是将这个字母所在线段树的区间全都改成1.然后把操作区间的别的地方改成0即可。

举个栗子:

acbcaab

然后将[1, 6]按升序排序。

那么我们首先分别在a, b, c所在的线段树上查到了[1, 6]的区间和,即统计出了每个字母的出现次数。

然后排序的时候,对于a所在的线段树,我们将[1, 3]都改成了1,[4, 6]改成了0;对于b所在线段树,我们将[4, 4]改成了1,[1, 3]和[5, 6]改成了0;对于c,我们将[5, 6]改成了1,[1, 4]改成了0.

这样这个区间就排完序了。

那么怎么输出最终答案呢?

只要对于每一位,暴力的从0到26循环,看哪个字母在这一位上是1,就说明这一位是这个字母了。

配合break,时间复杂度最坏为O(nlogn * 26)

 #include<cstdio>
#include<iostream>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<cstdlib>
#include<cctype>
#include<vector>
#include<stack>
#include<queue>
using namespace std;
#define enter printf("\n")
#define space printf(" ")
#define Mem(a) memset(a, 0, sizeof(a))
typedef long long ll;
typedef double db;
const int INF = 0x3f3f3f3f;
const db eps = 1e-;
const int maxn = 2e7 + ;
inline int read()
{
int ans = ;
char ch = getchar(), last = ' ';
while(!isdigit(ch)) {last = ch; ch = getchar();}
while(isdigit(ch))
{
ans = ans * + ch - ''; ch = getchar();
}
if(last == '-') ans = -ans;
return ans;
}
inline void write(ll x)
{
if(x < ) x = -x, putchar('-');
if(x >= ) write(x / );
putchar('' + x % );
} int n, q;
char s[]; int cnt = , root[], lson[maxn], rson[maxn], l[maxn], r[maxn];
//lson[now]和rson[now]分别记录now的左右儿子的编号,代替了now << 1和 now <<1 | 1
int sum[maxn], lazy[maxn];
void build(int& now, int L, int R) //我这个写法是先吧所有点开好了,不是动态开点(竟然比某位大佬的动态开点快)
{
now = ++cnt; lazy[now] = -;
l[now] = L; r[now] = R;
if(L == R) return;
int mid = (L + R) >> ;
build(lson[now], L, mid);
build(rson[now], mid + , R);
}
void add(int now, int id)
{
if(l[now] == r[now]) {sum[now]++; return;}
int mid = (l[now] + r[now]) >> ;
if(id <= mid) add(lson[now], id);
else add(rson[now], id);
sum[now] = sum[lson[now]] + sum[rson[now]];
}
void pushdown(int now)
{
if(lazy[now] != -) //因为lazy[now]=0代表将区间都改为0,所以没有标记要换一个记号
{
sum[lson[now]] = (r[lson[now]] - l[lson[now]] + ) * lazy[now];
sum[rson[now]] = (r[rson[now]] - l[rson[now]] + ) * lazy[now];
lazy[lson[now]] = lazy[now];
lazy[rson[now]] = lazy[now];
lazy[now] = -;
} }
void update(int now, int L, int R, int d)
{
if(L == l[now] && R == r[now])
{
sum[now] = (r[now] - l[now] + ) * d;
lazy[now] = d; return;
}
pushdown(now);
int mid = (l[now] + r[now]) >> ;
if(R <= mid) update(lson[now], L, R, d);
else if(L > mid) update(rson[now], L, R, d);
else {update(lson[now], L, mid, d); update(rson[now], mid + , R, d);}
sum[now] = sum[lson[now]] + sum[rson[now]];
}
int query(int now, int L, int R)
{
if(!sum[now]) return ; //优化
if(L == l[now] && R == r[now]) return sum[now];
pushdown(now);
int mid = (l[now] + r[now]) >> ;
if(R <= mid) return query(lson[now], L, R);
else if(L > mid) return query(rson[now], L, R);
else return query(lson[now], L, mid) + query(rson[now], mid + , R);
} int main()
{
n = read(); q = read();
scanf("%s", s + );
for(int i = ; i < ; ++i) build(root[i], , n);
for(int i = ; i <= n; ++i) add(root[s[i] - 'a'], i);
for(int i = ; i <= q; ++i)
{
int L = read(), R = read(), k = read();
if(k)
{
int pre = L - ;
for(int j = ; j < ; ++j) //枚举每一棵线段树
{
int ssum = query(root[j], L, R);
if(ssum)
{
update(root[j],L,R,); //先都改成0,在局部覆盖1
update(root[j], pre + , pre + ssum, );
}
pre += ssum;
}
}
else
{
int pre = L - ;
for(int j = ; j >= ; --j) //降序,就倒着枚举
{
int ssum = query(root[j], L, R);
if(ssum)
{
update(root[j],L,R,);
update(root[j], pre + , pre + ssum, );
}
pre += ssum;
}
}
}
for(int i = ; i <= n; ++i) //很暴力的查询
for(int j = ; j < ; ++j)
if(query(root[j], i, i)) {printf("%c", 'a' + j); break;}
enter;
return ;
}

这道题时限5秒,然而还特别容易TLE,所以得做好常数优化工作。

据说某位大佬线段树上每个节点记录26个字母出现的情况,所以只开了一棵线段树,自然就十分的快了,毫无TLE的烦恼。(很显然,我不会写,要不就不讲上述的方法了……)

CF558E A Simple Task的更多相关文章

  1. CF558E A simple task 线段树

    这道题好猥琐啊啊啊啊啊啊 写了一个上午啊啊啊啊 没有在update里写pushup啊啊啊啊 题目大意: 给你一个字符串s,有q个操作 l r 1 :把sl..rsl..r按升序排序 l r 0 :把s ...

  2. 计数排序 + 线段树优化 --- Codeforces 558E : A Simple Task

    E. A Simple Task Problem's Link: http://codeforces.com/problemset/problem/558/E Mean: 给定一个字符串,有q次操作, ...

  3. HDU-1339 A Simple Task

    http://acm.hdu.edu.cn/showproblem.php?pid=1339 正常做法超时,要有点小技巧存在. A Simple Task Time Limit: 2000/1000 ...

  4. A Simple Task

    A Simple Task Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Tot ...

  5. Codeforces 558E A Simple Task (计数排序&&线段树优化)

    题目链接:http://codeforces.com/contest/558/problem/E E. A Simple Task time limit per test5 seconds memor ...

  6. Codeforces Round #312 (Div. 2) E. A Simple Task 线段树

    E. A Simple Task 题目连接: http://www.codeforces.com/contest/558/problem/E Description This task is very ...

  7. Codeforces Round #312 (Div. 2) E. A Simple Task 线段树+计数排序

    题目链接: http://codeforces.com/problemset/problem/558/E E. A Simple Task time limit per test5 secondsme ...

  8. 【题解】 CF11D A Simple Task

    [题解] CF11D A Simple Task 传送门 \(n \le 20\) 考虑状态压缩\(dp\). 考虑状态,\(dp(i,j,O)\)表示从\(i\)到\(j\)经过点集\(O\)的路径 ...

  9. Codeforces 558E A Simple Task(权值线段树)

    题目链接  A Simple Task 题意  给出一个小写字母序列和若干操作.每个操作为对给定区间进行升序排序或降序排序. 考虑权值线段树. 建立26棵权值线段树.每次操作的时候先把26棵线段树上的 ...

随机推荐

  1. IIS日志自动清理

    IIS在运行的过程中日志会不停地增长,若iis的网站被频繁的调用或不当的调用,则会产生很多日志.我在系统运维的时候曾出现过20G的系统盘,由于合作商开发的程序有问题,每几百微秒调用一次web服务,短期 ...

  2. SWT table性能改善 -- 使用VirtualTable

    在SWT程序中使用table展示数据时,如果数据过多,执行起来会比较慢,不过,我们可以借助VirtualTable来解决这一问题. Eclipse官网中关于VirtualTable的说明见:http: ...

  3. VirtualBox配置centos7静态ip(详解)

    VirtualBox安装centos7配置静态ip地址可以本机访问,可以联网. 在开始之前先说一下,不知道为什么,我在网上百度的大多数是不能用的,或者只能主机访问,或者只能联网. 我的配置文件为ifc ...

  4. hadoop的namenode故障处理方法

    Namenode 故障后,可以采用如下两种方法恢复数据. 方法一:将 SecondaryNameNode 中数据拷贝到 namenode 存储数据的目录: 方法 二: 使用 -importCheckp ...

  5. 【代码笔记】iOS-产生随机数

    一,代码. - (void)viewDidLoad { [super viewDidLoad]; // Do any additional setup after loading the view, ...

  6. List中Add()与AddAll()的区别

    我们在开发过程中经常会使用到List<Object> list=new ArrrayList<>(); 这个集合,Object 也可以是String.Integer等. 当我们 ...

  7. vue 自定义组件的自定义属性

    <auto-com :value="value"></auto-com> //带 : 的属性传入的是动态的值 <auto-com value=&quo ...

  8. 图像增强算法(直方图均衡化、拉普拉斯、Log、伽马变换)

    一.图像增强算法原理 图像增强算法常见于对图像的亮度.对比度.饱和度.色调等进行调节,增加其清晰度,减少噪点等.图像增强往往经过多个算法的组合,完成上述功能,比如图像去燥等同于低通滤波器,增加清晰度则 ...

  9. php 实现简单加入购物车(1)

          这个购物车相对来说比较简单,用于短暂存储,并没有存储到数据库,购物车对于爱网购的人来说简直是熟悉的不能再熟悉了,在写购物车之前,我们首先要构思一下,我们需要先从数据库中调出一张表格,这里我 ...

  10. JavaScript Data.parse()转化时间戳安卓和ISO不兼容

    Data.parse()获取时间戳,在Android是没有问题的,但是在ISO就不行了,原因在于转化成时间戳的时间格式不一样. Android的格式是如“2017-12-12 12:12:12”,IS ...