D. String Mark
time limit per test

4 seconds

memory limit per test

256 megabytes

input

standard input

output

standard output

At the Byteland State University marks are strings of the same length. Mark x is considered better than y if string y is lexicographically smaller than x.

Recently at the BSU was an important test work on which Vasya recived the mark a. It is very hard for the teacher to remember the exact mark of every student, but he knows the mark b, such that every student recieved mark strictly smaller than b.

Vasya isn't satisfied with his mark so he decided to improve it. He can swap characters in the string corresponding to his mark as many times as he like. Now he want to know only the number of different ways to improve his mark so that his teacher didn't notice something suspicious.

More formally: you are given two strings ab of the same length and you need to figure out the number of different strings c such that:

1) c can be obtained from a by swapping some characters, in other words c is a permutation of a.

2) String a is lexicographically smaller than c.

3) String c is lexicographically smaller than b.

For two strings x and y of the same length it is true that x is lexicographically smaller than y if there exists such i, that x1 = y1, x2 = y2, ..., xi - 1 = yi - 1, xi < yi.

Since the answer can be very large, you need to find answer modulo 109 + 7.

Input

First line contains string a, second line contains string b. Strings a, b consist of lowercase English letters. Their lengths are equal and don't exceed 106.

It is guaranteed that a is lexicographically smaller than b.

Output

Print one integer  — the number of different strings satisfying the condition of the problem modulo 109 + 7.

Examples
input
abc
ddd
output
5
input
abcdef
abcdeg
output
0
input
abacaba
ubuduba
output
64
Note

In first sample from string abc can be obtained strings acb, bac, bca, cab, cba, all of them are larger than abc, but smaller than ddd. So the answer is 5.

In second sample any string obtained from abcdef is larger than abcdeg. So the answer is 0.

题目大意:给两个字符串a,b,重组a字符串得到c,使得c的字典序大于a小于b,问方案数.

分析:比较有难度,一开始没能想出来.

其实这道题有点像数位dp,有上下界嘛,求个数.但是不同的是要求的字符串需要是a重组得到的,每个字符的个数是一定的,也就是说在当前位放了一个数,可能会对后面的决策造成影响.于是就可以用记忆化搜索来解决.

设f[i][0/1][0/1]表示前i位中是否达到上界和是否达到下界的方案数.类似于数位dp,先处理既达到上界又达到下界的,如果当前处理到第i位并不是只能选一个数字的话,那么可以把这个过程拆成处理到达上界的部分和处理到达下界的部分.如果只能选一个数字,那么这个数字的个数-1,搜下去.再处理上下界的情况.将上下界的数字的个数分别-1,搜下去,这里要分别处理.

剩下的就是处理位于下界+1和上界-1之间的数了.枚举放哪一个,方案数就是排列数,只不过元素允许重复,需要用不全相异元素的排列公式来计算,即如果总共有n个数,第i个数有ni个,那么方案数为n! / (n1! * n2! * ...... * nn!).最后将所有转移的答案累加就是问题的答案了.

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <string>
#include <cmath> using namespace std; typedef long long ll; const int maxn = , mod = 1e9 + ;
ll f[maxn][][], len, a[maxn], b[maxn], num[maxn], jie[maxn];
char s1[maxn], s2[maxn]; ll qpow(ll a, ll b)
{
ll res = ;
while (b)
{
if (b & )
res = (res * a) % mod;
a = (a * a) % mod;
b >>= ;
}
return res;
} ll solve(ll cur, bool up, bool down)
{
ll ans = ;
if (f[cur][up][down] != -)
return f[cur][up][down];
if (up && down && a[cur] == b[cur] && num[a[cur]] > )
{
num[a[cur]]--;
ll temp = solve(cur + , up, down);
num[a[cur]]++;
return f[cur][up][down] = temp;
}
if (up && num[b[cur]])
{
num[b[cur]]--;
ans += solve(cur + , , );
ans %= mod;
num[b[cur]]++;
}
if (down && num[a[cur]])
{
num[a[cur]]--;
ans += solve(cur + , , );
ans %= mod;
num[a[cur]]++;
}
ll temp = ;
for (ll i = ; i <= ; i++)
temp = (temp * jie[num[i]]) % mod;
temp = qpow(temp, mod - );
int upx = (up ? b[cur] - : ), downx = (down ? a[cur] + : );
ll left = len - cur;
for (ll i = downx; i <= upx; i++)
{
if (num[i] > )
ans += jie[left] * temp % mod * num[i] % mod; //因为num[i]的数量要-1,所以分母的阶乘要减少num[i]
ans %= mod;
}
return f[cur][up][down] = ans;
} int main()
{
jie[] = ;
for (ll i = ; i <= ; i++)
jie[i] = (jie[i - ] * i) % mod;
scanf("%s", s1 + );
scanf("%s", s2 + );
len = strlen(s1 + );
for (ll i = ; i <= len; i++)
{
a[i] = s1[i] - 'a' + ;
num[a[i]]++;
b[i] = s2[i] - 'a' + ;
}
memset(f, -, sizeof(f));
cout << solve(,,) << endl; return ;
}

