大致题意:你有$1$到$N$的所有正整数每个数两个,现在需要你将它排成一个序列,使得序列为单峰序列(存在一个$j \in [1,2N]$满足$a_1 \leq a_2 \leq .. \leq a_j \geq a_{j+1} ... \geq a_{2N}$),并且满足输入中给出的$K$个约束条件。$N \leq 35 , K \leq 100$

因为序列要满足单峰,所以考虑从小到大在没有加入数字的序列位置的最左和最右加入元素,这样序列将会一直满足单峰性质
考虑状态$f_{p,q}$表示左边已经填了$p$个,右边填了$q$个的方案(其实可以压掉其中一维),考虑第$i$个数字的转移,有$3$种:
$a:$将这两个$i$都放在序列最左边;$b:$将这两个$i$放在序列最右边;$c.$将这两个$i$一个放一边。
对于$K$个约束条件我们可以在加入的时候直接判断:当前加入的两个数相等,没加入的位置的数必定比当前加的数大,而其他已经填好的地方一定比当前的数小。
最后的答案是$\frac{\sum\limits_{i=0}^{2n}f_{i,2n-i}}{3}$,除3是因为对于最后一次加入,$abc$三种方案都是可行的,但是我们只能算一种。
注意判断约束条件时的细节,约束条件的存储可以使用类似邻接表的方法,在加入约束条件时如加入无向边一样变成两个约束条件,判断起来会方便一些。
 #include<bits/stdc++.h>
 #define int long long
 using namespace std;

 ][] , numOp[];

 vector < ] , op[];
 map < string , int > opToNum;

 bool ifOK(int l , int r , int que1 , int que2){
 //特别注意判断que1位置与que2位置有联系的情况
      ; i < numOp[que1] ; i++)
         switch(op[que1][i]){
             :
                 if(num[que1][i] < r && num[que1][i] > l || num[que1][i] == que2)
                     ;
                 break;
             :
                 if(num[que1][i] < r && num[que1][i] > l && num[que1][i] != que2)
                     ;
                 break;
             :
                 if(!(num[que1][i] == que2))
                     ;
                 break;
             :
                 if(!(num[que1][i] <= r && num[que1][i] >= l || num[que1][i] == que2))
                     ;
                 break;
             :
                 if(!(num[que1][i] < r && num[que1][i] > l) || num[que1][i] == que2)
                     ;
         }
      ; i < numOp[que2] ; i++)
         switch(op[que2][i]){
             :
                 if(num[que2][i] < r && num[que2][i] > l || num[que2][i] == que1)
                     ;
                 break;
             :
                 if(num[que2][i] < r && num[que2][i] > l && num[que2][i] != que1)
                     ;
                 break;
             :
                 if(!(num[que2][i] == que1))
                     ;
                 break;
             :
                 if(!(num[que2][i] <= r && num[que2][i] >= l || num[que2][i] == que1))
                     ;
                 break;
             :
                 if(!(num[que2][i] < r && num[que2][i] > l) || num[que2][i] == que1)
                     ;
         }
     ;
 }

 main(){
     opToNum.insert(make_pair());
     opToNum.insert(make_pair());
     opToNum.insert(make_pair());
     opToNum.insert(make_pair());
     opToNum.insert(make_pair());
     int N , K;
     for(cin >> N >> K ; K ; K--){
         int a , b , t;
         string s;
         cin >> a >> s >> b;
         t = opToNum.find(s)->second;
         if(a == b)
              || t == ){
                 cout << ;
                 ;
             }
             else
                 continue;
         num[a].push_back(b);
         op[a].push_back(t);
         num[b].push_back(a);
         op[b].push_back( - t);
         numOp[a]++;
         numOp[b]++;
     }
     ans[][] = ;
      ; i <= N ; i++){
     //DP
          * i ; j >=  ; j--)
             ][ * i - j] && ifOK(j ,  * N -  * i + j +  , j , j - ))
                 ans[j][ * i - j] += ans[j - ][ * i - j];
          * i ; j >=  ; j--)
              * i - j][j - ] && ifOK( * i - j ,  * N - j +  ,  * N - j +  ,  * N - j + ))
                 ans[ * i - j][j] += ans[ * i - j][j - ];
          ; j <  * i ; j++)
              * i - j - ][j - ] && ifOK( * i - j ,  * N - j +  ,  * i - j ,  * N - j + ))
                 ans[ * i - j][j] += ans[ * i - j - ][j - ];
     }
     ;
      ; i <=  * N ; i++)
         all += ans[i][ * N - i];
     cout << all / ;
     ;
 }

