Codeforces 题目传送门 & 洛谷题目传送门

下设 \(n,m\) 同阶。

首先有一个傻子都会的暴力做法,枚举矩形的上、下边界 \(l,r\),考虑集合多重集 \(S=\{y|x\in[l,r],(x,y)\text{为标记点}\}\),也就是所有在第 \(l\) 行与第 \(r\) 行之间所有标记点的列坐标的集合,考虑进一步枚举矩形的左端点 \(x\),那么对于右端点 \(y\),以 \((l,x)\) 为左上角,\((r,y)\) 为右下角的矩形中至少有 \(k\) 个标记点当前仅当 \(S\) 中有至少 \(k\) 个元素 \(\in[x,y]\),故考虑用 two pointers 求出这个 \(y\),时间复杂度 \(n^3\)。

考虑优化这个过程,注意到 \(k\) 最多只有 \(10\),因此考虑从此下手。我们将 \(S\) 中的元素从小到大排个序,记作 \(x_1,x_2,\cdots,x_{|S|}\),记 \(nxt_x\) 为当矩形的左边界为 \(x\) 时候最小的右边界 \(y\) 使得 \(S\) 中至少有 \(k\) 个元素 \(\in[x,y]\),那么显然 \(\forall v\in(x_{i-1},x_i],nxt_v=x_{i+k-1}\),因此 \(i\) 会对答案产生 \((x_i-x_{i-1})\times(m-x_{i+k-1})\) 的贡献,我们再记 \(f_i=(x_i-x_{i-1})\times(m-x_{i+k-1})\),那么显然矩形矩形的上、下边界分别为 \(l,r\) 时的答案就是 \(\sum\limits_{i=1}^{|S|}f_i\)。

我们考虑还是枚举矩形的上边界、下边界,并在枚举的过程中动态维护 \(f_i\) 的变化。注意到你每加入/删除一个点,最多会影响 \(k+1\) 个 \(f_i\)。因此每次加入/删除时暴力修改 \(f_i\) 都没问题。那么现在问题就来了,怎么修改 \(f_i\) 呢?一个很显然的思路是 std::multiset<int>,不过非常抱歉,std::multiset<int> 会平白多一个 \(\log\),复杂度 \(n^2k\log n\),再加上 set 的大常数,无法通过此题。注意到每次修改我们只用找出前驱、后继,因此考虑使用擅长 \(\mathcal O(1)\) 求出前驱、后继的双向链表,这样就可以把 set 的 \(\log\) 去掉了。还有一点,就是双向链表不支持随机访问,因此使用双向链表动态加点是实现不了了。故考虑换个思路,既然双向链表不支持动态加点,我们就考虑动态删点,首先将横坐标 \(\ge l\) 的点按照纵坐标从小到大的顺序加入双向链表,然后每扫过一行就将这一行中的标记点从双向链表中删除即可。复杂度 \(n^2k\)。

