BZOJ 2038: [2009国家集训队]小Z的袜子【莫队算法裸题】
作为一个生活散漫的人,小Z每天早上都要耗费很久从一堆五颜六色的袜子中找出一双来穿。
终于有一天,小Z再也无法忍受这恼人的找袜子过程,于是他决定听天由命。
具体来说,小Z把这N只袜子从1到N编号,然后从编号L到R的袜子中随机选出两只来穿。
尽管小Z并不在意两只袜子是不是完整的一双,甚至不在意两只袜子是否一左一右,他却很在意袜子的颜色,毕竟穿两只不同色的袜子会很尴尬。
你的任务便是告诉小Z,他有多大的概率抽到两只颜色相同的袜子。
当然,小Z希望这个概率尽量高,所以他可能会询问多个(L,R)以方便自己选择。
输入格式
第一行包含两个正整数N和M,N为袜子的数量,M为小Z所提的询问的数量。
接下来一行包含N个正整数\(C_i\),其中\(C_i\)表示第i只袜子的颜色,相同的颜色用相同的数字表示。
再接下来M行,每行两个正整数L,R表示一个询问。
输出格式
包含M行,对于每个询问在一行中输出分数A/B表示从该询问的区间[L,R]中随机抽出两只袜子颜色相同的概率。
若该概率为0则输出0/1,否则输出的A/B必须为最简分数。
输入样例:
6 4
1 2 3 3 3 2
2 6
1 3
3 5
1 6
输出样例:
2/5
0/1
1/1
4/15
第一次接触莫队算法
kuangbin大佬的分析:莫队算法可以解决一类不修改、离线查询问题。
写了个直接分段解决的办法。把1~n分成sqrt(n)段。unit = sqrt(n)m个查询先按照第几个块排序,再按照 R排序。然后直接求解。
学习笔记:
对于一个区间的概率,就是每种颜色选2个相同的方案数的和/总的选择方案数
化简之后,就是区间内 (每种颜色的数量^2的和-区间长度)/(区间长度*区间长度减1)
问题变为快速求一个区间内每种颜色数量的平方的和
线段树?可以每种颜色单独维护平方,但是会被卡
所以用到了莫队算法
使用范围:
可离线且在得到区间[l,r]的答案后,能在O(1)或O(log2n)得到区间[l,r+1]或[l−1,r]的答案
其实就是找一个数据结构支持插入、删除时维护当前答案。
这样的话,如果已知[l,r]的答案,要求[l’,r’]的答案,我们很容易通过|l – l’|+|r – r’|次转移内求得。
抽象成平面上的点,我们要按一定顺序计算每个值,那开销就为曼哈顿距离的和。曼哈顿距离最小生成树
这里通常用分块解决
n个数分块
按区间排序,以左端点所在块内为第一关键字,右端点为第二关键字,进行排序,
复杂度分析是这样的:
1、\(i\)与\(i+1\)在同一块内,r单调递增,所以r是O(n)的。由于有\(n^{0.5}\)块,所以这一部分时间复杂度是\(n^{1.5}\)。
2、i与i+1跨越一块,r最多变化n,由于有\(n^{0.5}\)块,所以这一部分时间复杂度是\(n^{1.5}\)。
3、i与i+1在同一块内时l变化不超过\(n^{0.5}\),跨越一块也不会超过\(n^{0.5}\),由于有m次询问(和n同级),所以时间复杂度是\(n^{1.5}\).
于是就是\(O(n^{1.5})\)了
AC代码:
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn = 50050;
const int minn = 50050;
struct Query {
int L, R, id;
}node[maxn];
ll gcd(ll a, ll b) { return b == 0 ? a : gcd(b, a % b); }
struct Ans {
ll a, b;
//分数简化操作
void reduce() {
ll d = gcd(a, b);
a /= d, b /= d;
}
}ans[maxn];
int a[maxn], num[maxn], n, m, unit;
//把1~n分成sqrt(n)段,unit=sqrt(n)m个查询先按照第几个块排序,再按照R排序,分块处理
bool cmp(Query a, Query b) {
if (a.L / unit != b.L / unit)
return a.L / unit < b.L / unit;
else return a.R < b.R;
}
void solve() {
ll tmp = 0;
memset(num, false, sizeof num);
int L = 1, R = 0;
//莫队算法核心部分
for (int i = 0; i < m; ++i) {
while (R < node[i].R) {
R++;
tmp -= (ll)num[a[R]] * num[a[R]];
num[a[R]]++;
tmp += (ll)num[a[R]] * num[a[R]];
}
while (R > node[i].R) {
tmp -= (ll)num[a[R]] * num[a[R]];
num[a[R]]--;
tmp += (ll)num[a[R]] * num[a[R]];
R--;
}
while (L < node[i].L)
{
tmp -= (ll)num[a[L]] * num[a[L]];
num[a[L]]--;
tmp += (ll)num[a[L]] * num[a[L]];
L++;
}
while (L > node[i].L)
{
L--;
tmp -= (ll)num[a[L]] * num[a[L]];
num[a[L]]++;
tmp += (ll)num[a[L]] * num[a[L]];
}
ans[node[i].id].a = tmp - (R - L + 1);
ans[node[i].id].b = (ll)(R - L + 1) * (R - L);
ans[node[i].id].reduce();
}
}
int main() {
//freopen("in.txt", "r", stdin);
ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);
while (cin >> n >> m) {
for (int i = 1; i <= n; ++i)cin >> a[i];
for (int i = 0; i < m; ++i) {
node[i].id = i;
cin >> node[i].L >> node[i].R;
}
unit = (int)sqrt(n);
sort(node, node + m, cmp);
solve();
for (int i = 0; i < m; i++)
printf("%lld/%lld\n", ans[i].a, ans[i].b);
}
}
BZOJ 2038: [2009国家集训队]小Z的袜子【莫队算法裸题】的更多相关文章
- BZOJ 2038 [2009国家集训队]小Z的袜子 莫队
2038: [2009国家集训队]小Z的袜子(hose) 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=2038 Descriptionw ...
- [日常摸鱼]bzoj2038[2009国家集训队]小Z的袜子-莫队算法
今天来学了下莫队-这题应该就是这个算法的出处了 一篇别人的blog:https://www.cnblogs.com/Paul-Guderian/p/6933799.html 题意:一个序列,$m$次询 ...
- BZOJ 2038: [2009国家集训队]小Z的袜子
二次联通门 : BZOJ 2038: [2009国家集训队]小Z的袜子 /* BZOJ 2038: [2009国家集训队]小Z的袜子 莫队经典题 但是我并不认为此题适合入门.. Answer = ∑ ...
- BZOJ 2038: [2009国家集训队]小Z的袜子(hose) [莫队算法]【学习笔记】
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 7687 Solved: 3516[Subm ...
- BZOJ 2038: [2009国家集训队]小Z的袜子(hose)
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 7676 Solved: 3509[Subm ...
- Bzoj 2038: [2009国家集训队]小Z的袜子(hose) 莫队,分块,暴力
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 5763 Solved: 2660[Subm ...
- BZOJ 2038: [2009国家集训队]小Z的袜子(hose) ( 莫队 )
莫队..先按sqrt(n)分块, 然后按块的顺序对询问排序, 同块就按右端点排序. 然后就按排序后的顺序暴力求解即可. 时间复杂度O(n1.5) --------------------------- ...
- BZOJ 2038: [2009国家集训队]小Z的袜子(hose) 分块
分块大法好 2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MB Submit: 2938 Solved: 13 ...
- BZOJ 2038: [2009国家集训队]小Z的袜子(hose)【莫队算法裸题&&学习笔记】
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 9894 Solved: 4561[Subm ...
- BZOJ 2038: [2009国家集训队]小Z的袜子(hose) 【莫队算法模版】
任意门:https://www.lydsy.com/JudgeOnline/problem.php?id=2038 题意概括: 有 N 只袜子(分别编号为1~N),有 M 次查询 (L, R)里面随机 ...
随机推荐
- 【题解】HD2016.X1,HD2016.X3,HD2016.X4,HD2016.X5
[HD2016.X1] 价钱统计 题目描述 夏天到了,超市里摆满了各种各样的应季水果.现在知道:西瓜的价钱是每斤 1.2 元:桃子的价钱是每斤 3.5 元:葡萄的价钱是每斤 4.5 元:苹果的价钱是每 ...
- C#使用SqlSugar操作MySQL数据库实现简单的增删改查
公众号「DotNet学习交流」,分享学习DotNet的点滴. SqlSugar简介 SqlSugar 是一款 老牌 .NET 开源多库架构ORM框架(EF Core单库架构),由果糖大数据科技团队 维 ...
- [编程] AI助力软件项目正向生成,注释编写的革命
引言 软件项目质量直接影响着用户体验和企业效益.随着软件的应用范围不断扩大,提高软件质量的重要性也日益凸显.传统上,软件工程师通常采用自下而上的开发模式,自行设计实现代码并进行测试,这给质量把控带来一 ...
- 使用 VS 2019 将 c# 生成 DLL 动态链接库文件
主要步骤: ChatGPT 的回答: 你可以尝试使用 Visual Studio 创建一个类库项目,然后将你写的两个类添加到该项目中,并进行编译,最终生成 DLL 文件.具体步骤如下: 打开 Visu ...
- Linux常用命令(持续完善中......)
1.查看内存 top 2.查看磁盘存储情况 df -h 3.查看端口占用情况 netstat -tunlp | grep 端口号 4.查看报告系统运行时长及平均负载 uptime 5.查看进程 ps ...
- .net 温故知新【16】:Asp.Net Core WebAPI 筛选器
一.筛选器 通过使用筛选器可在请求处理管道中的特定阶段之前或之后运行代码. 这即是我们经常听到的面向切面编程AOP(Aspect Oriented Programming)技术,AOP通过预编译方式和 ...
- 数字孪生系统为何需要将GIS系统进行融合?
数字孪生是一种通过数字模型实时仿真现实世界的技术,而GIS(地理信息系统)则是用于收集.存储.处理和展示地理数据的工具.将数字孪生系统与GIS系统进行融合,可以为各行业带来诸多优势和创新.那么数字孪生 ...
- ElasticSearch之线程池
ElasticSearch节点可用的CPU核的数量,通常可以交给ElasticSearch来自行检测和判定,另外可以在``elasticsearch.yml`中显式指定.样例如下: node.proc ...
- Python——Html(表格<table>, <tr>,<td>,<th>、表单<form>、自定义标签<div>和<span>)
一.表格<table>, <tr>,<td>或<th> <table> 元素是 HTML 中用于创建表格的主要标记.表格是一种用于展示数据的 ...
- go的html模板template格式化时间
go的html模板template格式化时间 go的html模板template格式化时间,网上一搜挺尴尬找不到想要的yyyy-MM-dd HH:mm:ss // 代码中是这样的 //2021-09- ...