DAG图。

  1. 【题意】
  2. n(50)个城市m(c(n,2))条单向边(x,y),保证x<y
  3. 对于三个点(x,y,z)如果abs(w[x]-w[y])<=K && abs(w[x]-w[y])<=K && abs(w[x]-w[y])<=K则这是一个合法状态。
  4. 问你,如果我们从(x,y,z)出发,可以在合法状态中任意行走任意终止,有多少种不同的行走路径数

f[i][j][k] = ∑f[ii][jj][kk],ii,jj,kk分别为i,j,k的直接后继

时间复杂度是O(n^6)的,需要优化。

另开一维枚举当前要走的人。

我们假定先走k,再走j,最后走i,目前在i,j,k。

f[i][j][k][0]表示k,j,i走完下一轮继续走k,j,i的方案数f[u][j][k][2] += f[i][j][k][0];

f[i][j][k][1]表示k走完下一步走j,再走i的方案数f[i][u][k][1] += f[i][j][k][2];

f[i][j][k][2]表示k,j走完下一步走i的方案数f[i][j][u][0] += f[i][j][k][1];

倒着dp

  1. 具体转移是这样子的——
  2. if (!ok(i, j) || !ok(i, k) || !ok(j, k))f[i][j][k][0] = 0;
  3. else gadd(f[i][j][k][0], 1);
  4. //这个DP的起点条件并不一定是要满足ok(i,j)&&ok(i,k)&&ok(j,k),因为这个状态可能是中途状态
  5. if (f[i][j][k][0])
  6. for (int u = 1; u < i; ++u)if (e[u][i])
  7. gadd(f[u][j][k][2], f[i][j][k][0]);
  8. if(f[i][j][k][2])
  9. for (int u = 1; u < j; ++u)if (e[u][j])
  10. gadd(f[i][u][k][1], f[i][j][k][2]);
  11. if(f[i][j][k][1])
  12. for (int u = 1; u < k; ++u)if (e[u][k])
  13. gadd(f[i][j][u][0], f[i][j][k][1]);
  1. 【题意】
  2. n(50)个城市m(c(n,2))条单向边(x,y),保证x<y
  3. 对于三个点(x,y,z)如果abs(w[x]-w[y])<=K && abs(w[x]-w[y])<=K && abs(w[x]-w[y])<=K则这是一个合法状态。
  4. 问你,如果我们从(x,y,z)出发,可以在合法状态中任意行走任意终止,有多少种不同的行走路径数
  5. 【类型】
  6. 分段式DP 打破题目约束
  7. 【分析】
  8. 这道题可以AC的复杂度最多只能为O(n^4)
  9. 而一个状态是O(n^3),如果我们暴力枚举两个状态,并做转移,复杂度是O(n^6)的。
  10. 于是我们要尝试优化——
  11. 我们发现,我们在转移的时候,可以考虑的不再是三重循环转移,而是分步式转移。
  12. 即,虽然题目要求是三个人同时走,但是我们可以把其转化为三个人轮流走的情况。
  13. 因为同时走的复杂度是是要做三种枚举。所以我们定义状态的一二三步
  14. 即f[i][j][k][0]表示,下一步是i走
  15. 即f[i][j][k][1]表示,下一步是j走
  16. 即f[i][j][k][2]表示,下一步是k走
  17. 这样答案的输出是f[i][j][k][0],这时三个人步长相同。
  18. 因为我们计算的时候,按照基本转移方程,f[i][j][k]+=f[ii][jj][kk],(ii,jj,kk)是(i,j,k)的合法后继
  19. 所以,(i,j,k)较大的要先算出来。于是我们倒着展开DP。
  20. 具体转移是这样子的——
  21. if (!ok(i, j) || !ok(i, k) || !ok(j, k))f[i][j][k][0] = 0;
  22. else gadd(f[i][j][k][0], 1);
  23. //这个DP的起点条件并不一定是要满足ok(i,j)&&ok(i,k)&&ok(j,k),因为这个状态可能是中途状态
  24. if (f[i][j][k][0])
  25. for (int u = 1; u < i; ++u)if (e[u][i])
  26. gadd(f[u][j][k][2], f[i][j][k][0]);
  27. if(f[i][j][k][2])
  28. for (int u = 1; u < j; ++u)if (e[u][j])
  29. gadd(f[i][u][k][1], f[i][j][k][2]);
  30. if(f[i][j][k][1])
  31. for (int u = 1; u < k; ++u)if (e[u][k])
  32. gadd(f[i][j][u][0], f[i][j][k][1]);
  33. 【时间复杂度&&优化】
  34. O(n^4)

