题面

有 n 个学生和 n 所大学,每个学生在其中一所大学中学习,且各有一个能力值

s

i

s_i

si​ 。

某次组队打比赛的召集令会给一个数字 k ,表示团队数量。然后每所大学会先把自己的所有学生按照

a

i

a_i

ai​ 从大到小排序,选前

k

k

k 个组个队,前

k

+

1

k+1

k+1 到

2

k

2k

2k 个组个队,……剩下最后不足

k

k

k 个学生,这些学生就不能组队。

每次召集的总能力值为所有组出来的队伍的每个学生的能力值之和。现在有

n

n

n 次召集令,给出的

k

k

k 分别是 1~n,分别求每次召集的总能力值。

题解

我这个做法被 nlogn 做法吊打,本愧于过此题,然所用方法有点思维,不如写来搏之一笑。

分别求每个学生的贡献。

假设当前学生在他(她)的大学里排名为倒数第

y

y

y ,而大学里总共

x

x

x 个学生,那么该学生对数字为

k

k

k 的召集令有贡献当且仅当

x
 ⁣ ⁣ ⁣ ⁣

m

o

d

  

k

<

y

x\!\!\!\!\mod k<y

xmodk<y
变一下式子:

x

x

k

k

<

y

x

y

<

x

k

k

x

y

k

<

x

k

x-\left\lfloor \frac{x}{k}\right\rfloor*k<y\\ ~~\Leftrightarrow~~ x-y<\left\lfloor \frac{x}{k}\right\rfloor*k\\ ~~\Leftrightarrow~~ \left\lfloor \frac{x-y}{k}\right\rfloor<\left\lfloor \frac{x}{k}\right\rfloor

x−⌊kx​⌋∗k<y  ⇔  x−y<⌊kx​⌋∗k  ⇔  ⌊kx−y​⌋<⌊kx​⌋

如果我们已知

x

k

=

d

\left\lfloor \frac{x}{k}\right\rfloor=d

⌊kx​⌋=d,那么

x

y

k

<

d

x

y

<

d

k

x

y

d

<

k

\left\lfloor \frac{x-y}{k}\right\rfloor<d\\ ~~\Leftrightarrow~~ x-y<dk\\ ~~\Leftrightarrow~~ \left\lfloor \frac{x-y}{d}\right\rfloor<k

⌊kx−y​⌋<d  ⇔  x−y<dk  ⇔  ⌊dx−y​⌋<k

好,这是个关于

k

k

k 的范围的表达式了,由于我们知道

x

k

\left\lfloor \frac{x}{k}\right\rfloor

⌊kx​⌋ 随着

k

k

k 的不同只有大约

x

\sqrt x

x

​ 个取值,因此我们可以数论分块枚举,每次枚举到一个区间

[

l

,

r

]

[l,r]

[l,r] 和

d

d

d,就对答案序列的

[

max

(

x

y

d

+

1

,

l

)

,

r

]

[\max(\left\lfloor \frac{x-y}{d}\right\rfloor+1,l),r]

[max(⌊dx−y​⌋+1,l),r] 产生贡献。

对每个学生都计算一次,复杂度

O

(

n

n

)

O(n\sqrt n)

O(nn

​)。

CODE

#include<set>
#include<queue>
#include<cmath>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define MAXN 200005
#define ENDL putchar('\n')
#define LL long long
#define DB double
#define lowbit(x) ((-x) & (x))
LL read() {
LL f = 1,x = 0;char s = getchar();
while(s < '0' || s > '9') {if(s=='-')f = -f;s = getchar();}
while(s >= '0' && s <= '9') {x=x*10+(s-'0');s = getchar();}
return f * x;
}
int n,m,i,j,s,o,k;
int a[MAXN];
vector<int> u[MAXN];
LL sm[MAXN];
bool cmp(int x,int y) {return a[x] > a[y];}
int main() {
int T = read();
while(T --) {
n = read();
for(int i = 1;i <= n;i ++) u[i].clear(),sm[i] = 0;
for(int i = 1;i <= n;i ++) {
s = read(); u[s].push_back(i);
}
for(int i = 1;i <= n;i ++) {
a[i] = read();
}
for(int i = 1;i <= n;i ++) {
sort(u[i].begin(),u[i].end(),cmp);
int X = u[i].size();
for(int j = 0,nm = X;j < (int)u[i].size();j ++,nm --) {
int con = a[u[i][j]];
sm[1] += con; sm[nm+1] -= con;
for(int l = nm+1,r = 1;l <= X;l = r+1) {
r = X/(X/l); int d = X / l;
int ll = max(l,((X-nm)/d) + 1);
if(ll <= r) {
sm[ll] += con; sm[r+1] -= con;
}
}
}
}
for(int i = 1;i <= n;i ++) {
sm[i] += sm[i-1];
printf("%lld ",sm[i]);
}ENDL;
}
return 0;
}

