Sicily 1211. 商人的宣传
Bruce是K国的商人,他在A州成立了自己的公司,这次他的公司生产出了一批性能很好的产品,准备宣传活动开始后的第L天到达B州进行新品拍卖,期间Bruce打算将产品拿到各个州去做推销宣传,以增加其影响力。
K国有很多个州,每个州都与其他一些州相邻,但是K国对商人作宣传却有一些很奇怪的规定:
1、 商人只能从某些州到达另外一些州,即连通路线是单向的,而且有些州可能是到达不了的。
2、 商人不允许在同一个州连续宣传两天或以上,每天宣传完必须离开该州。
3、 商人可以多次来到同一个州进行宣传。
"我必须找出一条影响力最大的路线才行",Bruce想,"但我首先必须知道到底有多少这种符合规定的宣传路线可供我选择。"现在Bruce把任务交给了你。并且出于考虑以后的需要,你还要帮他算出给出的两州之间的路线的总数。
第m+2行为一个整数q(1≤q≤100),表示Bruce有q个询问。下面q行每行两个整数A,B(1≤A,B≤n),即A、B州的位置。
输入数据中的询问将保证答案t在长整数范围内,即t<231。
4 5 6
1 2
2 3
3 4
4 1
2 4
2
1 4
4 2
2
1
Problem Source: ZSUACM Team Member
思路:
一开始看到这道题觉得应该是用深搜来写,因为N<=100,觉得规模并不大,没想到提交之后却超时了,后来想了想每次深搜需要的时间复杂度位O(n^3),可是要查询Q次,Q<=100,所以整个算法的时间复杂度位O(n^4),超时也就不奇怪了。
接着用动态规划做这道题,想了想,状态转移方程是:f[a][b][l] = ∑ f[c][b][l-1],其中c为a可以用一天到达的州(即有一条从a到c的路)。转移方程的意思是:如果从c州到b州可以在"l-1"天之内完成,那么从a州到b州可以在"l"天之内完成。
我的代码:
#include <iostream>
#include <cstring>
#include <vector>
using namespace std; const int MAX = ;
int n, m, l, q;
int start, end;
vector<int> rodes[MAX]; //用来记录路
int dp[MAX][MAX][MAX]; void findWays() {
for (int k = ; k <= l; k++) { //从2天开始找
for (int i = ; i <= n; i++) {
int len = rodes[i].size();
for (int j = ; j <= n; j++) {
for (int x = ; x < len; x++) { //rodes[i].at((x) 即为i州可以一天之内到达的州
dp[i][j][k] += dp[rodes[i].at(x)][j][k-]; //状态转移方程
//cout << i << ' ' << j << ' ' << k << ' ' << dp[i][j][k] << endl;;
}
}
}
}
} int main() {
cin >> n >> m >> l;
memset(dp, , sizeof(dp)); //dp全部初始化为0
int a, b;
for (int i = ; i < m; i++) {
cin >> a >> b;
rodes[a].push_back(b); //在a州添加一条到b州的路
dp[a][b][] = ; //从a到b在一天内的走法为一种
}
findWays();
cin >> q;
for (int i = ; i < q; i++) {
cin >> start >> end;
cout << dp[start][end][l] << endl;
}
return ;
}
另一种思路:
只是另一种代码的思路,上述代码中的空间复杂度有些大,所以考虑到可以用斐波那契数列的思路考虑,所以可以将MATRIX[A][B][L],转化为三个MATRIX[A][B]进行状态转移,即第一个矩阵存L=1时的路径数,第二个矩阵存L=i-1(2 <= i <= m)时的路径数,第三个矩阵存L=i时的路径数,这样大大节省了内存占有。
代码如下:
#include <iostream>
#include <cstring>
using namespace std; const int MAX = ; int matrix_one[MAX][MAX];
int matrix_two[MAX][MAX];
int matrix_three[MAX][MAX]; int main() {
int n, m, l, q;
int start, end;
int a, b;
cin >> n >> m >> l;
memset(matrix_one, , sizeof(matrix_one)); //现将前两格矩阵初始化为0,即无路
memset(matrix_two, , sizeof(matrix_two));
for (int i = ; i < m; i++) {
cin >> a >> b;
matrix_one[a][b]++; //每增加一条路,从a到b的路径数加一 即位L=1时的路径数矩阵
matrix_two[a][b]++; //第二个矩阵也记录L=1时的情况 从而推倒出之后的矩阵
}
for (int x = ; x <= l; x++) { //L从2开始
memset(matrix_three, , sizeof(matrix_three));
for (int i = ; i <= n; i++) {
for (int j = ; j <= n; j++) {
for (int k = ; k <= n; k++) {
matrix_three[i][j] += matrix_one[i][k] * matrix_two[k][j];
}
}
}
/* 将第三个矩阵的值给第二个矩阵 */
memcpy(matrix_two, matrix_three, sizeof(matrix_one));
}
cin >> q;
while (q--) {
cin >> start >> end;
cout << matrix_three[start][end] << endl;
}
return ;
}
两种思路各有各的优势,第一种速度快一些,第二种内存占用少一些,下图(第一行为第二种,第二行第一种):
其实差不多啦 能ac就好。
Sicily 1211. 商人的宣传的更多相关文章
- [SOJ] 商人的宣传
Description Bruce是K国的商人,他在A州成立了自己的公司,这次他的公司生产出了一批性能很好的产品,准备宣传活动开始后的第L天到达B州进行新品拍卖,期间Bruce打算将产品拿到各个州去做 ...
- sicily 题目分类
为了方便刷题,直接把分类保存下来方便来找. 转自:http://dengbaoleng.iteye.com/blog/1505083 [数据结构/图论] 1310Right-HeavyTree笛卡尔树 ...
- [原创]webapp/css3实战,制作一个《炉石传说》宣传页
在移动网页,尤其是webapp中常需要用到大量的css3动画,来获得良好交互体验 我之前帮朋友做了一个,可惜没帮上忙现在和大家分享一下 目标是要做一个<炉石传说>游戏的介绍宣传页面,文字内 ...
- sicily 中缀表达式转后缀表达式
题目描述 将中缀表达式(infix expression)转换为后缀表达式(postfix expression).假设中缀表达式中的操作数均以单个英文字母表示,且其中只包含左括号'(',右括号‘)’ ...
- PMD宣传文案
队名:Clover 李烈争 031402614 林昊斌 031402615 李坤隆 031402612 林瑞斌 031402617 解宇虹 031402338 林 锦 031402339 目录 一. ...
- sicily 1934. 移动小球
Description 你有一些小球,从左到右依次编号为1,2,3,...,n. 你可以执行两种指令(1或者2).其中, 1 X Y表示把小球X移动到小球Y的左边, 2 X Y表示把小球X移动到小球Y ...
- IT小喇叭-企业品牌宣传、产品营销推广的首选
IT小喇叭-企业品牌宣传.产品营销推广的首选 IT小喇叭,成立于2015年6月初,成都芮嘉科技有限公司旗下产品,主要进行媒体资源整合.宣传报道:使移动互联网等相关企业的产品宣传.品牌营销变得更加方便. ...
- 三星s4宣传片配色有惊喜
三星s4宣传片配色有惊喜据了解,一周前,三星曾对外发布了新旗舰手机galaxy s4的宣传视频,不过那份视频里所含信息仅仅只有s4发布会的邀请函.而日前,三星官方发布了s4的第二弹宣传片则暗示该机在配 ...
- 商人过河问题(DFS)
问题描述:3个商人带着3个仆人过河,过河的工具只有一艘小船,只能同时载两个人过河,包括划船的人.在河的任何一边,只要仆人的数量超过商人的数量,仆人就会联合起来将商人杀死并抢夺其财物,问商人应如何设计过 ...
随机推荐
- 【Linux笔记】CentOS 7 systemctl、firewalld
一.CentOS7 systemctl 在CentOS7中,进行chkconfig命令操作时会发现有类似“systemctl.....”的提示,systemctl可以简单实现service和chkco ...
- 知乎回答:每日完成任务用于打卡的APP
其实我也一直在寻找最佳的解决方案.安卓上有这样的,叫做HabitHub.用过一段时间,不容易坚持,这个软件对于培养习惯来说无疑是最好的,但是仅限于记录,这样的软件实在太多.我这里提供一下我自己的方法, ...
- 【刷题】BZOJ 2005 [Noi2010]能量采集
Description 栋栋有一块长方形的地,他在地上种了一种能量植物,这种植物可以采集太阳光的能量.在这些植物采集能量后,栋栋再使用一个能量汇集机器把这些植物采集到的能量汇集到一起. 栋栋的植物种得 ...
- java10 新特性 详解
引言: 点击-->java9 新特性 详解 点击-->java8 新特性 详解 正题: 1.局部变量var 将前端思想var关键字引入java后段,自动检测所属于类型,一种情况除外,不能为 ...
- THUWC2017随机二分图
题面链接 洛谷 sol 唯一的重点是拆边... 0的不管,只看1.2. 先无论如何把两条边的边权赋为\(0.5\)然后我们发现如果两个都选了. 对于第一种边,我们发现如果\(\frac{1}{2} * ...
- C函数调用机制及栈帧指针
http://blog.csdn.net/jjiss318/article/details/7185802
- java随机数的有趣用法
直接用代码说明,比较容易理解 package com.wz.other; import java.util.Random;import java.util.concurrent.ThreadLocal ...
- HEOI 2017 游记
HEOI2017也算是落下帷幕了,那就写一篇 流水账 游记好了. DAY 0 又是熟悉的大学,又是熟悉的机房 YD宾馆的房间依旧破的不行. 晚上在房间颓颓颓....=.= DAY 1 上午去试机,唯一 ...
- fzyzojP3782 -组合数问题
这个ai<=2000有点意思 启发我们用O(W^2)的算法 FFT不存在,对应关系过紧 考虑组合意义转化建模,再进行分离 (除以2不需要逆元不懂为啥,但是算个逆元总不费事) 由于终点可能在起点的 ...
- for,while,do while
long i; ;i<;i++) printf( printf("%ld\n",i); ) printf("b\n"); i=; do { printf( ...