HDU5807分段dp的更多相关文章

  1. hdu_5807_Keep In Touch(分段dp)

    题目链接:hdu_5807_Keep In Touch 题意: 在Byteland一共有nn个城市,编号依次为11到nn,同时有mm条单向道路连接着这些城市,其中第ii条道路的起点为u_iu​i​​, ...

  2. hdu3480 Division(dp平行四边形优化)

    题意:将n个数分成m段,每段的代价为最大值减最小值的平方,为代价最小是多少n<=10000 ,m<=5000 题解:先拍好序,从小到大,这样绝对是花费最小的,不过怎么样来做呢?一定很容易想 ...

  3. BZOJ3193 [JLOI2013]地形生成 【dp】

    题目链接 BZOJ3193 题解 注意\(key\)是小于 第一问,显然按高度降序排序,逐个插入 如果高度各不相同,那么之前插入的都比当前插入的\(i\)大,可插入的位置个数就确定了 由于存在高度相同 ...

  4. BZOJ1090: [SCOI2003]字符串折叠

    区间dp. 一种是分段dp[i][j]=min(dp[i][j],dp[i][k]+dp[k+1][j]); 一种是这一段可以缩写dp[i][j]=min(dp[i][j],dp[i][l]+2+ca ...

  5. 2326: [HNOI2011]数学作业 - BZOJ

    首先是DP,分段DP(按位数讨论) 然后每一段构造出它对应的矩阵,用矩阵快速幂加速 type matrix=..,..]of int64; var n,m:int64; a,b,c,d:matrix; ...

  6. Educational Codeforces Round 62 (Rated for Div. 2)

    A. Detective Book 题意:一个人读书  给出每一章埋的坑在第几页可以填完 . 一个人一天如果不填完坑他就会一直看 问几天能把这本书看完 思路:模拟一下 取一下过程中最大的坑的页数  如 ...

  7. BZOJ题目(持续更新)

    bzoj1009:kmp想法+递推+矩阵快速幂.很好的想法,考虑用长串去kmp匹配短串,dp[i][j]表示匹配指针分别指在i.j位置时候,前i位母字符串一共有多少种可能性,那么dp[i][j]=Σd ...

  8. 2017多校Round3(hdu6056~hdu6066)

    补题进度:7/11 1001 待填坑 1002 待填坑 1003(set) 题意: 给定长度为n(n<=5e5)的数组(是n的一个排列)和一个整数k(k<=80),f[l,r]定义为区间[ ...

  9. poj 3744 题解

    题目 题意: $ yyf $ 一开始在 $ 1 $ 号节点他要通过一条有 $ n $ 个地雷的道路,每次前进他有 $ p $ 的概率前进一步,有 $ 1-p $ 的概率前进两步,问他不领盒饭的概率. ...

随机推荐

  1. HDU 3887:Counting Offspring(DFS序+树状数组)

    http://acm.hdu.edu.cn/showproblem.php?pid=3887 题意:给出一个有根树,问对于每一个节点它的子树中有多少个节点的值是小于它的. 思路:这题和那道苹果树是一样 ...

  2. Unix/Linux编程实践教程(二:socket、多线程、进程间通信)

    同一接口不同的数据源: 协同进程: fdopen以文件描述符为参数: fopen和popen: 为了实现popen,必须在子进程中调用sh,因为只有shell本身即/bin/sh可以运行任意shell ...

  3. Effective C++:条款27——条款

    条款27:尽量少做转型动作 单一对象可能拥有一个以上的地址!

  4. okhttp封装

    对这玩意并不熟,网上有很多大神封装好的,但是懒得看里面的封装逻辑,索性自己简单做个封装,方便使用,出现bug也好查找: get请求: /** * get请求 * @param url * @param ...

  5. String类方法

    1.charAt(int index)  返回指定索引处的 char 值. 2. length() 返回此字符串的长度. 3.String replace(char oldChar, char new ...

  6. (摘至)程序员老鸟写sql语句的经验之谈

    做管理系统的,无论是bs结构的还是cs结构的,都不可避免的涉及到数据库表结构的设计,sql语句的编写等.因此在开发系统的时候,表结构设计是否合理,sql语句是否标准,写出的sql性能是否优化往往会成为 ...

  7. MVC模式(Model View Controller)下实现数据库的连接,对数据的删,查操作

    MVC模式(Model View Controller): Model:DAO模型 View:JSP  在页面上填写java代码实现显示 Controller:Servlet 重定向和请求的转发: 若 ...

  8. poj3263 Tallest Cow

    题意略去. 考虑给定的R对pair(A, B). 即A能看见B,这意味着B不比A低,并且区间内部的所有元素的高度严格小于A的高度. 我们规定区间的方向:若A > B,为反方向,反之称为正方向. ...

  9. 关于List.ToArray()方法的效率测试

    之前一直认为因为List内部是数组,ToArray的实现只是将数组返回出去而已. 今天测了一下发现并不是那样 var a = new List<int>(); ; i < ; i++ ...

  10. 静态方法和类成员方法(Python)

    静态方法和成员方法分别在创建时分别被装入Staticmethod 类型和 Classmethod类型的对象中.静态方法的定义没有 self参数,且能够被类本身直接调用,类方法在定义时需要名为 cls的 ...