UVA 1393 Highways,UVA 12075 Counting Triangles —— (组合数,dp)
先看第一题,有n*m个点,求在这些点中,有多少条直线,经过了至少两点,且不是水平的也不是竖直的。
分析:由于对称性,我们只要求一个方向的线即可。该题分成两个过程,第一个过程是求出n*m的矩形中,dp[i][j]代表在这个矩形中终点是到(i,j)这个点的满足题意的直线条数,那么,用dp的话就可以得出递推关系:由长和宽分别小1的左右两个矩形中满足题意的线的条数减去他们共有的矩形中满足的线的条数(容斥减去重复部分),之后还要判断从最左上角的点(1,1)到(i,j)是否可以组成一条线,这个条件是gcd(i,j)是否等于1。
之后第二个过程就是递推答案了,设ans[i][j]表示在这个矩形中满足题意的条数,那么同样的,可以由上面的容斥来递推,同时,还要加上这个矩形内到(i,j)这个点满足的条数,另外还要减去一半规模大小的到这个点的线的条数,因为如果(i,j)为(6,8),那么一半规模下,(1,1)到(3,4)这个点的线和到(6,8)这条线是重复的。
这样就做完了题目(最后不要忘了乘以2)。具体见代码:
#include <stdio.h>
#include <algorithm>
#include <string.h>
#include <iostream>
#include <vector>
#include <queue>
using namespace std;
typedef long long ll;
const int N = +; int dp[N][N];
int ans[N][N]; int gcd(int a,int b) {return a%b?gcd(b,a%b):b;} void init()
{
for(int i=;i<N;i++)
{
for(int j=;j<N;j++)
{
dp[i][j] = dp[i-][j] + dp[i][j-] - dp[i-][j-] + (gcd(i,j)==);
}
}
for(int i=;i<N;i++)
{
for(int j=;j<N;j++)
{
ans[i][j] = ans[i-][j] + ans[i][j-] - ans[i-][j-] + dp[i][j] - dp[i>>][j>>];
}
}
} int main()
{
init();
int n,m;
while(scanf("%d%d",&n,&m)==)
{
if(n== && m==) break;
printf("%d\n",*ans[n-][m-]);
}
}
做出了这题,第二题就是类似的了。先在所有的点中枚举出选3个点的可能性,然后,减去一条水平或者竖直线上重复的,再减去同在一条斜线上重复的即可。相当类似,具体见代码吧:
#include <stdio.h>
#include <algorithm>
#include <string.h>
#include <iostream>
#include <vector>
#include <queue>
using namespace std;
typedef long long ll;
const int N = +; ll dp[N][N];
ll ans[N][N]; int gcd(int a,int b) {return a%b?gcd(b,a%b):b;} void init()
{
for(int i=;i<N;i++)
{
for(int j=;j<N;j++)
{
dp[i][j] = dp[i-][j] + dp[i][j-] - dp[i-][j-] + (ll)(gcd(i,j)-);
}
}
for(int i=;i<N;i++)
{
for(int j=;j<N;j++)
{
ans[i][j] = ans[i-][j] + ans[i][j-] - ans[i-][j-] + dp[i][j];
}
}
} ll C(ll x) {return x*(x-)*(x-)/;} int main()
{
init();
int n,m;
int cnt = ;
while(scanf("%d%d",&n,&m)==)
{
if(n== && m==) break;
ll Ans = C((n+)*(m+)) - (n+)*C(m+) - (m+)*C(n+);
Ans -= *ans[n][m];
cout<<"Case "<<cnt++<<": "<<Ans<<endl;
}
}
但是,这两题都要注意的地方是,递推dp时三个矩形都是在右下角的(因为向下或者向右平移一个单位的话条数是不变的),这样递推起来的话只要再考虑从(1,1)这个点到(i,j)这个点的情况即可;而递推ans的时候,矩形是偏左上方的,那么,只要再加上整个大矩形内到(i,j)这个点的情况即可。当然,纯属个人理解。
UVA 1393 Highways,UVA 12075 Counting Triangles —— (组合数,dp)的更多相关文章
- UVA 12075 - Counting Triangles(容斥原理计数)
题目链接:12075 - Counting Triangles 题意:求n * m矩形内,最多能组成几个三角形 这题和UVA 1393类似,把总情况扣去三点共线情况,那么问题转化为求三点共线的情况,对 ...
- uva 1393 - Highways(容斥原理)
题目连接:uva 1393 - Highways 题目大意:给定一个m∗n的矩阵,将矩阵上的点两两相连,问有多少条直线至少经过两点. 解题思路:头一次做这样的题目,卡了一晚上. dp[i][j]即为i ...
- UVA 12075 Counting Triangles
https://icpcarchive.ecs.baylor.edu/index.php?option=com_onlinejudge&Itemid=8&page=show_probl ...
- UVa 1393 - Highways(数论)
链接: https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...
- UVA 1393 Highways(数学思想)
题意:给你n.m(n,m<=200),问你有多少条非水平.非垂直的直线有多少条经过至少两个点 题解:我们需要枚举的是只画一条线的矩形,对于大小a*b的矩形必须保证gcd(a,b)=1才能不重复 ...
- UVA 1393 Highways
https://vjudge.net/problem/UVA-1393 题意: a*b的点阵中能画多少条非水平非竖直的直线 方向‘/’ 和 方向 ‘\’ 对称 枚举直线所在矩形的i*j 直线可能重复的 ...
- UVa 1393 (容斥原理、GCD) Highways
题意: 给出一个n行m列的点阵,求共有多少条非水平非竖直线至少经过其中两点. 分析: 首先说紫书上的思路,编程较简单且容易理解.由于对称性,所以只统计“\”这种线型的,最后乘2即是答案. 枚举斜线包围 ...
- hdu 1396 Counting Triangles(递推)
Counting Triangles Problem Description Given an equilateral triangle with n thelength of its side, p ...
- Counting Triangles(hd1396)
Counting Triangles Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Other ...
随机推荐
- python 比对PDF文件
基本思路: 1.读取pdf内容,存放到不同的 list 2.比较 list 的相似度 ------------------------ 实现------------------------- 1.PD ...
- js实现div转图片并保存
最近工作中遇到的需求,将div转成图片并保存. 1.准备需要用到的js插件jquery-1.8.2.js,html2canvas.min.js(将div转换为canvas),bluebird.js(用 ...
- 设计模式风格<一>;回调风格
主程序,是一个人,有一个类是同事: static void Main(string[] args) { Console.WriteLine("Hello Go to Lunch?" ...
- 【opencv 源码剖析】 四、 Mat的赋值构造函数 和 拷贝构造函数
1.赋值构造函数 右值引用 inline Mat& Mat::operator = (Mat&& m) { if (this == &m) return *this; ...
- JavaScript 的编译原理
JavaScript 是一门编译语言. JavaScript 的编译是发生在代码执行前的几微米(甚至更短)的事件内,所以 JavaScript 没有其他语言那么多的时间来进行优化. 当 JavaScr ...
- 4.性能下降原因和常见的Join查询
性能下降 SQL慢,执行时间长,等待时间长 1.查询语句写的烂 2.索引失效 单值索引失效 和 复合索引失效 3.关联查询太多join(设计缺陷或不得已的需求) 4.服务器调优及各个参数设置(缓冲.线 ...
- CentOS7.x忘记root密码如何破解
在CentOS7.x中,有一个单用户模式.CentOS7.x进入单用户模式与CentOS6.x略有不同,要复杂一些. 如果我们忘记了root的密码,可以在单用户模式下重置密码. 注意:此操作必须在服务 ...
- 通过快捷方式lnk获得文件真实路径
通过快捷方式.lnk获得文件真实路径前提最近开发资源管理,需要预先上传大量资源,负责整理资源的同学因为空间不足,直接用快捷键方式整理视频资源OTZ,所以只能想办法通过.lnk文件获得文件的真实地址. ...
- linux——系统命令
(1) 显示系统日期和时间:date 显示系统当前时间 例如:date (1) 切换用户:su 用户名 以其他用户身份使用系统,(类似windows10系统,有些程序以管理员身份执行) ① 从r ...
- 洛谷 P2163 [SHOI2007]园丁的烦恼 (离线sort,树状数组,解决三维偏序问题)
P2163 [SHOI2007]园丁的烦恼 题目描述 很久很久以前,在遥远的大陆上有一个美丽的国家.统治着这个美丽国家的国王是一个园艺爱好者,在他的皇家花园里种植着各种奇花异草. 有一天国王漫步在花园 ...