细节有亿点点多,大约也就调了 2h 罢(((

#include <bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define fill0(a) memset(a,0,sizeof(a))
#define fill1(a) memset(a,-1,sizeof(a))
#define fillbig(a) memset(a,63,sizeof(a))
#define pb push_back
#define ppb pop_back
#define mp make_pair
template<typename T1,typename T2> void chkmin(T1 &x,T2 y){if(x>y) x=y;}
template<typename T1,typename T2> void chkmax(T1 &x,T2 y){if(x<y) x=y;}
typedef pair<int,int> pii;
typedef long long ll;
typedef unsigned int u32;
typedef unsigned long long u64;
namespace fastio{
#define FILE_SIZE 1<<23
char rbuf[FILE_SIZE],*p1=rbuf,*p2=rbuf,wbuf[FILE_SIZE],*p3=wbuf;
inline char getc(){return p1==p2&&(p2=(p1=rbuf)+fread(rbuf,1,FILE_SIZE,stdin),p1==p2)?-1:*p1++;}
inline void putc(char x){(*p3++=x);}
template<typename T> void read(T &x){
x=0;char c=getchar();T neg=0;
while(!isdigit(c)) neg|=!(c^'-'),c=getchar();
while(isdigit(c)) x=(x<<3)+(x<<1)+(c^48),c=getchar();
if(neg) x=(~x)+1;
}
template<typename T> void recursive_print(T x){if(!x) return;recursive_print(x/10);putc(x%10^48);}
template<typename T> void print(T x){if(!x) putc('0');if(x<0) putc('-'),x=~x+1;recursive_print(x);}
void print_final(){fwrite(wbuf,1,p3-wbuf,stdout);}
}
const int MAXN=3e3;
int R,C,n,k,x[MAXN+5],y[MAXN+5];
vector<int> vr[MAXN+5],vc[MAXN+5];
int l[MAXN+5],r[MAXN+5];ll ans=0;
int main(){
scanf("%d%d%d%d",&R,&C,&n,&k);
for(int i=1;i<=n;i++){
scanf("%d%d",&x[i],&y[i]);
vr[x[i]].pb(i);vc[y[i]].pb(i);
}
for(int i=1;i<=MAXN;i++) sort(vr[i].begin(),vr[i].end(),[](int lhs,int rhs){return y[lhs]>y[rhs];});
for(int i=1;i<=MAXN;i++) sort(vc[i].begin(),vc[i].end(),[](int lhs,int rhs){return x[lhs]<x[rhs];});
for(int i=1;i<=R;i++){
memset(l,0,sizeof(l));memset(r,0,sizeof(r));
int pre=0;
for(int j=1;j<=C;j++) for(int t=0;t<vc[j].size();t++)
if(x[vc[j][t]]>=i){
int id=vc[j][t];l[id]=pre;
if(pre) r[pre]=id;pre=id;
}
ll ret=0;
for(int j=1;j<=C;j++) for(int t=0;t<vc[j].size();t++)
if(x[vc[j][t]]>=i){
int id=vc[j][t],cur=id;
for(int stp=1;stp<k;stp++) cur=r[cur];
// printf("%d %d\n",id,cur);
if(cur) ret+=1ll*(C-y[cur]+1)*(y[id]-y[l[id]]);
}
// printf("%d %d %lld\n",i,R,ret);
ans+=ret;
for(int j=R;j>=i;j--){
for(int t=0;t<vr[j].size();t++){
int id=vr[j][t];
if(!r[id]){
int cur=id;
for(int stp=1;stp<k;stp++) cur=l[cur];
if(cur) ret-=1ll*(y[cur]-y[l[cur]])*(C-y[id]+1);
if(r[id]) l[r[id]]=l[id];
if(l[id]) r[l[id]]=r[id];
continue;
}
int cr=r[id],cl=r[id],stp=0;
while(stp<k-1){if(!r[cr]) break;cr=r[cr];++stp;}
for(int o=1;o<=k-1-stp;o++) cl=l[cl];
if(cl) ret-=1ll*(C-y[cr]+1)*(y[cl]-y[l[cl]]);
for(int o=1;o<=stp+1;o++){
cr=l[cr];cl=l[cl];
if(cl) ret-=1ll*(C-y[cr]+1)*(y[cl]-y[l[cl]]);
}
if(r[id]) l[r[id]]=l[id];
if(l[id]) r[l[id]]=r[id];
if(!r[id]) continue;
cr=r[id],cl=r[id],stp=0;
while(stp<k-1){if(!r[cr]) break;cr=r[cr];++stp;}
for(int o=1;o<=k-1-stp;o++) cl=l[cl];
if(cl) ret+=1ll*(C-y[cr]+1)*(y[cl]-y[l[cl]]);
for(int o=1;o<=stp;o++){
cr=l[cr];cl=l[cl];
if(cl) ret+=1ll*(C-y[cr]+1)*(y[cl]-y[l[cl]]);
}
} ans+=ret;
// printf("%d %d %lld\n",i,j-1,ret);
}
} printf("%lld\n",ans);
return 0;
}
/*
10 10 10 4
7 9
8 1
5 7
6 8
6 4
5 8
4 3
1 3
5 1
6 9 4 4 3 1
1 3
2 2
3 4 5 5 5 2
1 2
1 5
3 4
4 3
5 2
*/

Codeforces 627E - Orchestra(双向链表,思维题)的更多相关文章

  1. CF--思维练习-- CodeForces - 215C - Crosses(思维题)

    ACM思维题训练集合 There is a board with a grid consisting of n rows and m columns, the rows are numbered fr ...

  2. Codeforces 675C Money Transfers 思维题

    原题:http://codeforces.com/contest/675/problem/C 让我们用数组a保存每个银行的余额,因为所有余额的和加起来一定为0,所以我们能把整个数组a划分为几个区间,每 ...

  3. Codeforces 1090D - Similar Arrays - [思维题][构造题][2018-2019 Russia Open High School Programming Contest Problem D]

    题目链接:https://codeforces.com/contest/1090/problem/D Vasya had an array of n integers, each element of ...

  4. codeforces 1140D(区间dp/思维题)

    D. Minimum Triangulation time limit per test 2 seconds memory limit per test 256 megabytes input sta ...

  5. Codeforces 957 水位标记思维题

    A #include <bits/stdc++.h> #define PI acos(-1.0) #define mem(a,b) memset((a),b,sizeof(a)) #def ...

  6. ACM思维题训练 Section A

    题目地址: 选题为入门的Codeforce div2/div1的C题和D题. 题解: A:CF思维联系–CodeForces -214C (拓扑排序+思维+贪心) B:CF–思维练习-- CodeFo ...

  7. codeforces ~ 1009 B Minimum Ternary String(超级恶心的思维题

    http://codeforces.com/problemset/problem/1009/B B. Minimum Ternary String time limit per test 1 seco ...

  8. 贪心/思维题 Codeforces Round #310 (Div. 2) C. Case of Matryoshkas

    题目传送门 /* 题意:套娃娃,可以套一个单独的娃娃,或者把最后面的娃娃取出,最后使得0-1-2-...-(n-1),问最少要几步 贪心/思维题:娃娃的状态:取出+套上(2),套上(1), 已套上(0 ...

  9. C. Nice Garland Codeforces Round #535 (Div. 3) 思维题

    C. Nice Garland time limit per test 1 second memory limit per test 256 megabytes input standard inpu ...

随机推荐

  1. javascript-jquery介绍

    jquery优势 1.轻量级 2.强大的选择器 3.出色的DOM封装 4.可靠的事件处理机制 5.完善的Ajax 6.不污染顶级变量 7.出色的浏览器兼容 8.链式操作方式 9.隐式迭代 10.行为层 ...

  2. vue基础-组件&插槽

    组件 组件化的意义:封装(复用,把逻辑隐藏起来,提高可维护性),快速开发(搭积木) 约定:我们通常把那些除了HTML标签以外的自定义组件,才称为'组件',结论是,我们说"父组件"& ...

  3. Linux服务器装Anaconda&TensorFlow

    远程Linux服务器装Anaconda&指定版本TensorFlow 说明: 由于疫情影响,原先使用的服务器已断电,故重选了一台服务器对环境重选进行搭建,正好补上这篇博文. 01 下载Anac ...

  4. Beta_Scrum Meeting_0

    日期:2021年5月26日 参会人员:cy.hcc.lsc.dxh 会议主题:为Beta阶段最早两日的开发制定目标 一.进度情况 组员 负责 两日内完成的任务 接下来两日预计完成的任务 hcc 前端 ...

  5. linux下的IO模型---学习笔记

    1.linux文件系统和缓存 文件系统接口 文件系统-一种把数据组织成文件和目录的存储方式,提供了基于文件的存取接口,并通过文件权限控制访问. 存储层次 文件系统缓存 主存(通常时DRAM)的一块区域 ...

  6. c++中virtual 虚函数

    转载: https://www.cnblogs.com/weiyouqing/p/7544988.html 在面向对象的C++语言中,虚函数(virtual function)是一个非常重要的概念. ...

  7. 顺时针打印矩阵 牛客网 剑指Offer

    顺时针打印矩阵 牛客网 剑指Offer 题目描述 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 ...

  8. 【网络好文】---MySQL为Null导致的四大坑

    正式开始之前,我们先来看下 MySQL 服务器的配置和版本号信息,如所示: select version(); -- 版本为 8.0.22 "兵马未动粮草先行",看完了相关的配置之 ...

  9. Oracle ORA 12541 报错解决过程

    Oracle 导入全库之后使用plsql登陆时报错 版本12C版本2 ORA-12541: TNS: No Listener 再oracle主机本地可以使用sqlplus 登陆,但是使用plsql无法 ...

  10. 一文搞懂js中的typeof用法

    基础 typeof 运算符是 javascript 的基础知识点,尽管它存在一定的局限性(见下文),但在前端js的实际编码过程中,仍然是使用比较多的类型判断方式. 因此,掌握该运算符的特点,对于写出好 ...