洛谷 P1548 棋盘问题

本文介绍了一种算法,用于计算任意尺寸的棋盘中包含的所有正方形和非正方形长方形的数量,并提供了详细的推导过程及C++实现代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

题目描述

设有一个N*M方格的棋盘(l<=N<=100,1<=M<=100)(30%)

求出该棋盘中包含有多少个正方形、多少个长方形(不包括正方形)。

例如:当 N=2, M=3时: 

正方形的个数有8个:即边长为1的正方形有6个;

边长为2的正方形有2个。

长方形的个数有10个:

即2*1的长方形有4个:

          1*2的长方形有3个:

          3*1的长方形有2个:

          3*2的长方形有1个:

如上例:输入:2 3

输出:8 10

输入输出格式

输入格式:

 

N和M

 

输出格式:

 

正方形的个数与长方形的个数

 

输入输出样例

输入样例#1:
2 3
输出样例#1:
8 10


 1 说说公式是怎么推导的吧
 2 
 3 找规律:
 4 
 5 正方形:
 6 
 7 边长为1的正方形个数为n*m
 8 
 9 边长为2的正方形个数为(n-1)*(m-1) (自己动手想想)
10 
11 边长为3的正方形为个数(n-2)*(m-2)
12 
13 边长为min(n,m)的正方形为个数s1=(n-min(n,m)+1)*(m-min(n,m)+1)
14 
15 然后从边长为1到min(m,m)的正方形个数全部加起来;
16 
17 长方形:(包括正方形,好像正方形属于长方形来着?)
18 
19 长为1的长方形(包括正方形)有n个
20 
21 长为2的长方形(包括正方形)有n-1个
22 
23 长为n的长方形(包括正方形)有1个
24 
25 长为1到n的长方形1+2+...+n个
26 
27 同理 宽为1的长方形(包括正方形)有m个
28 
29 宽为2的长方形(包括正方形)有m-1个
30 
31 宽为m的长方形(包括正方形)有1个
32 
33 宽为1-m的长方形1+2+...+m个
34 
35 然后把它们乘起来,根据乘法原理,总数s2=((1+n)*(1+m)*n*m)/4;
36 
37 题目要求的是“非正方形的长方形”,因此要减去s1;

 

#include<cstdio>
#include<iostream>
using namespace std;
int n,m,ans1,ans2;
inline void read(int&x) {
    int f=1;x=0;char c=getchar();
    while(c>'9'||c<'0') {if(c=='-') f=-1;c=getchar();}
    while(c>='0'&&c<='9') {x=(x<<1)+(x<<3)+c-48;c=getchar();}
    x=x*f;
}
int main() {
    read(n);read(m);
    int i=n,j=m;
    for(;i>=1&&j>=1;i--,j--) 
      ans1+=i*j;
    int t1=0,t2=0;
    for(int k=1;k<=n;k++) t1+=k;
    for(int k=1;k<=m;k++) t2+=k;
    ans2=t1*t2;
    printf("%d %d\n",ans1,ans2-ans1);
    return 0;
}
代码

 

### 洛谷 P1002 题目解析 洛谷 P1002 是一道经典的递推问题,其核心在于通过动态规划的思来解决问题。以下是关于该题目的详细分析。 #### 题目概述 题目要求计算棋盘一个卒从起点到终点的不路径数量[^2]。具体来说,给定一个 $N \times M$ 的棋盘,某些格子上有障碍物,求出从左下角走到右上角的合法路径数目。需要注意的是,卒只能向上或者向右移动。 #### 思路解析 此问题可以通过 **递推** 方法高效解决。设 $f[i][j]$ 表示到达位置 $(i, j)$ 的不路径数,则状态转移方程可以表示为: $$ f[i][j] = f[i-1][j] + f[i][j-1] $$ 其中,如果当前格子有障碍物,则 $f[i][j]=0$ 。初始条件通常设定为起始点的路径数为 1 即可,即 $f[0][0] = 1$ (如果没有障碍的情况下)[^3]。 下面给出基于 C++ 实现的一个标准解决方案: ```cpp #include <bits/stdc++.h> using namespace std; int main() { int n, m; cin >> n >> m; // 输入棋盘大小n*m vector<string> grid(n); for(int i=0;i<n;i++) cin>>grid[i]; // 输入棋盘布局 vector<vector<long long>> dp(n, vector<long long>(m, 0)); bool flag=true; for(int i=0;i<n && flag;i++){ if(grid[i][0]=='*')flag=false; else dp[i][0]=1; } flag=true; for(int j=0;j<m && flag;j++){ if(grid[0][j]=='*')flag=false; else dp[0][j]=1; } for(int i=1;i<n;i++) for(int j=1;j<m;j++) if(grid[i][j]=='.') dp[i][j]=(dp[i-1][j]+dp[i][j-1])%(long long)(1e9+7); cout << dp[n-1][m-1]; } ``` 上述代码实现了二维数组 `dp` 来存储每一个可能的位置上的路径计数值,并考虑了边界情况和障碍物的影响。 #### 推荐技巧 为了更有效地使用洛谷平台进行编程学习和竞赛训练,建议多尝试不的算法实现方式并对比效率差异。此外,在遇到困难时,积极查阅官方或其他用户的高质量题解可以帮助快速提升理解能力[^1]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值