51NOD1522 上下序列/CF567F Mausoleum DP的更多相关文章

  1. 括号序列(区间dp)

    括号序列(区间dp) 输入一个长度不超过100的,由"(",")","[",")"组成的序列,请添加尽量少的括号,得到一 ...

  2. 区间和序列上的dp

    区间上的dp状态设计最基本的形式: \(F[i]\)表示以i结尾的最优值或方案数. \(F[i][k]\)表示以i结尾附加信息为k的最优值或方案数. 当然可以有多维附加信息. 转移的话往往是枚举上一个 ...

  3. bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp)

    bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp) bzoj Luogu 题解时间 给两个小写字母串 $ A $ , $ B $ ,请你计算: ...

  4. LGTB与序列 状压dp

    考试一看我就想到了状压dp.当时没有想到素数,以为每一位只有0~9这些数,就开始压了.后来发现是小于30,然后改到了15,发现数据一点不给面子,一个小点得数都没有,完美爆零.. 考虑到bi最多变成58 ...

  5. 「长乐集训 2017 Day10」划分序列 (二分 dp)

    「长乐集训 2017 Day10」划分序列 题目描述 给定一个长度为 n nn 的序列 Ai A_iA​i​​,现在要求把这个序列分成恰好 K KK 段,(每一段是一个连续子序列,且每个元素恰好属于一 ...

  6. 【Foreign】划分序列 [线段树][DP]

    划分序列 Time Limit: 20 Sec  Memory Limit: 256 MB Description Input Output 仅一行一个整数表示答案. Sample Input 9 4 ...

  7. 九度OJ 1077:最大序列和 (DP)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:5600 解决:1637 题目描述: 给出一个整数序列S,其中有N个数,定义其中一个非空连续子序列T中所有数的和为T的"序列和&qu ...

  8. 算法复习——求最长不下降序列长度(dp算法)

    题目: 题目背景 161114-练习-DAY1-AHSDFZ T2 题目描述 有 N 辆列车,标记为 1,2,3,…,N.它们按照一定的次序进站,站台共有 K 个轨道,轨道遵从先进先出的原则.列车进入 ...

  9. hdu 4521 小明系列问题——小明序列(线段树+DP或扩展成经典的LIS)

    小明系列问题--小明序列 Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others) Tot ...

随机推荐

  1. 伪类选择器 E:nth-child(n)、E:nth-of-type(n)

    结构伪类是css3新增的类型选择器. E:nth-child(n).E:nth-of-type(n)就是其中之二.搞懂了这两个,基本上跟他俩一家的其他几个兄弟伙基本上也可以弄懂了. 首先先用文字语言来 ...

  2. 通过url动态获取图片大小方法总结

    很多时候再项目中,我们往往需要先获取图片的大小再加载图片,但是某些特定场景,如用过cocos2d-js的人都知道,在它那里只能按比例缩放大小,是无法设置指定大小的图片的,这就是cocos2d-js 的 ...

  3. 记一次nginx php配置的心路历程

    1.本来搞好了php的配置,想把目录下移一层 从 www.abc.com 变成 www.abc.com/wxapi ,由于我的真实文件目录比路由少了一层public 尝试了很多办法都不行 甚至想到了u ...

  4. aws s3文件上传设置accesskey、secretkey、sessiontoken

    背景: 最近跟进的项目会封装aws S3资源管理细节,对外提供获取文件上传凭证的API,业务方使用获取到的凭证信息直接请求aws进行文件上传.因此,测试过程需要验证S3文件上传的有效性.aws官网有提 ...

  5. Linux下postgresql数据库部署与配置

    1.检查postgresql是否已经安装:rpm -qa | grep postgres 2.检查PostgreSQL 安装位置:rpm -qal | grep postgres 3.卸载Postgr ...

  6. 洗礼灵魂,修炼python(9)--灵性的字符串

    python几大核心之——字符串 1.什么是字符串 其实前面说到数据类型时说过了,就是带有引号的参数,“”引号内的一切东西就是字符串,字符串又叫文本. 2.创建字符串的两种方式: 3.字符串的方法: ...

  7. jquery判断checkBox的checked

    jquery判断checked的三种方法:.attr('checked):   //看版本1.6+返回:”checked”或”undefined” ;1.5-返回:true或false.prop('c ...

  8. weblogic CVE-2018-2628漏洞利用工具

    weblogic CVE-2018-2628漏洞利用 漏洞环境: Windows2018R2 weblogic10.3.6 漏洞利用过程: 搭建好存在CVE-2018-2628漏洞的weblogic平 ...

  9. 请问在EXECUTE IMMEDIATE中如何使用带有引号

    转自:http://bbs.csdn.net/topics/300191423 从第一引号向后,如果遇到第二个引号,则看这个引号后面时候有紧挨着的引号,如果有则第二个引号被转义,即该保留该引号后面紧跟 ...

  10. tidb在DDL语句方面的测试

    Mysql与tidb测试数据为8000万行. 1.修改一个字段的列名,比如将“ctime”修改为“cctime”. Tidb测试: MySQL测试: 2.同一属性之间切换,即修改一个字段的属性大小.比 ...