#include<stdio.h>

#include<string>

#include<map>

#include<vector>

#include<cmath>

#include<stdlib.h>

#include<string.h>

#include<algorithm>

#include<iostream>

using namespace std;

const int N=105;

const int MOD=1e9+7;

int n;

int a[N];

char ch[N];

void rd(int&x){

    char ch;

    for(ch=getchar();ch<'0' || ch>'9';ch=getchar());

    x=0;

    for(;ch>='0' && ch<='9';ch=getchar()) x=x*10+ch-'0';

}

long long res;

long long sum[N][N][N];  /*sum[i][j][k]表示区间i到j最后一个取第k个符号的答案和   ,sum[i][j][n+1]表示区间i到的答案总和*/

long long A[N];

long long calc(long long x,long long y,char c){

    if(c=='+') return (x+y)%MOD;

    else if(c=='-') return (x-y+MOD)%MOD;

    else return (x*y)%MOD;

}



long long C[N][N];

void solve(){

    memset(sol,0,sizeof sol);

    memset(sum,0,sizeof sum);

    for(int i=1;i<=n;i++){

        sum[i][i][n+1]=a[i];

    }

    for(int i=1;i<n;i++){

        sum[i][i+1][n+1]=sum[i][i+1][i]=calc(a[i],a[i+1],ch[i]);

    }

    for(int j=3;j<=n;j++)

        for(int i=1;i+j-1<=n;i++){

            int l=i,r=i+j-1;

            for(int k=l;k<r;k++){

                if(ch[k]=='+'){

                /*

                第k个符号左边   第l到k个数字。这个区间的方案个数 A[k-l]          sum[l][k][n+1] l到k的和

                第k个符号右边   第k+1到r个数字  这个区间的方案个数 A[r-k-1]    sum[k+1][r][n+1]  k+1到r的和

                设左边 A[k-l]个值   a1,a2, ........   ai指的都是左边这个区间由某一顺序得到的一个值,能够将它和一个操作序列等同

                设右边 A[r-k-1] 个值   b1,b2, ........   

                (a1+a2........) +  (b1,b2.............)         对于每一个a,被加A[r-k-1]次 ,对于每一个b。被加A[k-l]次

                减法同理

                乘法特殊一点

                (a1+a2........) *  (b1,b2.............)  乘法分配率,直接将两部分的总和相乘就可以

            */

            sum[l][r][k]=(sum[l][k][n+1]*A[r-k-1]+sum[k+1][r][n+1]*A[k-l])%MOD;

                }else if(ch[k]=='-'){

                    sum[l][r][k]=(sum[l][k][n+1]*A[r-k-1]-sum[k+1][r][n+1]*A[k-l])%MOD;

                    sum[l][r][k]=(sum[l][r][k]+MOD)%MOD;

                }else{

                    sum[l][r][k]=(sum[l][k][n+1]*sum[k+1][r][n+1])%MOD;

                }

                /*

                之前算的是(左边的选择序列)+(右边的选择序列)。可是左右两边也有先后。再乘个组合数,比赛时这个想了好久才想到

                对于上面的每一个ai和bj。他们在  遇到区间的最后一个操作符号后,须要再乘以一个组合数,由于之前的ai和bj是这种

                ai:(操作1,操作2,操作3.。。。。操作n)+  bj:(操作n+1,操作n+2,操作n+3.。

。。

。)。可是不一定是ai的操作序列全在前面的。至于为什是组合不是排列

                由于每一个ai和bj自己内部已经是有序了的。感觉这样讲的挺清楚了吧。自己比赛的时候这里卡了挺就久的。可是细致想一下。还是能够明确的

                */

                   sum[l][r][k]=(sum[l][r][k]*C[r-l-1][k-l])%MOD;

                   sum[l][r][n+1]=(sum[l][r][n+1]+sum[l][r][k])%MOD;

                   sum[l][r][n+1]=(sum[l][r][n+1]+MOD)%MOD;

            }

        }

        printf("%I64d\n",sum[1][n][n+1]);

}

int main(){

#ifndef ONLINE_JUDGE

    freopen("aaa","r",stdin);

#endif

    int T;

    int q;

    A[0]=1;/*A是全排列 C是组合 ,预处理这两个*/

    for(int i=1;i<N;i++) A[i]=(A[i-1]*i)%MOD;

    

    for(int i=0;i<N;i++) C[i][0]=1;

    for(int i=1;i<N;i++) for(int j=0;j<=i;j++) {

        if(j==0 || j==i) C[i][j]=1;

        else C[i][j]=(C[i-1][j-1]+C[i-1][j])%MOD;

    }

    while(~scanf("%d",&n)){

        for(int i=1;i<=n;i++) scanf("%d",&a[i]);

        scanf("%s",ch+1);

        solve();

    }

    return 0;

}

