给定一个n*m的网格,求面积为奇数的正方形有多少个.

首先是n*m个面积为1的,然后剩下的要么是边长为奇数,要么被这样一个奇数边长所包围。

原因如下:

对于一个边长不平行于坐标抽的正方形,其边长一定是某个长方形的对角线,而且长方形长宽a,b一定是一奇数,一偶数,这样area = a^2+b^2才是奇数。

所以可以对任何奇数i <= min(n, m) 求出这样的边长正方形以及被包围的正方形个数。

注意对于一个奇数例如5,被包围的正方形可以是以1和4的对角线,2和3的对角线为边,这样对任何一个奇数i,被包围的正方形有i/2个,根据对称性还应该*2。

w-i+1表示宽方向能够移动的次数,l-i+1表示长度方向能够移动的次数,例如图5,长和宽方向均能移动2次。

ans = n*m(单位正方形) +

{2*(w-i+1)*(l-i+1)*(i/2) + (w-i+1)*(l-i+1)}    i为 3<=i<=min(n, m)的奇数。

后来在最后完成这篇博客的时候想到能不能把式子简化一下呢,然后化简就得到这个:ans = ∑{ (w*l+w+l+1)*i – (w+l+2)*i*i + i*i*i} i为1到min(l,w)之间的奇数。

甚至可以推广到面积为偶数的情况:

公式:ans = ∑{ (w*l+w+l+1)*i – (w+l+2)*i*i + i*i*i} i为1到min(l,w)之间的偶数。

如果是求所有的正方形个数的话,只要把i从1取到min(w,l)就行了。

注意程序中预处理一下i,i*i,i*i*i的和。

下面只上化简后公式实现的代码好了。

time:29ms

//Date: 20140211
#include <iostream>
#include <sstream>
#include <cstdio>
#include <climits>
#include <ctime>
#include <cstring>
#include <cstdlib>
#include <string>
#include <stack>
#include <map>
#include <cmath>
#include <vector>
#include <queue>
#include <algorithm>
#define esp 1e-3
#define pi acos(-1.0)
#define inf 0x0f0f0f0f
#define pb push_back
#define lson l, m, rt<<1
#define rson m+1, r, rt<<1|1
#define mp(a, b) make_pair((a), (b))
#define in freopen("test_in.txt", "r", stdin);
#define out freopen("test_out.txt", "w", stdout);
#define bug puts("********))))))");
#define inout in out
#define stop system("pause");
#define PRD(a) printf("%d\n",(a))
#define PRLD(a) printf("%lld\n", (a))
#define PRID(a) printf("%I64d\n", (a))
#define PRU(a) printf("%u\n", (a))
#define PRLU(a) printf("%llu\n", (a))
#define PRIU(a) printf("%I64u\n", (a))
#define SET(a, v) memset(a, (v), sizeof(a))
#define READ(a, n) {REP(i, n) cin>>a[i];}
#define REP(i, n) for(int i = 0; i < (n); i++)
#define Rep(i, base, n) for(int i = base; i < n; i++)
#define REPS(s) for(int i = 0; s[i]; i++)
#define pf(x) ((x)*(x))
#define Log(a, b) (log((double)b)/log((double)a))
#define Srand() srand((int)time(0))
#define random(number) (rand()%number)
#define random_range(a, b) (int)(((double)rand()/RAND_MAX)*(b-a) + a) /*
1. 点分治,atan和atan2返回弧度值,atan值域为(-pi/2, pi/2), atan2值域为(-pi, pi) ;
2. exp(x)用于计算e^x ;
3. log2和log10分别用来计算对数, log默认以e为底 ;
4. sort中比较函数的原型:bool cmp(const Type &a, const Type &b);
5. lower_bound和upper_bound返回的是相应下标对应的指针 ;
6. dfs时注意利用强剪枝和避免重复状态进行优化 ;
7. 尽量减少不必要的状态表示的维度 ;
8. greater<T> () less<T> () ;
9. 尽量少用strlen,尤其是在递归深度较大,字符串较长的时候,容易超时;少用memset ;
10.不要在函数里面开数组,易暴栈 ;
*/ using namespace std;
typedef long long LL;
typedef unsigned long long ULL;
typedef vector<int> VI;
typedef pair<int,int> pii;
typedef vector<pii> VII;
typedef vector<pii, int> VIII;
typedef VI:: iterator IT;
typedef map<string, int> Mps;
typedef map<int, int> Mpi;
typedef map<int, pii> Mpii;
typedef map<pii, int> Mpiii; template<class T> inline const T& Max(const T& a, const T& b)
{
return a < b ? b : a;
}
template<class T> inline const T& Min(const T& a, const T& b)
{
return a < b ? a : b;
}
template<class T> inline void checkMax(T& a, const T& b)
{
if(a < b) a = b;
}
template<class T> inline void checkMin(T& a, const T& b)
{
if(a > b) a = b;
} const int maxn = + ;
LL f[maxn], g[maxn], h[maxn];
void pre()
{
f[] = g[] = h[] = ;
for(int i = ; i < maxn; i += ) {
f[i] = f[i-] + i;
g[i] = g[i-] + (LL)i*i;
h[i] = h[i-] + (LL)i*i*i;
}
}
int main()
{ LL n, m;
pre();
while(scanf("%lld%lld", &n, &m), n||m) {
LL k = min(n, m);
if((k&) == )
k--;
LL ans = (n*m+n+m+)*f[k]-(n+m+)*g[k]+h[k];
printf("%lld\n", ans);
}
return ;
}

