错了一个小地方调了一晚上。。。。

题目大意:

  给出最多2E+5种不同的矩形,每种有它的长h和宽v还有数量d,现在你要构造大矩形,使得在上面沿着平行于长或宽的边划刀,切出来的矩形正好是给出的所有矩形。问你能构造几种不同的大矩形。其中给出的矩形不能旋转。大矩形亦不能旋转,这意味着长为A,宽为B的大矩形和长为B,宽为A的大矩形不同。

题目分析:

  首先我们令矩形的总数目为tot。那么横着切p刀,竖着且q刀,则(p+1)(q+1)=tot。为了简便,下文的p表示横着切了p-1刀,q表示竖着切了q-1刀。

  以此为突破口,不难证明对于一组(p,q),矩形的大小固定。那么一组(p,q)可以作为大矩形的条件是什么?

  为了得到这个结论,我们设xi表示所有hi相同的矩形的数量,共m种不同的hi。我们不难发现xi是p的倍数。

  理由是这样子的:若xi不是p的倍数,那么在划分后必然在大矩形内有多余的长为hi的小矩形,这样必定非法。所以上面的结论得证!

  由于上面的理由,我们不难发现所有的hi相同的矩形必然集中在一坨,贯穿上下。即如果一列有一个是hi,那么一整列都是hi。

  现在我们解决了长上面的问题,还有宽没解决。

  首先我们发现如果一个v没有对应所有存在的h,那么这个必然无解。理由也很简单,一开始我安排好了p,现在安排q的时候受p的影响。

  同样的我们还能得出两个结论,一是任意的xi/p必然是v的因数,理由和上面相同;二是对于相同的v,它们占的行数相同,这个也很容易想到。

  现在我们通过对h和v的分析,得到了一个比较简单的O(n*sqrt(Σd))的算法,经过计算会发现超时,想办法解决这个问题。

  首先xi是p的倍数意味着p一定是xi的因数,这样p一定是所有xi的因数,也就是说p是所有xi的最大公因数的因数,我们解决了h上的问题。

  考虑v上的第一个结论如何优化,xi/p必然是v的因数。意味着我们可以找到一个si使得gcd(qi,v)=si,接着p一定是qi/si的倍数。所以p是所有qi/si的最小公倍数的倍数。这个可以通过唯一分解定理证明。

  至于v上的第二个结论,首先写出式子,推导后发现这是一个可以预判的式子,预处理的时候直接搞定。

  这样我们得到了一个O(sqrt(Σd))的算法。

代码:

  

 #include<bits/stdc++.h>
using namespace std; #define mp make_pair typedef long long ll;
typedef long double ld; const int maxn = ; int n,m;
struct node{ll h,v,d;}mt[maxn];
vector <pair<ll,ll> > V[maxn];
ll rt[maxn],tot;
ll mx;
ll zzll=; void init(){
for(int i=,now=;i<=n;i++){
if(mt[i].h == mt[now].h){V[m].push_back(mp(mt[i].v,mt[i].d));rt[m]+=mt[i].d;}
else{now = i; m++; V[m].push_back(mp(mt[i].v,mt[i].d)); rt[m]+=mt[i].d;}
}
} void read(){
scanf("%d",&n);
for(int i=;i<=n;i++){
scanf("%lld%lld%lld",&mt[i].h,&mt[i].v,&mt[i].d); tot += mt[i].d;
}
sort(mt+,mt+n+,[](node a,node b){return a.h==b.h?a.v<b.v:a.h<b.h;});
init();
} int NoSolution(){
int hh = V[].size();
for(int i=;i<=m;i++) if(V[i].size()!=hh){return ;}
for(int i=;i<hh;i++){
ll kk = V[][i].second;
for(int j=;j<=m;j++){
ll jj = V[j][i].second;
if(V[j][i].first != V[][i].first){return ;}
if((ld)kk/(ld)rt[] != (ld)jj/(ld)rt[j]){return ;}
}
}
for(int i=;i<=m;i++){
for(int j=;j<V[i].size();j++){
ll mm = __gcd(rt[i],V[i][j].second);
mm = rt[i]/mm;
ll pp = __gcd(zzll,mm);
zzll = (zzll/pp)*mm;
if(zzll > mx){return ;}
}
}
return ;
} ll ans = ;
void solve(ll ht,ll kd){
if(ht % zzll == ){ans ++;}
} void work(){
mx = rt[];
for(int i=;i<=m;i++) mx = __gcd(mx,rt[i]);
if(NoSolution()){puts("");return;}
for(ll i=;i*i<=mx;i++){
if(mx % i != ) continue;
solve(i,tot/i);
if(i*i==mx)break;
solve(mx/i,tot/(mx/i));
}
printf("%lld",ans);
} int main(){
read();
work();
return ;
}

