先看第一题,有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)的更多相关文章

  1. UVA 12075 - Counting Triangles(容斥原理计数)

    题目链接:12075 - Counting Triangles 题意:求n * m矩形内,最多能组成几个三角形 这题和UVA 1393类似,把总情况扣去三点共线情况,那么问题转化为求三点共线的情况,对 ...

  2. uva 1393 - Highways(容斥原理)

    题目连接:uva 1393 - Highways 题目大意:给定一个m∗n的矩阵,将矩阵上的点两两相连,问有多少条直线至少经过两点. 解题思路:头一次做这样的题目,卡了一晚上. dp[i][j]即为i ...

  3. UVA 12075 Counting Triangles

    https://icpcarchive.ecs.baylor.edu/index.php?option=com_onlinejudge&Itemid=8&page=show_probl ...

  4. UVa 1393 - Highways(数论)

    链接: https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...

  5. UVA 1393 Highways(数学思想)

    题意:给你n.m(n,m<=200),问你有多少条非水平.非垂直的直线有多少条经过至少两个点 题解:我们需要枚举的是只画一条线的矩形,对于大小a*b的矩形必须保证gcd(a,b)=1才能不重复 ...

  6. UVA 1393 Highways

    https://vjudge.net/problem/UVA-1393 题意: a*b的点阵中能画多少条非水平非竖直的直线 方向‘/’ 和 方向 ‘\’ 对称 枚举直线所在矩形的i*j 直线可能重复的 ...

  7. UVa 1393 (容斥原理、GCD) Highways

    题意: 给出一个n行m列的点阵,求共有多少条非水平非竖直线至少经过其中两点. 分析: 首先说紫书上的思路,编程较简单且容易理解.由于对称性,所以只统计“\”这种线型的,最后乘2即是答案. 枚举斜线包围 ...

  8. hdu 1396 Counting Triangles(递推)

    Counting Triangles Problem Description Given an equilateral triangle with n thelength of its side, p ...

  9. Counting Triangles(hd1396)

    Counting Triangles Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Other ...

随机推荐

  1. IDEA中安装go插件,如何能够配置go SDK?

    最近在学习go语言,一个是因为区块链的技术热潮,另一个是接手的项目有用到go写多线程高并发,因此决定自学go. 第一个遇到的问题就是环境! 通过一个晚上的摸索,大概步骤如下: 在IDEA中先打开set ...

  2. QT调用CHM方法

    QDesktopServices desktopServices;QString strUrl=QCoreApplication::applicationDirPath () ;strUrl=QStr ...

  3. OpenCl入门——实现简单卷积

    现在的卷积实现无非是那么几种:直接卷积.im2col+gemm.局部gemm.wingrod.FFT.如果直接卷积的话,其实kernel函数是比较好实现.以下代码参考至<OpenCL Progr ...

  4. SQL学习——BETWEEN运算符

    原文链接 BETWEEN的作用 BETWEEN 操作符用于选取介于两个值之间的数据范围内的值. BETWEEN的边界 BETWEEN运算符选择给定范围内的值.值可以是数字,文本或日期. BETWEEN ...

  5. 网页免费转换为可编辑的PDF

    Chrome自带的"打印"功能中,另存为PDF 可选择保存选中的内容.如果浏览器/网络出错,不能纠正.(推荐0) https://www.printfriendly.com (有C ...

  6. Spring源码解析 - springMVC核心代码

    一.首先来讲解下springMVC的底层工作流程 1.首先我们重点放在前端控制器(DispatcherServlet) 其类图: 因为从流程图看,用户的请求最先到达就是DispatcherServle ...

  7. 3.第一个MyBatis程序_进化

    1.使用工具类 将SqlSession的获取 封装成一个工具 private static SqlSession session = null; static { try { InputStream ...

  8. asp.net网站部署在云服务器windows server 2008上

    搭建一个网站需要以下4个准备: 1.域名解析 2.(云)服务器 3.数据库 4.网站代码 其中1可以可以去DNSPOD申请,同时需要进行备案,在上面就都可以完成.2用的是阿里云服务器windows s ...

  9. Linux内核的arch目录

  10. Django单表查询及其方法

    单表查询 前期准备 首先新建一个test的python文件,然后再manage.py中导入main语句及其下面的复制到新文件中 并导入django 写上django.setup() 就可以导入对应的m ...