洛谷P1282 多米诺骨牌 (DP)
洛谷P1282 多米诺骨牌
题目描述
多米诺骨牌有上下2个方块组成,每个方块中有1~6个点。现有排成行的
上方块中点数之和记为S1,下方块中点数之和记为S2,它们的差为|S1-S2|。例如在图8-1中,S1=6+1+1+1=9,S2=1+5+3+2=11,|S1-S2|=2。每个多米诺骨牌可以旋转180°,使得上下两个方块互换位置。 编程用最少的旋转次数使多米诺骨牌上下2行点数之差达到最小。

对于图中的例子,只要将最后一个多米诺骨牌旋转180°,可使上下2行点数之差为0。
输入输出格式
输入格式:
输入文件的第一行是一个正整数n(1≤n≤1000),表示多米诺骨牌数。接下来的n行表示n个多米诺骨牌的点数。每行有两个用空格隔开的正整数,表示多米诺骨牌上下方块中的点数a和b,且1≤a,b≤6。
输出格式:
输出文件仅一行,包含一个整数。表示求得的最小旋转次数。
输入输出样例
输入样例#1:
4
6 1
1 5
1 3
1 2
输出样例#1:
1
Solution
先打了个搜索?45分
#include<bits/stdc++.h>
using namespace std;
const int N=1010,inf=2e9;
int n,tot,ans=inf,tq=inf;
int a[N],b[N];
void dfs(int x,int sum,int cnt) {
if(x==n+1) {
if(abs(tot-sum-sum)<ans) ans=abs(tot-sum-sum),tq=cnt;
else if(abs(tot-sum-sum)==ans) tq=min(tq,cnt);
return;
}
dfs(x+1,sum+a[x],cnt);
dfs(x+1,sum+b[x],cnt+1);
}
int main()
{
ios::sync_with_stdio(0);
cin>>n;
for(int i=1;i<=n;i++) cin>>a[i]>>b[i],tot+=a[i]+b[i];
dfs(1,0,0);
cout<<tq<<endl;
}
为什么要贴这份代码,因为我们可以把我们搜索中记录的变量变成dp中的状态(一位dalao告诉我的做dp的方法?)
我们发现因为当题目给出塔牌的点数后,无论是否翻转,前i张牌的上下点数之和是不变的,那么我们知道一个就可以求出另一个
ok?我们现在来设计\(dp\)数组,按照搜索中的变量,设dp[i][j]表示到第i张纸牌,第一行的和为j的翻转次数
那么状态转移方程就呼之欲出了?
\(dp[i][j]=min(dp[i][j],dp[i-1][j-a[i]])\) 这是不翻转的情况
\(dp[i][j]=min(dp[i][j],dp[i-1][j-b[i]]+1)\) 翻转
我们发现这样转移可能会越界?那就加个条件嘛
if(j-a[i]>=0) dp[i][j]=min(dp[i][j],dp[i-1][j-a[i]]);
if(j-b[i]>=0) dp[i][j]=min(dp[i][j],dp[i-1][j-b[i]]+1);
那么初始化,第二维最大可能是6*n(每张都是6点),所以初始化
for(int i=1;i<=n;i++)
for(int j=0;j<=6*n;j++) dp[i][j]=inf;
然后就是dp边界
dp[1][b[1]]=1,dp[1][a[1]]=0;//为什么是这个顺序,因为如果先a后b,那么当第一张纸牌上下点数相同时,dp[1][a[1]]会被赋值成1
最后枚举一下点数统计答案就可以了,和dfs一样
Code
#include<bits/stdc++.h>
using namespace std;
const int N=1010,inf=2e5;
int n,tot,ans=inf,tq=inf;
int a[N],b[N],dp[N][6*N];
int main()
{
ios::sync_with_stdio(0);
cin>>n;
for(int i=1;i<=n;i++) cin>>a[i]>>b[i],tot+=a[i]+b[i];
for(int i=1;i<=n;i++)
for(int j=0;j<=6*n;j++) dp[i][j]=inf;
dp[1][b[1]]=1,dp[1][a[1]]=0;
for(int i=2;i<=n;i++) {
for(int j=0;j<=6*n;j++) {
if(j-a[i]>=0) dp[i][j]=min(dp[i][j],dp[i-1][j-a[i]]);
if(j-b[i]>=0) dp[i][j]=min(dp[i][j],dp[i-1][j-b[i]]+1);
}
}
for(int i=0;i<=tot;i++) {
if(dp[n][i]!=inf) {
if(abs(tot-i-i)<ans)
ans=abs(tot-i-i),tq=dp[n][i];
else if(abs(tot-i-i)==ans) tq=min(tq,dp[n][i]);
}
}cout<<tq<<endl;
}
博主蒟蒻,随意转载.但必须附上原文链接
http://www.cnblogs.com/real-l/
洛谷P1282 多米诺骨牌 (DP)的更多相关文章
- 洛谷P1282 多米诺骨牌
P1282 多米诺骨牌 题目描述 多米诺骨牌有上下2个方块组成,每个方块中有1~6个点.现有排成行的 上方块中点数之和记为S1,下方块中点数之和记为S2,它们的差为|S1-S2|.例如在图8-1中,S ...
- poj 1717==洛谷P1282 多米诺骨牌
Dominoes Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 6571 Accepted: 2178 Descript ...
- 【01背包】洛谷P1282多米诺骨牌
题目描述 多米诺骨牌有上下2个方块组成,每个方块中有1~6个点.现有排成行的 上方块中点数之和记为S1,下方块中点数之和记为S2,它们的差为|S1-S2|.例如在图8-1中,S1=6+1+1+1=9, ...
- 洛谷 P1282 多米诺骨牌 ( 线性DP )
题意 : 题目链接 分析 : 一开始这个想法也有想到,但是貌似要开很大数组,就感觉应该不行 遂放弃想其他方法,万万没想到注意到可以滚动优化(其实不优化也可以过) 定义 dp[i][j] 表示 到第 ...
- 洛谷P1282 多米诺骨牌【线性dp】
题目:https://www.luogu.org/problemnew/show/P1282 题意: 给定n个牌,每个牌有一个上点数和下点数.可以通过旋转改变交换上下点数. 问使得上点数之和和下点数之 ...
- 洛谷 P1282 多米诺骨牌("01"背包)
传送门 https://www.cnblogs.com/violet-acmer/p/9852294.html 参考资料: [1]:https://blog.csdn.net/Darost/artic ...
- 洛谷 [P1282] 多米诺骨牌
这道题是一道背包问题,考虑一个背包, 显然如果我们直接设dp[i]表示前i个使差值最小所需的最少翻转次数,是具有后效性的. 所以我们将直接求最值,改为求某个值是否可行,这种求最值转变为求可行性的思想是 ...
- yzoj P2043 & 洛谷 P1282 多米诺骨牌 题解
题意 类似于就是背包. 解析 代码 跟解析有点不一样v[i]价值,w[i]重量,s背包容积,背包转移即可. #include<bits/stdc++.h> using namespace ...
- P1282 多米诺骨牌 dp
思路:dp[i][j] 的j是上半段的和的值 这里表示的是达到上半段值是j的最小次数 答案在最小的可达到的j #include<bits/stdc++.h> using namespa ...
随机推荐
- python并发编程之多进程、多线程、异步、协程、通信队列Queue和池Pool的实现和应用
什么是多任务? 简单地说,就是操作系统可以同时运行多个任务.实现多任务有多种方式,线程.进程.协程. 并行和并发的区别? 并发:指的是任务数多余cpu核数,通过操作系统的各种任务调度算法,实现用多个任 ...
- Python学习之模块基础
模块就是程序 编写以下简单代码 print('hello python') 并将py文件保存在c盘的python(假设新建)文件下,通过pycharm的Terminal 或者windom命令窗口调出p ...
- Leecode刷题之旅-C语言/python-67二进制求和
/* * @lc app=leetcode.cn id=67 lang=c * * [67] 二进制求和 * * https://leetcode-cn.com/problems/add-binary ...
- GIT LFS 使用笔记
一.背景 由于git上传文件大小受限,所以我们需要使用GIT LFS对大小超过一定上限的大文件进行处理. 二.安装 linux上安装参见 https://askubuntu.com/questions ...
- Spring事务:一种编程式事务,三种声明式事务
事务隔离级别 隔离级别是指若干个并发的事务之间的隔离程度.TransactionDefinition 接口中定义了五个表示隔离级别的常量: TransactionDefinition.ISOLATIO ...
- 用命令部署WebPart
Webpart一般是一个wsp文件,可以在VS里面通过右键来部署.但一般真正的生产服务器上面是不会安装VS的,所以一般情况下是把wsp文件拷贝到服务器上面然后启动PowerShell用命令来部署. 部 ...
- Android TV 开发(3)
本文来自网易云社区 作者:孙有军 <LinearLayout android:id="@+id/input_num_line_3" and ...
- Spring MVC 开发 配置
1.首先在web.xml中配置一个DispatcherServlet,并通过<servlet-mapping>指定需要拦截的url. 下面xml中配置一个拦截.html为后缀的url. & ...
- gitbook生成的_book文件本地打开后链接失效问题
Gitbook 生成本地 html 的问题 在本地用 gitbook-cli根据 Summary 生成目录 然后在每个 md 文件里书写内容 然后用 gitbook serve .生成本地 html ...
- 最短路径——Bellman-Ford算法以及SPFA算法
说完dijkstra算法,有提到过朴素dij算法无法处理负权边的情况,这里就需要用到Bellman-Ford算法,抛弃贪心的想法,牺牲时间的基础上,换取负权有向图的处理正确. 单源最短路径 Bellm ...