Codeforces 895.D String Mark的更多相关文章

  1. String Mark Codeforces - 895D

    一看好像会做的样子,就去做了一下,结果 猝不及防地T掉了 赶紧查了一下,没有死循环,复杂度也是对的,无果,于是翻了题解 题解没看懂,但是找到了标程,然后发现我被卡常了... 而且好像当时还过了前10个 ...

  2. CodeForces 797C Minimal string:贪心+模拟

    题目链接:http://codeforces.com/problemset/problem/797/C 题意: 给你一个非空字符串s,空字符串t和u.有两种操作:(1)把s的首字符取出并添加到t的末尾 ...

  3. Codeforces 827E Rusty String - 快速傅里叶变换 - 暴力

    Grigory loves strings. Recently he found a metal strip on a loft. The strip had length n and consist ...

  4. Codeforces 797C - Minimal string

    C. Minimal string 题目链接:http://codeforces.com/problemset/problem/797/C time limit per test 1 second m ...

  5. codeforces 825F F. String Compression dp+kmp找字符串的最小循环节

    /** 题目:F. String Compression 链接:http://codeforces.com/problemset/problem/825/F 题意:压缩字符串后求最小长度. 思路: d ...

  6. Codeforces - 1117E - Crisp String - 进制 - 交互

    https://codeforces.com/problemset/problem/1117/E 就用abc表示数字来给每个数编码,编完直接问出移动的结果,反构造就行了,比C和D还简单. #inclu ...

  7. codeforces 828 C. String Reconstruction(思维+优先队列)

    题目链接:http://codeforces.com/contest/828/problem/C 题解:有点意思的题目,可用优先队列解决一下具体看代码理解.或者用并查集或者用线段树都行. #inclu ...

  8. codeforces 779 D. String Game(二分)

    题目链接:http://codeforces.com/contest/779/problem/D 题意:给你一段操作序列,按顺序依次删掉字符串1中相应位置的字符,问你最多能按顺序删掉多少个字符,使得s ...

  9. Codeforces 494B Obsessive String

    http://www.codeforces.com/problemset/problem/494/B 题意:给出两个串S,T,求有几种将S分成若干个子串,满足T都是这若干个子串的子串. 思路:f[n] ...

随机推荐

  1. 239. [LeetCode ]Sliding Window Maximum

    Given an array nums, there is a sliding window of size k which is moving from the very left of the a ...

  2. Windows下使用WinRAR命令自动备份文件

    最近有一个需求:为了防止数据丢失,每天对固定文件夹下的文件进行打包压缩备份. 解决办法:使用Windows的任务计划程序,每天执行一下压缩命令: Windows任务计划程序在这里就不再介绍了,网上有很 ...

  3. [leetcode-915-Partition Array into Disjoint Intervals]

    Given an array A, partition it into two (contiguous) subarrays left and right so that: Every element ...

  4. Beta周王者荣耀交流协会第三次Scrum会议

    1.立会照片 成员王超,高远博,冉华,王磊,王玉玲,任思佳,袁玥全部到齐. master:王玉玲 2.时间跨度: 2017年11月12日 18:00 — 18:20 ,总计20分钟. 3.地点: 一食 ...

  5. 王者荣耀交流协会第四次Scrum立会

    会议时间:2017年10月23号   18:00-18:28,时长28分钟. 会议地点:二食堂一楼第四个档口对着的靠路边的桌子. 立会内容: 1.小组成员汇报今日工作: 2.关于折线图与饼状图生成问题 ...

  6. c# Application.run和form.show区别

    Application.run(form):在当前线程上开始运行标准应用程序消息循环,并使指定窗体可见. form.show() :使指定窗体可见: 参照:https://blog.csdn.net/ ...

  7. 线段树---成段更新hdu1698 Just a Hook

    hdu1698 Just a Hook 题意:O(-1) 思路:O(-1) 线段树功能:update:成段替换 (由于只query一次总区间,所以可以直接输出1结点的信息) 题意:给一组棍子染色,不同 ...

  8. 第九章 Mysql函数

    简介 数学函数:处理数字 字符串函数:处理字符串 日期和时间函数:处理日期和时间,获取时间 条件判断函数:控制条件选择 系统信息函数:获取MySQL系统信息,包括数据库名称,当前用户名和数据库版本 加 ...

  9. 使用git下载编译erlang

    git clone https://github.com/erlang/otp cd otp git tag git checkout -b OTP- OTP- ./otp_build all exp ...

  10. spring表单—乱码解决方案

    1.POST方式提交 A.jsp页面POST方式提交表单 B.web.xml配置过滤器 <!-- spring过滤器解决乱码问题 --> <filter> <filter ...