Gym - 101128E Wooden Signs DP

题目大意:

一共n块木板,前两个数给出最底下木块的两个端点,后面n-1个数给出第i层的一个固定端点,问你木块的所有放置情况。

分析:

状态:

d[i][j]表示第i个木块,第i-1块木板的未固定端点为j的所有方案数

状态转移:

如果a[i]<=min(j,a[i-1),也就是说固定的那一点在i-1块木板的两个端点的左侧,那么只能有一种情况。d[i][r]=(d[i][r]+d[i-1][j])%mod;

同理,如果a[i]>= max(j,a[i-1)。d[i][l]=(d[i][l]+d[i-1][j])%mod;

如果在中间,那么以上两种情况都要算上

#include <bits/stdc++.h>
using namespace std;

typedef long long ll;
const ll mod=2147483647;
const int maxn=2000+5;
int n,a[maxn];
ll d[maxn][maxn];

int main()
{
    while(~scanf("%d",&n))
    {
        for(int i=0; i<=n; i++)
            scanf("%d",&a[i]);
        memset(d,0,sizeof(d));
        d[1][a[0]]=1;
        for(int i=2; i<=n; i++)
        {
            for(int j=1; j<=n+1; j++)
            {
                int l=min(j,a[i-1]);
                int r=max(j,a[i-1]);
                if(a[i]<=l)
                    d[i][r]=(d[i][r]+d[i-1][j])%mod;
                else if(a[i]>=r)
                    d[i][l]=(d[i][l]+d[i-1][j])%mod;
                else
                {
                    d[i][r]=(d[i][r]+d[i-1][j])%mod;
                    d[i][l]=(d[i][l]+d[i-1][j])%mod;
                }
            }
        }
        ll ans=0;
        for(int j=1; j<=n+1; j++)
            ans=(ans+d[n][j])%mod;
        printf("%I64d\n",ans);
    }
    return 0;
}
时间: 2024-10-03 22:53:54

Gym - 101128E Wooden Signs DP的相关文章

dp+分类讨论 Gym 101128E

题目链接:http://codeforces.com/gym/101128 感觉这个人写的不错的(我只看了题目大意):http://blog.csdn.net/v5zsq/article/details/61428924 Description n个小木条,一段前面有一个小箭头,给出第一个小木条的非箭头端端点横坐标以及每个小木条箭头端的坐标,现在要从下往上把这n'个木条按顺序叠放好,要求相邻两个小木条必须有一个共同端点且有交叠部分,问小木条有多少种放法 Input 第一行一整数n表示木条数量,之

Gym 101334D 记忆化dp

大致题意: 给你9堆扑克牌,每堆牌有4张,大小从A~K.每次从9堆牌牌顶抽走两张大小相同的牌,且抽走每一对相同的牌的概率都相等.问可以全部抽完的概率. 分析: 这是一道概率dp题.剩余的牌数作为状态,有9堆,意味着要一个9维数组来存d[i1][i2][i3][i4][i5][i6][i7][i8][i9]表示这个状态的概率,0<=i<=4. 状态转移: 当前状态的概率等于抽走两张牌后所能达到的状态的概率和除以所能达到的状态数 边界d[0][0][0][0][0][0][0][0][0]=1 #

Gym 101201H Paint (离散化+DP)

题意:给定 n 个区间,让你选出一些,使得每个选出区间不交叉,并且覆盖区间最大. 析:最容易想到的先是离散化,然后最先想到的就是 O(n^2)的复杂度,dp[i] = max(dp[j] + a[i].r - a[i].l) 区间不相交,这个可以用线段树来维护一个最大值,因为有区间性,但是也可以不用线段树,直接进行线性DP,因为要选的区间越多越好,相比而言,有就要选,但是有更优的就选最优的. 代码如下: #pragma comment(linker, "/STACK:1024000000,102

Gym - 101615J Grid Coloring DP 2017-2018 ACM-ICPC Pacific Northwest Regional Contest (Div. 1)

题目传送门 题目大意: 给出n*m的网格,有红蓝两种颜色,每个格子都必须被染色,当一个格子被染成蓝色后,这个格子左上方的一块都必须被染成蓝色,问最后的方案数量. 思路: 按照题目条件,如果有一个格子被染成了红色,则这个格子的右下方要全部被染成红色,也就是这个给出的网格能让我们染色的,是一个左上方和右下方都是阶梯型的图形,而对于每一行来说,当一个格子被染成了蓝色,那么左边的所有格子都必须被染成蓝色,所以我们设 f[ i ][ j ] 表示第 i 行 第 j 个格子被染成蓝色的方案数量,那么这个dp

【每日dp】 Gym - 101889E Enigma 数位dp 记忆化搜索

题意:给你一个长度为1000的串以及一个数n 让你将串中的'?'填上数字 使得该串是n的倍数而且最小(没有前导零) 题解:dp,令dp[len][mod]为是否出现过 填到第len位,余数为mod 的状态(dp=0 or 1) 用记忆化搜索来实现,dfs返回1或0. 如果搜到最后一位并且余数为0,返回1. 如果搜到已经更新过的dp状态,直接返回0. 将mod作为全局变量,不断更新mod. 对于每一位 的'? ' 暴力枚举0~9 #include<stdio.h> #include<std

Gym 101170A Arranging Hat dp

Arranging Hat 题目大意: 给你n,m n个m位的数,保证m位,问要是n个按照从小到大排序,求改变最少个数字,使得这n个按照不递增排序,求最后排序的结果. //dp[i][j] 表示前i个数,修改不超过j次的最小值. dp[i][j]向dp[i+1][j+k]转移//pre[i][j]表示第i个数修改j次是由第i-1个数修改pre[i][j]次转移过来的. #include <cstdio> #include <cstring> #include <queue&g

Gym 101490K Safe Racing (dp转换, 超超超级详细,包你看懂)

题意:给你一个长为L的圆形跑道,让你放置警示牌,相邻两个警示牌相隔距离不能超过S,让你求有多少种方案数放置.数据L,S都是1e6. 来个例子:L = 13, S = 5.一个圈表示长度为1. 思路:因为是一个圈,我们必须得判断最后一个牌子和第一个牌子的相距距离,所以如果我们规定了第一个放在那,最后一个放在那,这道题就简单多了.比如这个. 第一步思考: 1 当我们第一个放在1号位子时,我们最后一个能放的位置有12,13,14,15,16. 2 当我们第一个放在2号位子时,我们最后一个能放的位置有1

Codeforces Gym 100676G Training Camp 状压dp

http://codeforces.com/gym/100676 题目大意是告诉你要修n门课,每门课有一个权值w[i], 在第k天修该课程讲获得k*w[i]的学习点数,给出了课程与先修课程的关系,要修该课程必须修完先修课程.问最多能学到多少点数. 非常简单的一道状压dp(一开始我还误导队友写成两维的去了 T^T); dp[s] : s 的二进制存放的是已经选择的课程,在该状态下的能获得的最大的点数. 这时如果再学一门课程k,将转移到状态ss (s | (1 << k) ) ,能否转移需要判断合

Codeforces GYM 100114 D. Selection 线段树维护DP

D. Selection Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/100114 Description When selecting files in an application dialog, Vasya noted that he can get the same selection in different ways. A simple mouse click selects a sing