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个仆人过河,过河的工具只有一艘小船,只能同时载两个人过河,包括划船的人.在河的任何一边,只要仆人的数量超过商人的数量,仆人就会联合起来将商人杀死并抢夺其财物,问商人应如何设计过 ...
随机推荐
- P1155 双栈排序
题目描述 Tom最近在研究一个有趣的排序问题.如图所示,通过2个栈S1和S2,Tom希望借助以下4种操作实现将输入序列升序排序. 操作aaa 如果输入序列不为空,将第一个元素压入栈S1 操作b 如果 ...
- java中的变量各占得字节数
boolen,8位1个字节int,32位,4个字节float,32位 4个字节double,64位8个字节char 16位,2个字节byte 8位1个字节short 16位 2个字节long 64位 ...
- SAM
后缀自动机能识别字符串S的所有子串,是一个DAG. http://blog.csdn.net/huanghongxun/article/details/51112764 https://blog.xe ...
- HNOI/AHOI2018题解
作为一名高二老年选手来补一下我省去年的省选题. D1T1:寻宝游戏 按顺序给出\(n\)个\(m\)位的二进制数\(a_i\),再在最前方添一个\(0\), 给出\(q\)次询问,每次询问给出一个同样 ...
- 【BZOJ2989】数列(二进制分组,主席树)
[BZOJ2989]数列(二进制分组,主席树) 题面 BZOJ 权限题啊... Description 给定一个长度为n的正整数数列a[i]. 定义2个位置的graze值为两者位置差与数值差的和,即g ...
- 【NOI2016】区间
目链接:http://uoj.ac/problem/222 在数轴上有 n 个闭区间 [l1,r1],[l2,r2],...,[ln,rn].现在要从中选出 m 个区间,使得这 m 个区间共同包含至少 ...
- codevs2464超级麻将
题目链接http://codevs.cn/problem/2464/ 题目描述 Description 很多人都知道玩麻将,当然也有人不知道,呵呵,不要紧,我在这里简要地介绍一下麻将规则: 普通麻将有 ...
- php版本的code review软件
phabricator, http://www.oschina.net/p/phabricator
- phpredis用法笔记
项目中用到redis集群, 发现phpredis对集群,分布式是有支持的.翻译下相关资料备用. redis扩展地址:https://github.com/phpredis/phpredis, 看到如下 ...
- Python完成RF测试用例
Robot Framework 框架是基于 Python 语言开发的,所以,它本质上是 Python 的一个库. from robot.api import TestSuite from robot. ...