hdu多校第五场1007 (hdu6630) permutation 2 dp

题意:

给你n个数,求如下限制条件下的排列数:1,第一位必须是x,2,最后一位必须是y,3,相邻两位之差小于等于2

题解:

如果x<y,那么考虑把整个数列翻转过来,减少讨论分支。

设dp[n]为限制1和n在两边,相邻的数之差小于等于2的排列方案。

dp[0]=1  dp[1]=1  dp[2]=2  dp[3]=3

如果x==1 y==n 直接用公式dp[i]=dp[i-1]+dp[i-3]求解,将i代入为n-1-1。

如果x!=1 或者y!=n,假如对于12个的情况,x=4,y=9

那么,1,2,3,5与8,10,11,12必然挤在两边,

以这样的形式:4,2,1,3,5......8,10,12,11,9

因为必须把x+1,y-1放在6,7的两边,答案就是dp[9-1-4-1-1]=dp[2]

两边的数只有一边不靠边则分类讨论。

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int M = 1e5 + 5;
const LL mod = 998244353;
const LL lINF = 0x3f3f3f3f3f3f3f3f;
LL dp[M];
int l, r, n;
int t;
int main()
{
    dp[0] = 1;
    dp[1] = 1;
    dp[2] = 2;
    dp[3] = 3;
    for (int i = 4; i <= M - 2; i++)
    {
        dp[i] = (dp[i - 1] + dp[i - 3])%mod;
    }
    scanf("%d", &t);
    while (t--)
    {
        scanf("%d%d%d", &n, &l, &r);
        if (l > r)
            swap(l, r);
        if (l != 1)
            l++;
        if (r != n)
            r--;
        if (l > r)
            printf("0\n");
        else if (l == r)
            printf("1\n");
        else
            printf("%lld\n", dp[r - l - 1]);
    }
}

原文地址:https://www.cnblogs.com/isakovsky/p/11306281.html

时间: 2024-08-29 15:15:02

hdu多校第五场1007 (hdu6630) permutation 2 dp的相关文章

2014 HDU多校弟五场J题 【矩阵乘积】

题意很简单,就是两个大矩阵相乘,然后求乘积. 用 Strassen算法 的话,当N的规模达到100左右就会StackOverFlow了 况且输入的数据范围可达到800,如果变量还不用全局变量的话连内存开辟都开不出来 1 #pragma comment(linker, "/STACK:16777216") 2 #include <iostream> 3 #include <stdio.h> 4 #define ll long long 5 using namesp

2014 HDU多校弟五场A题 【归并排序求逆序对】

这题是2Y,第一次WA贡献给了没有long long 的答案QAQ 题意不难理解,解题方法不难. 先用归并排序求出原串中逆序对的个数然后拿来减去k即可,如果答案小于0,则取0 学习了归并排序求逆序对的方法,可以拿来当模板 TVT 贴代码了: 1 #include <stdio.h> 2 #include <string.h> 3 #include <stdlib.h> 4 #include <math.h> 5 #include <iostream&g

2014 HDU多校弟九场I题 不会DP也能水出来的简单DP题

听了ZWK大大的思路,就立马1A了 思路是这样的: 算最小GPA的时候,首先每个科目分配到69分(不足的话直接输出GPA 2),然后FOR循环下来使REMAIN POINT减少,每个科目的上限加到100即可 算最大GPA的时候,首先每个科目分配到60分,然后FOR循环下来使REMAIN POINT减少,每个科目的上限加到85即可,如果还有REMAIN POINT,就FOR循环下来加到100上限即可 不会DP 阿 QAQ 过段时间得好好看DP了  =  = 于是默默的把这题标记为<水题集> //

hdu多校第三场 1007 (hdu6609) Find the answer 线段树

题意: 给定一组数,共n个,第i次把第i个数扔进来,要求你删掉前i-1个数中的一些(不许删掉刚加进来这个数),使得前i个数相加的和小于m.问你对于每个i,最少需要删掉几个数字. 题解: 肯定是优先删大数,一开始想的方法类似于尺取,就是维护一个大顶堆作为现有的数,小顶堆作为要删的数,每次大顶堆的元素总和大于m了,就把堆顶扔进小顶堆,扔完了,再把小顶堆的堆顶扔回大顶堆,在会导致大顶堆值总和大于m时停止. 但是很容易会被1 1 1 1 1 1 1 1 1 1 100 这样的数据卡掉. 然后想到了,维护

hdu多校第四场 1007 (hdu6620) Just an Old Puzzle 逆序对

题意: 给你一个数字拼图,问你数字拼图能否能复原成原来的样子. 题解: 数字拼图的性质是,逆序数奇偶相同时,可以互相转化,逆序数奇偶不同,不能互相转化. 因此统计逆序对即可. #include<iostream> using namespace std; int main(){ int t; scanf("%d",&t); while(t--){ int a[20]; int kg; for(int i=1;i<=16;i++){ scanf("%d

hdu多校第五场1004 (hdu6627) equation 1 计算几何

题意: 给你一个C,再给你n组a,b,让你求x取什么值的时候,$ \sum_{i=1}^n |a_i*x+b_i| =C $,要求求出解的个数,并用最简分数从小到大表示,如果有无穷多解,输出-1. 题解: 其实这些方程就是在平面上的一组曲线,都是V形的,最低点都在x轴上,求出所有的零点,以这个零点从左到右排序. 容易看出,这些函数之和也是一条曲线y=f(i),这条曲线最多有n个转折点,那么就在这n个转折点分出的n+1个区间内,和n个点上,用比例公式找和y=C的交点即可.无穷多解的情况是存在一条与

2014多校第五场1010 || HDU 4920 Matrix multiplication(矩阵乘法优化)

题目链接 题意 : 给你两个n*n的矩阵,然后两个相乘得出结果是多少. 思路 :一开始因为知道会超时所以没敢用最普通的方法做,所以一直在想要怎么处理,没想到鹏哥告诉我们后台数据是随机跑的,所以极端数据是不可能会有的,而我们一开始一直在想极端数据能接受的方法......后来看了鹏哥的做法,就是把是0的地方都跳过就可以了,用矩阵保存前一个非0数的位置是多少.二师兄给我看了一个代码,人家根本没用别的优化,直接将最里层k的循环提到了最外层,然后就AC了,对此我表示无语. 1 #include <cstd

2014多校第五场1001 || HDU 4911 Inversion (归并求逆序数)

题目链接 题意 : 给你一个数列,可以随意交换两相邻元素,交换次数不超过k次,让你找出i < j 且ai > aj的(i,j)的对数最小是多少对. 思路 : 一开始想的很多,各种都想了,后来终于想出来这根本就是求逆序数嘛,可以用归并排序,也可以用树状数组,不过我们用树状数组做错了,也不知道为什么.求出逆序数来再减掉k次,就可以求出最终结果来了.求逆序数链接1,链接2 1 #include <stdio.h> 2 3 int left[250003], right[250003];

多校第五场 归并排序

HDU 4911 Inversion 考点:归并排序 思路:这题呀比赛的时候忘了知道可以用归并排序算出逆序数,但是忘了归并排序的实质了,然后不会做-- 因为看到题上说是相邻的两个数才能交换的时候,感觉归并排序好像不是得要相邻的呀,然后就这样晕--刚才重新看了才发现,归并就是相邻的交换的,正好是用来求逆序数的,唉--真的是做这个归并排序比赛就来了--真好! #include<iostream> #include<cstdio> #include<cstring> #inc