hdu5396 Expression 区间dp +排列组合的更多相关文章

  1. HDU 5151 Sit sit sit 区间DP + 排列组合

    Sit sit sit 问题描述 在一个XX大学中有NN张椅子排成一排,椅子上都没有人,每张椅子都有颜色,分别为蓝色或者红色. 接下来依次来了NN个学生,标号依次为1,2,3,...,N. 对于每个学 ...

  2. [hdu5396 Expression]区间DP

    题意:给一个表达式,求所有的计算顺序产生的结果总和 思路:比较明显的区间dp,令dp[l][r]为闭区间[l,r]的所有可能的结果和,考虑最后一个符号的位置k,k必须在l,r之间,则l≤k<r, ...

  3. 【BZOJ】2111: [ZJOI2010]Perm 排列计数 计数DP+排列组合+lucas

    [题目]BZOJ 2111 [题意]求有多少1~n的排列,满足\(A_i>A_{\frac{i}{2}}\),输出对p取模的结果.\(n \leq 10^6,p \leq 10^9\),p是素数 ...

  4. 【BZOJ】4559: [JLoi2016]成绩比较 计数DP+排列组合+拉格朗日插值

    [题意]n位同学(其中一位是B神),m门必修课,每门必修课的分数是[1,Ui].B神碾压了k位同学(所有课分数<=B神),且第x门课有rx-1位同学的分数高于B神,求满足条件的分数情况数.当有一 ...

  5. G.subsequence 1(dp + 排列组合)

    subsequence 1 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言524288K 64bit IO Format: %lld 题目描述 You are ...

  6. hdu 5396 Expression(区间dp)

    Problem Description Teacher Mai has n numbers a1,a2,⋯,anand n−1 operators("+", "-&quo ...

  7. LightOJ1005 Rooks(DP/排列组合)

    题目是在n*n的棋盘上放k个车使其不互相攻击的方案数. 首先可以明确的是n*n最多只能合法地放n个车,即每一行都指派一个列去放车. dp[i][j]表示棋盘前i行总共放了j个车的方案数 dp[0][0 ...

  8. HDU 5816 状压DP&排列组合

    ---恢复内容开始--- Hearthstone Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java ...

  9. bzoj 3398 [Usaco2009 Feb]Bullcow 牡牛和牝牛——前缀和优化dp / 排列组合

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3398 好简单呀.而且是自己想出来的. dp[ i ]表示最后一个牡牛在 i 的方案数. 当前 ...

随机推荐

  1. 为什么字符串类型可以调用构造函数String的方法,却又不是它的实例

    从所周知,在js中定义一个字符串我们有两种办法: var a = new String("a"); var a = "a"; 第一种方法使用构造函数创建,作为S ...

  2. 08使用NanoPiM1Plus在Android4.4.2下接TF卡

    08使用NanoPiM1Plus在Android4.4.2下接TF卡 大文实验室/大文哥 壹捌陆捌零陆捌捌陆捌贰 21504965 AT qq.com 完成时间:2017/12/5 17:51 版本: ...

  3. Microsoft SQL Server学习(五)--操作符聚合函数

    算术运算符 逻辑运算符 比较运算符 聚合函数 算术运算符(+ - * / ) select score*2 as 成绩翻倍 from class_A update class_A set score= ...

  4. outlook 2016 接收发送无法及时收下邮件,如何更改接收时间?

    1. 单击“文件” > “选项” > “高级” > “发送和接收”,单击”发送/接收“ 2. 组“所有账户”的设置 > 打勾“安排自动发送/接收的时间间隔为(V)” 1 分钟 ...

  5. vuex理解之modules小记

    好记性不如烂笔头 demo预览 源代码 前情提要 关于vuex,其实很久以前就研究使用过,还研究过 flux,redux之类的体系,当时感觉对于 state,action,dispatch,views ...

  6. 网络编程基础_4.1TCP_服务端

    TCP_服务端 #include <stdio.h> // 1. 包含必要的头文件和库, 必须位于 windows之前 #include <WinSock2.h> #pragm ...

  7. SpringBoot中如何使用jpa和jpa的相关知识总结

    jpa常用的注解: 注解 解释 @Entity 声明类为实体或表. @Table 声明表名. @Basic 指定非约束明确的各个字段. @Embedded 指定类或它的值是一个可嵌入的类的实例的实体的 ...

  8. CAD二次开发控件,dwg控件,网页DWG控件,手机浏览编辑DWG控件

    梦想绘图插件5.2(MxDraw5.2) 是国内最强,最专业的CAD开发插件(控件),不需要AutoCAD就能独立运行. 控件使用VC 2010开发,具有30万行代码规模,最早从2007年第一个版本完 ...

  9. Pytorch 加载保存模型【直播】2019 年县域农业大脑AI挑战赛---(三)保存结果

    在模型训练结束,结束后,通常是一个分割模型,输入 1024x1024 输出 4x1024x1024. 一种方法就是将整个图切块,然后每张预测,但是有个不好处就是可能在边界处断续. 由于这种切块再预测很 ...

  10. Linux kernel 内存 - 页表映射(SHIFT,SIZE,MASK)和转换(32位,64位)

    0. Intro 如下是在32位下的情况,32位下,只有三级页表:PGD,PMD,PTE 在64位情况下,会有四级页表:PGD,PUD,PMD,PTE 但是原理基本上是一样的,本文主要是想记录一下页表 ...