HDU 4456 Crowd (cdq分治)

2024-08-24 11:58
文章标签 分治 hdu crowd cdq 4456

本文主要是介绍HDU 4456 Crowd (cdq分治),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

大意就是给出一个矩阵

初始每个位置上的值都为0

然后有两种操作

一种是更改某个位置上的值

另一个是求某个位置附近曼哈顿距离不大于K的所有位置的值的总和


网络上众多题解都是  二维树状数组

但是这题也被认为是cdq分治的基础题

下面提供两种基于cdq分治的解法



解法一:

将所有点绕原点左旋45°

然后新的坐标也很好计算

x' = (x - y) * sqrt(2) / 2

y' = (x + y) * sqrt(2) / 2

由于都是小数

所以乘个sqrt(2) 就成整数了

x' = (x - y) 

y' = x + y

由于x- y可能是负数。所以把点都右移一下  x' = x + y + n (n是矩阵宽度)然后矩阵的宽度和长度也就各自扩大了一倍

然后我们就可以惊奇的发现

原先是求 abs(x - x0) + abs(y - y0) <= k 的所有位置的值的和

现在变成了 abs(x' - x0') <= k 或者abs(y' - y0') <= k 就可以了

也就变成了求一个子矩阵的和

然后用CDQ分治维护一个普通矩阵的和,类似于经典题bzoj 2683

<span style="font-size:18px;"><span style="font-size:18px;">//      whn6325689
//		Mr.Phoebe
//		http://blog.csdn.net/u013007900
#include <algorithm>
#include <iostream>
#include <iomanip>
#include <cstring>
#include <climits>
#include <complex>
#include <fstream>
#include <cassert>
#include <cstdio>
#include <bitset>
#include <vector>
#include <deque>
#include <queue>
#include <stack>
#include <ctime>
#include <set>
#include <map>
#include <cmath>
#include <functional>
#include <numeric>
#pragma comment(linker, "/STACK:1024000000,1024000000")using namespace std;typedef long long ll;
typedef long double ld;
typedef pair<ll, ll> pll;
typedef complex<ld> point;
typedef pair<int, int> pii;
typedef pair<pii, int> piii;
typedef vector<int> vi;#define CLR(x,y) memset(x,y,sizeof(x))
#define mp(x,y) make_pair(x,y)
#define pb(x) push_back(x)
#define lowbit(x) (x&(-x))
#define MID(x,y) (x+((y-x)>>1))
#define eps 1e-9
#define PI acos(-1.0)
#define INF 0x3f3f3f3f
#define LLINF 1LL<<62template<class T>
inline bool read(T &n)
{T x = 0, tmp = 1; char c = getchar();while((c < '0' || c > '9') && c != '-' && c != EOF) c = getchar();if(c == EOF) return false;if(c == '-') c = getchar(), tmp = -1;while(c >= '0' && c <= '9') x *= 10, x += (c - '0'),c = getchar();n = x*tmp;return true;
}
template <class T>
inline void write(T n)
{if(n < 0){putchar('-');n = -n;}int len = 0,data[20];while(n){data[len++] = n%10;n /= 10;}if(!len) data[len++] = 0;while(len--) putchar(data[len]+48);
}
//-----------------------------------const int MAXN=80010;struct Node
{int kind;int x,y,z,idx;Node(int k=0,int xx=0,int yy=0,int zz=0,int idx=0):kind(k),x(xx),y(yy),z(zz),idx(idx){}
}ask[4*MAXN],tmp1[4*MAXN],tmp2[4*MAXN];
int n,m,tot;
int c[MAXN];
int ans[MAXN];bool cmp(Node a,Node b)
{if(a.x==b.x)return a.y<b.y;return a.x<b.x;
}void update(int i,int v)
{for(;i<MAXN;i+=lowbit(i))c[i]+=v;
}int getsum(int i)
{int sum=0;for(;i;i-=lowbit(i))sum+=c[i];return sum;
}void clr(int i)
{for(;i<MAXN;i+=lowbit(i))c[i]=0;
}void change(int x,int y,int &xx,int &yy)
{xx=x-y+n;yy=x+y;
}void cdq(int l,int r)
{if(l==r)	return;int mid=MID(l,r);int l1=0,l2=0;cdq(l,mid);for(int i=l;i<=r;i++){if(i<=mid && ask[i].kind==1) tmp1[l1++]=ask[i];if(i> mid && ask[i].kind==2) tmp2[l2++]=ask[i];}sort(tmp1,tmp1+l1,cmp);sort(tmp2,tmp2+l2,cmp);for(int i=0,j=0;i<l2;i++){while(j<l1 && tmp1[j].x <= tmp2[i].x){update(tmp1[j].y,tmp1[j].z);j++;}ans[tmp2[i].idx]+=getsum(tmp2[i].y)*tmp2[i].z;}for(int i=0;i<l1;i++)clr(tmp1[i].y);cdq(mid+1,r);
}int main()
{
//    freopen("data.txt","r",stdin);
//    freopen("wa.txt","w",stdout);while(read(n)&&n){read(m);CLR(c,0);CLR(ans,0);CLR(ask,0);tot=0;for(int i=0,x,y,xx,yy,k,z;i<m;i++){read(k),read(x),read(y),read(z);change(x,y,xx,yy);if(k==1){tot++;ask[tot].kind=1;ask[tot].x=xx;ask[tot].y=yy;ask[tot].z=z;ask[tot].idx=i;}else{int x1=max(0,xx-z-1);int y1=max(0,yy-z-1);int x2=min(2*n,xx+z);int y2=min(2*n,yy+z);tot++;ask[tot].kind=2;ask[tot].x=x1;ask[tot].y=y1;ask[tot].z=1; ask[tot].idx=i;tot++;ask[tot].kind=2;ask[tot].x=x2;ask[tot].y=y1;ask[tot].z=-1;ask[tot].idx=i;tot++;ask[tot].kind=2;ask[tot].x=x1;ask[tot].y=y2;ask[tot].z=-1;ask[tot].idx=i;tot++;ask[tot].kind=2;ask[tot].x=x2;ask[tot].y=y2;ask[tot].z=1; ask[tot].idx=i;}}cdq(1,tot);for(int i=1;i<=tot;i++)if(ask[i].kind==2){printf("%d\n",ans[ask[i].idx]);i+=3;}}return 0;
}
</span></span>