UVALive 6602 Counting Lattice Squares的更多相关文章

  1. UVaLive 6602 Counting Lattice Squares (找规律)

    题意:给定一个n*m的矩阵,问你里面有几面积为奇数的正方形. 析:首先能知道的是,大的矩阵是包括小的矩阵的,而且面积为奇数,我们只要考虑恰好在边界上的正方形即可,画几个看看就知道了,如果是3*3的有3 ...

  2. UVALive 5058 Counting BST 数学

    B - Counting BST Time Limit:3000MS     Memory Limit:0KB     64bit IO Format:%lld & %llu Submit S ...

  3. UVALive 5058 Counting BST --组合数

    题意:排序二叉树按照数插入的顺序不同会出现不同的结构,现在要在1~m选n个数,使按顺序插入形成的结构与给出的结构相同,有多少种选法. 解法:先将给出的结构插入,构造出一棵排序二叉树,再dfs统计,首先 ...

  4. UVALive 3295 Counting Triangles

    题目链接:https://icpcarchive.ecs.baylor.edu/index.php?option=com_onlinejudge&Itemid=8&page=show_ ...

  5. UVALIVE 3571 Visible Lattice Points

    就欧拉函数然后地推一下. #include <map> #include <set> #include <list> #include <cmath> ...

  6. UVALive 6527 Counting ones dfs(水

    题目链接:点击打开链接 #include <cstdio> #include <vector> using namespace std; typedef long long l ...

  7. Individual Contest #1 and Private Training #1

    第一次的增补赛,也是第一场个人排位赛,讲道理打的和屎一样,手速题卡了好久还WA了好多发,难题又切不出来,这种情况是最尴尬的吧! Individual Contest #1: Ploblem D: 题意 ...

  8. Rocky(dfs)

    题目描述 Sylvester Stallion is an old horse who likes nothing better than to wander around in the fields ...

  9. The 2013 South America/Brazil Regional Contest 题解

    A: UVALive 6525 cid=61196#problem/A" style="color:blue; text-decoration:none">Atta ...

随机推荐

  1. 异步请求---Get

    前端 <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> &l ...

  2. mysql安全

    安装MySql时,尽量选择别的端口(默认是3306),密码设复杂一点!在next的步骤中,注意不要选中"允许远程登录". Web漏洞检测及修复 http://wiki.open.q ...

  3. C++里的int 和string类型相互转换

    C++不像Java和C#一样在进行数据类型转换时直接调用一些类方法就可以了,使用起来很简单. 一个很简单的例子就是string str=“D:\\”+1+“.txt”;这在Java或者C#里面是可以自 ...

  4. Trie的C++实现及HDU1251,hdu1671

    #include<iostream> #include<cstdio> #include<string> #include<cstring> #incl ...

  5. HTML5之图像处理

    --- 内嵌图像 - drawImage可以绘制图像context.drawImage(image,dx,dy)context.drawImage(image,dx,dy,dw,dh)context. ...

  6. socket通信_笔记

    (socket通信) 客户端与服务器端通信问题: 我们首先要了解一个概念性的词汇:Socket socket的英文原义是“孔”或“插座”.作为进程通信机制,取后一种意思.通常也称作“套接字”,用于描述 ...

  7. Linux+svn无法显示日志

    自己在linux中配置了一个svn服务器,但是客户端在浏览日志的时候报错.提示不能连接到服务器,是否离线查看... 谷歌了一下,是因为svnserve.conf中配置了anon-access = re ...

  8. C++字符串函数与C字符串函数比较

    赋值拷贝: #include <iostream> #include <string> using namespace std; void main(){ string a=& ...

  9. php学习日志(1)-php介绍

    在学习Php之前,我们要搞懂php是什么.英文全称php: php hypertext preprocessor,即php超文本预处理器.php脚本在服务器上执行,故它是一种服务器编程语言. php文 ...

  10. can't add foreign key in mysql?

    create table department (dept_name ), building ), budget numeric(,) ), primary key (dept_name) ); cr ...