[CF1519C] Berland Regional (数论分块)的更多相关文章

  1. 【BZOJ1257】余数之和(数论分块,暴力)

    [BZOJ1257]余数之和(数论分块,暴力) 题解 Description 给出正整数n和k,计算j(n, k)=k mod 1 + k mod 2 + k mod 3 + - + k mod n的 ...

  2. 51nod“省选”模测第二场 B 异或约数和(数论分块)

    题意 题目链接 Sol 这题是来搞笑的吧.. 考虑一个数的贡献是\(O(\frac{N}{i})\) 直接数论分块. #include<bits/stdc++.h> #define Pai ...

  3. 洛谷P2261 [CQOI2007] 余数求和 [数论分块]

    题目传送门 余数求和 题目背景 数学题,无背景 题目描述 给出正整数n和k,计算G(n, k)=k mod 1 + k mod 2 + k mod 3 + … + k mod n的值,其中k mod ...

  4. luoguP3235 [HNOI2014]江南乐 数论分块 + 博弈论

    感觉其实很水? 题目就是一个Multi SG游戏,只需要预处理出所有的\(sg\)值即可\(O(Tn)\)计算 对于计算\(sg[n]\)而言,显然我们可以枚举划分了\(x\)堆来查看后继状态 那么, ...

  5. bzoj 3834 [Poi2014]Solar Panels 数论分块

    3834: [Poi2014]Solar Panels Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 367  Solved: 285[Submit] ...

  6. 洛谷P1403 [AHOI2005] 约数研究 [数论分块]

    题目传送门 约数研究 题目描述 科学家们在Samuel星球上的探险得到了丰富的能源储备,这使得空间站中大型计算机“Samuel II”的长时间运算成为了可能.由于在去年一年的辛苦工作取得了不错的成绩, ...

  7. 「BZOJ 2440」完全平方数「数论分块」

    题意 \(T\)组数据,每次询问第\(k\)个无平方因子的数(\(1\)不算平方因子),\(T\leq 50,k\leq 10^9\) 题解 \(k\)的范围很大,枚举肯定不行,也没什么奇妙性质,于是 ...

  8. bzoj 1257 余数之和 —— 数论分块

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1257 \( \sum\limits_{i=1}^{n}k\%i = \sum\limits_ ...

  9. 【数论分块】bzoj2956: 模积和

    数论分块并不精通……第一次调了一个多小时才搞到60pts:因为不会处理i==j的情况,只能枚举了…… Description $\sum_{i=1}^{n}\sum_{j=1 \land i \not ...

随机推荐

  1. 编程式导航路由跳转到当前路由(参数不变), 多次执行会抛出NavigationDuplicated的警告错误?

    注意:编程式导航(push|replace)才会有这种情况的异常,声明式导航是没有这种问题,因为声明式导航内部已经解决这种问题. 这种异常,对于程序没有任何影响的. 为什么会出现这种现象: 由于vue ...

  2. c++ 超长整数加法 高精度加法

    c++ 超长整数加法 高精度加法 实现思路 不能直接使用加法,因为int和long long都已超出最大数据表示范围 数据读入采用string类型,读入后将数据的每一位存储到vector中 vecto ...

  3. 十分钟快速实战Three.js

    前言 本文不会对Three.js几何体.材质.相机.模型.光源等概念详细讲解,会首先分成几个模块给大家快速演示一盒小案例.大家可以根据这几个模块快速了解Three.js的无限魅力.学习 我们会使用Th ...

  4. 好用到爆!GitHub 星标 32.5k+的命令行软件管理神器,功能真心强大!

    前言(废话) 本来打算在公司偷偷摸摸给星球的用户写一篇编程喵整合 MongoDB 的文章,结果在通过 brew 安装 MongoDB 的时候竟然报错了.原因很简单,公司这台 Mac 上的 homebr ...

  5. sql-扩展sql

    复制表结构 create table 表名 like 被复制的表名; -- 仅复制表表结构 oracle不支持 复制表结构和数据(子查询方式) CREATE TABLE 表名 [AS] SELECT ...

  6. Linux 安装Apche服务

    用yum 进行在线安装apche服务 yum install -y httpd 我这边是centos7 需要开启一下端口: 1 firewall-cmd --zone=public --add-por ...

  7. 星际争霸的虫王IA退役2年搞AI,自叹不如了

    ------------恢复内容开始------------ 金磊 发自 凹非寺 量子位|公众号 QbitA 这年头,直播讲AI,真算不上什么新鲜事.但要是连职业电竞选手,都开播主讲呢?没开玩笑,是真 ...

  8. go语言学习笔记-初识Go语言

    Go语言是怎样诞生的? Go语言的创始人有三位,分别是图灵奖获得者.C语法联合发明人.Unix之父肯·汤普森(Ken Thompson).Plan 9操作系统领导者.UTF-8编码的最初设计者罗伯·派 ...

  9. 编写可维护的webpack配置

    为什么要构建配置抽离成npm包 通用性 业务开发者无需挂住配置 统一团队构建脚本 可维护性 构建配置合理的拆分 README文档, chan 构建配置管理的可选方案 通过多个配置管理不同环境的构建, ...

  10. SpringBoot快速整合通用Mapper

    前言 后端业务开发,每个表都要用到单表的增删改查等通用方法,而配置了通用Mapper可以极大的方便使用Mybatis单表的增删改查操作. 通用mapper配置 1.添加maven: <depen ...