解法二:

同样是子矩阵

但是可以用斜线法解答

省略了旋转的部分

但是cdq分治的部分略显麻烦

<span style="font-size:18px;"><pre name="code" class="cpp">//      whn6325689
//		Mr.Phoebe
//		http://blog.csdn.net/u013007900
#include <algorithm>
#include <iostream>
#include <iomanip>
#include <cstring>
#include <climits>
#include <complex>
#include <fstream>
#include <cassert>
#include <cstdio>
#include <bitset>
#include <vector>
#include <deque>
#include <queue>
#include <stack>
#include <ctime>
#include <set>
#include <map>
#include <cmath>
#include <functional>
#include <numeric>
#pragma comment(linker, "/STACK:1024000000,1024000000")using namespace std;typedef long long ll;
typedef long double ld;
typedef pair<ll, ll> pll;
typedef complex<ld> point;
typedef pair<int, int> pii;
typedef pair<pii, int> piii;
typedef vector<int> vi;#define CLR(x,y) memset(x,y,sizeof(x))
#define mp(x,y) make_pair(x,y)
#define pb(x) push_back(x)
#define lowbit(x) (x&(-x))
#define MID(x,y) (x+((y-x)>>1))
#define eps 1e-9
#define PI acos(-1.0)
#define INF 0x3f3f3f3f
#define LLINF 1LL<<62template<class T>
inline bool read(T &n)
{T x = 0, tmp = 1; char c = getchar();while((c < '0' || c > '9') && c != '-' && c != EOF) c = getchar();if(c == EOF) return false;if(c == '-') c = getchar(), tmp = -1;while(c >= '0' && c <= '9') x *= 10, x += (c - '0'),c = getchar();n = x*tmp;return true;
}
template <class T>
inline void write(T n)
{if(n < 0){putchar('-');n = -n;}int len = 0,data[20];while(n){data[len++] = n%10;n /= 10;}if(!len) data[len++] = 0;while(len--) putchar(data[len]+48);
}
//-----------------------------------const int MAXN=80010;
const int POS=10000;struct Node
{int kind,idx;int x,y,z;
}ask[MAXN],tmp1[MAXN],tmp2[MAXN];
int n,m;
ll ans[MAXN],c[MAXN];bool cmp1(Node a,Node b)
{return a.x+a.y<=b.x+b.y;
}bool cmp2(Node a,Node b)
{return a.x+a.y+a.z<=b.x+b.y+b.z;
}bool cmp3(Node a,Node b)
{return a.x+a.y-a.z<=b.x+b.y-b.z;
}void update(int i,int v)
{for(;i<MAXN;i+=lowbit(i))c[i]+=v;
}ll getsum(int i)
{ll sum=0;for(;i>0;i-=lowbit(i))sum+=c[i];return sum;
}void clr(int i)
{for(;i<MAXN;i+=lowbit(i))c[i]=0;
}void cdq(int l,int r)
{if(l==r)	return;int mid=MID(l,r);int l1=0,l2=0;cdq(l,mid);for(int i=l;i<=r;i++){if(i<=mid && ask[i].kind==1) tmp1[l1++]=ask[i];else if(i> mid && ask[i].kind==2) tmp2[l2++]=ask[i];}sort(tmp1,tmp1+l1,cmp1);sort(tmp2,tmp2+l2,cmp2);for(int i=0,j=0;i<l2;i++){int lim=tmp2[i].x+tmp2[i].y+tmp2[i].z;while(j<l1 && tmp1[j].x+tmp1[j].y<=lim){update(tmp1[j].x-tmp1[j].y+POS,tmp1[j].z);j++;}ans[tmp2[i].idx]+=getsum(tmp2[i].x-tmp2[i].y+tmp2[i].z+POS)-getsum(tmp2[i].x-tmp2[i].y-tmp2[i].z-1+POS);}for(int i=0;i<l1;i++)clr(tmp1[i].x-tmp1[i].y+POS);sort(tmp2,tmp2+l2,cmp3);for(int i=0,j=0;i<l2;i++){int lim=tmp2[i].x+tmp2[i].y-tmp2[i].z-1;while(j<l1 && tmp1[j].x+tmp1[j].y<=lim){update(tmp1[j].x-tmp1[j].y+POS,tmp1[j].z);j++;}ans[tmp2[i].idx]-=getsum(tmp2[i].x-tmp2[i].y+tmp2[i].z+POS)-getsum(tmp2[i].x-tmp2[i].y-tmp2[i].z-1+POS);}for(int i=0;i<l1;i++)clr(tmp1[i].x-tmp1[i].y+POS);cdq(mid+1,r);
}int main()
{while(read(n)&&n){read(m);CLR(ans,0);CLR(c,0);for(int i=1;i<=m;i++){read(ask[i].kind),read(ask[i].x),read(ask[i].y),read(ask[i].z);ask[i].idx=i;}cdq(1,m);for(int i=1;i<=m;i++)if(ask[i].kind==2)printf("%lld\n",ans[i]);}return 0;
}
</span>

 