Codeforces963C Cutting Rectangle 【数学】的更多相关文章

  1. *Codeforces963C. Cutting Rectangle

    $n \leq 200000$种互不相同的矩形,给长宽和数量,都$\leq 1e12$,问有多少种大矩形只沿平行长和宽切正好切成这些矩形. 首先可以发现在一个合法情况下,有些矩形的位置是可以乱挪的,比 ...

  2. 【CF963C】Cutting Rectangle(数论,构造,map)

    题意: 思路:考虑构造最小的单位矩形然后平铺 单位矩形中每种矩形的数量可以根据比例算出来,为c[i]/d,其中d是所有c[i]的gcd,如果能构造成功答案即为d的因子个数 考虑如果要将两种矩形放在同一 ...

  3. Tinkoff Internship Warmup Round 2018 and Codeforces Round #475 (Div. 1)

    A. Alternating Sum 就是个等比数列,特判公比为 $1$ 的情况即可. #include <bits/stdc++.h> using namespace std; ; ; ...

  4. LeetCode之“数学”:Rectangle Area

    题目链接 题目要求: Find the total area covered by two rectilinear rectangles in a 2D plane. Each rectangle i ...

  5. UVA 11880 Ball in a Rectangle(数学+平面几何)

    Input: Standard Input Output: Standard Output � There is a rectangle on the cartesian plane, with bo ...

  6. CF1027C Minimum Value Rectangle 贪心 数学

    Minimum Value Rectangle time limit per test 2 seconds memory limit per test 256 megabytes input stan ...

  7. 【CF1027C】Minimum Value Rectangle(贪心,数学)

    题意:给定n根木棍,不允许拼接或折断,选择四根组成矩形,求所有合法矩形中周长平方与面积比最小的一个,输出拼成这个矩形的四根木棍 n<=1e6 思路:猜结论:答案必定从相邻的4根中产生 证明见ht ...

  8. ACM: SCU 4440 Rectangle - 暴力

     SCU 4440 Rectangle Time Limit:0MS     Memory Limit:0KB     64bit IO Format:%lld & %llu  Practic ...

  9. 二维剪板机下料问题(2-D Guillotine Cutting Stock Problem) 的混合整数规划精确求解——数学规划的计算智能特征

    二维剪板机下料问题(2-D Guillotine Cutting Stock Problem) 的混合整数规划精确求解——数学规划的计算智能特征 二维剪板机下料(2D-GCSP) 的混合整数规划是最优 ...

随机推荐

  1. List,DataTable实现行转列的通用方案

    最近在做报表统计方面的需求,涉及到行转列报表.根据以往经验使用SQL可以比较容易完成,这次决定挑战一下直接通过代码方式完成行转列.期间遇到几个问题和用到的新知识这里整理记录一下. 阅读目录 问题介绍 ...

  2. Django的model form组件

    前言 首先对于form组件通过全面的博客介绍,对于form我们应该知道了它的大致用法,这里我们需要明确的一点是,我们定义的form与model其实没有什么关系,只是在逻辑上定义form的时候字段名期的 ...

  3. Python_内置函数之map()

    map 会根据提供的函数对指定序列做映射. 代码如下: def square(x): return x ** 2 ret = map(square, [1, 2, 3, 4, 5]) # 计算列表各元 ...

  4. Python_生产者消费者模型、管道、数据共享、进程池

    1.生产者消费者模型 生产者 —— 生产数据的人 消费者 —— 消费数据的人 生产者消费者模型:供销数据不平衡的现象. import time import random from multiproc ...

  5. 结对项目3-功能增强型带基本函数计算java计算器

    -----------------------------------------------------实验报告------------------------------------------- ...

  6. Jmeter操作之跨线程组传递参数

    思路:将某一线程组内的变量通过“__setProperty”函数设置成jmeter的全局变量,在另一线程组中通过“__P”函数调用即可. 1.添加-后置处理器-BeanShell PostProces ...

  7. React不同版本之间需要注意的地方

    React Fiber react fiber指的是react16.0机器之后的版本,相对于之前的版本来说,这一个版本的更新做了很多的优化,所以和之前的版本中的用法可能会发生不同,所以,平常开发中,主 ...

  8. IdentityServer4【QuickStart】之切换到混合流并且添加API访问

    切换到混合流并且添加API访问 前面的示例中我们开发了API访问和用户认证,现在我们要将两个合并到一起. OpenID Connect&OAuth 2.0组合的美妙之处是,你可以使用单一协议和 ...

  9. PHP中对象的深拷贝与浅拷贝

    先说一下深拷贝和浅拷贝通俗理解 深拷贝:赋值时值完全复制,完全的copy,对其中一个作出改变,不会影响另一个 浅拷贝:赋值时,引用赋值,相当于取了一个别名.对其中一个修改,会影响另一个 PHP中, = ...

  10. [转帖]Linux下fork函数及pthread函数的总结

    Linux下fork函数及pthread函数的总结 https://blog.csdn.net/wangdd_199326/article/details/76180514 fork Linux多进程 ...