题意:
考拉有两个长度相同的字符串A和B(|A|=|B|=n),由前20个小写英文字母组成(即从a到t)。
在一步棋中,Koa。
(选择A的某个位置子集p1,p2,...,pk(k≥1;1≤pi≤n;pi≠pj如果i≠j),如果Ap1=Ap2=...=Apk=x(即这个位置上的所有字母都等于某个字母x)。
从英语字母表的前20个小写字母中选择一个字母y,使y>x(即字母y在字母表上严格大于x)。
将位置p1,p2,...,pk中的每个字母设置为字母y。更正式地说:对于每个i(1≤i≤k)Koa设置Api=y。
注意,你只能修改字符串A中的字母。)
Koa想知道她为使字符串相互相等(A=B)或确定没有办法使它们相等所要做的最小的移动次数。请帮助她!
输入
每个测试包含多个测试用例。第一行包含t(1≤t≤10)--测试用例的数量。测试用例的描述如下。
每个测试用例的第一行包含一个整数n (1≤n≤105) - 字符串A和B的长度。
每个测试用例的第二行包含字符串A(|A|=n)。
每个测试用例的第三行包含字符串B(|B|=n)。
这两个字符串由前20个小写英文字母组成(即从a到t)。
保证所有测试用例的n之和不超过105。
题解:
a只能变大,很明显,如果某个ai大于bi是一定无解的
那对于有解的情况
拿例一来说吧
aab
bcc
如果正常来想
a->b
a->c
b->c
三步
接着我们结合上面这个图来看
a->b
a->b->c
b->c
我们可以发现可以先让a到可以到的最小的字母b
如果b和a都要到一个更大的字母c
就相当于省去了步数
- #include<iostream>
- #include<algorithm>
- #include<map>
- #include<cstring>
- #include<vector>
- using namespace std;
- char a[100050],b[100050];
- int f[30][30];
- void solve()
- {
- int n;
- cin >> n;
- cin >> a+1>>b+1;
- memset(f,0,sizeof f);
- for(int i = 1;i <= n; i++)
- {
- if(a[i] > b[i])
- {
- cout<<"-1\n";
- return ;
- }
- if(a[i]<b[i])
- f[a[i]-'a'][b[i]-'a'] = 1;
- }
- int ans = 0;
- for(int i = 0;i < 20;i++)
- {
- for(int j = i+1;j < 20;j++)
- {
- if(f[i][j])
- {
- ans++;
- for(int k = j+1;k < 20;k++)
- {
- if(f[i][k])
- {
- f[j][k] = 1;
- }
- }
- break;
- }
- }
- }
- cout<<ans<<"\n";
-
- }
- int main()
- {
- int t;
- cin >> t;
- while(t--)
- {
- solve();
- }
-
- }