推荐 cdq分治基础题目

HDU 4742

HDU 5126

UVALIVE 5871

S菊苣训练之cdq和kd树

这篇关于HDU 4456 Crowd (cdq分治)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1102417

相关文章

usaco 1.3 Mixing Milk (结构体排序 qsort) and hdu 2020(sort)

到了这题学会了结构体排序 于是回去修改了 1.2 milking cows 的算法~ 结构体排序核心: 1.结构体定义 struct Milk{int price;int milks;}milk[5000]; 2.自定义的比较函数,若返回值为正,qsort 函数判定a>b ;为负,a<b;为0,a==b; int milkcmp(const void *va,c

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

hdu 2093 考试排名(sscanf)

模拟题。 直接从教程里拉解析。 因为表格里的数据格式不统一。有时候有"()",有时候又没有。而它也不会给我们提示。 这种情况下,就只能它它们统一看作字符串来处理了。现在就请出我们的主角sscanf()! sscanf 语法: #include int sscanf( const char *buffer, const char *format, ... ); 函数sscanf()和

hdu 2602 and poj 3624(01背包)

01背包的模板题。 hdu2602代码: #include<stdio.h>#include<string.h>const int MaxN = 1001;int max(int a, int b){return a > b ? a : b;}int w[MaxN];int v[MaxN];int dp[MaxN];int main(){int T;int N, V;s

hdu 1754 I Hate It(线段树,单点更新,区间最值)

题意是求一个线段中的最大数。 线段树的模板题,试用了一下交大的模板。效率有点略低。 代码: #include <stdio.h>#include <string.h>#define TREE_SIZE (1 << (20))//const int TREE_SIZE = 200000 + 10;int max(int a, int b){return a > b ? a :

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

hdu 3790 (单源最短路dijkstra)

题意: 每条边都有长度d 和花费p,给你起点s 终点t,要求输出起点到终点的最短距离及其花费,如果最短距离有多条路线,则输出花费最少的。 解析: 考察对dijkstra的理解。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstrin

hdu 2489 (dfs枚举 + prim)

题意: 对于一棵顶点和边都有权值的树,使用下面的等式来计算Ratio 给定一个n 个顶点的完全图及它所有顶点和边的权值,找到一个该图含有m 个顶点的子图,并且让这个子图的Ratio 值在所有m 个顶点的树中最小。 解析: 因为数据量不大,先用dfs枚举搭配出m个子节点,算出点和,然后套个prim算出边和,每次比较大小即可。 dfs没有写好,A的老泪纵横。 错在把index在d

hdu 1102 uva 10397(最小生成树prim)

hdu 1102: 题意: 给一个邻接矩阵,给一些村庄间已经修的路,问最小生成树。 解析: 把已经修的路的权值改为0,套个prim()。 注意prim 最外层循坏为n-1。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstri

hdu 1285(拓扑排序)

题意: 给各个队间的胜负关系,让排名次,名词相同按从小到大排。 解析: 拓扑排序是应用于有向无回路图(Direct Acyclic Graph,简称DAG)上的一种排序方式,对一个有向无回路图进行拓扑排序后,所有的顶点形成一个序列,对所有边(u,v),满足u 在v 的前面。该序列说明了顶点表示的事件或状态发生的整体顺序。比较经典的是在工程活动上,某些工程完成后,另一些工程才能继续,此时