Description

题目描述

Recently, you have found your interest in string theory. Here is an interesting question about strings.

You are given a string S of length n consisting of the first k lowercase letters.

You are required to find two non-empty substrings (note that substrings must be consecutive) of S, such that the two substrings don't share any same letter. Here comes the question, what is the maximum product of the two substring lengths?

最近,你突然迷上了字串论。下面是一个关于字符串的有趣问题。

给定一个字符串S,长度为n且包含前k个小写字母。

你被要求从S中找出两个非空子串(注意,子串必须是连续的),并且两个子串不共享任何字母。问题来了,两个子串长度乘积的最大值是多少?

Input

输入

The first line contains an integer T, meaning the number of the cases. 1 <= T <= 50.

For each test case, the first line consists of two integers n and k. (1 <= n <= 2000, 1 <= k <= 16).

The second line is a string of length n, consisting only the first k lowercase letters in the alphabet. For example, when k = 3, it consists of a, b, and c.

输入的首行是一个整数T,表示测试样例的数量。1 <= T <= 50。

对于每个测试样例,第一行有2个整数n和k。(1 <= n <= 2000, 1 <= k <= 16)。

第二行有有一个长度为n的字符串,仅由字母表前k个小写字母组成。比如,k=3,组成元素为a,b,和c。

Output

输出

For each test case, output the answer of the question.

对于每个测试样例,输出问题的答案。

Sample Input - 输入样例

Sample Output - 输出样例

4

25 5

abcdeabcdeabcdeabcdeabcde

25 5

aaaaabbbbbcccccdddddeeeee

25 5

adcbadcbedbadedcbacbcadbc

3 2

aaa

6

150

21

0

Hint

提示

One possible option for the two chosen substrings for the first sample is "abc" and "de".

The two chosen substrings for the third sample are "ded" and "cbacbca".

In the fourth sample, we can't choose such two non-empty substrings, so the answer is 0.

对于第一个例子,其中一种可行的子串选择方案为"abc"和"de"。

第三个例子选择的子串为"ded"和"cbacbca"。

第四个例子中,我们无法找出两个非空子串,所以答案为0。

【题解】

解题思路:枚举,贪心/搜索,状态压缩/转移。

代码构架:枚举,状态压缩/转移,贪心。

枚举:

  因为最多只有前16个字母,一共216种状态,可以使用二进制压缩成一个int来记录每种字母出现的状态(当然你要用unsigned short我也不栏你_(:3 」∠)_,不过int明显看着舒服敲着方便。某人:你再说就要离题了啊,喂!)

  枚举的时候可以指定开头再叠加状态,也可以按01背包来写,记录下每种状态子串长度的最大值。

贪心/搜索:

  假如我们现在选了0001状态的子串,那么秉持着不拿人民一针一线(误)的优(hen)良(tai)作风,只要最后面不是1的我都要了!

  因此得到在可选情况下最好的状态1110。

那么问题来了,学挖掘机……哦不,1110的字符串长度可能是0啊,我们要找的可能是1100,1010,***0。那么把***0中的最大长度丢到1110中,就可以得到1110(其实是***0)的最大长度。

使用位运算可以很方便地完成比较。

状态压缩/转移:

大道无言,百态汇流。

【代码 C++】

  1. #include<cstdio>
  2. #include<cstring>
  3. #include<algorithm>
  4. int n, k, bag[ << ], bit[];
  5. int fid(){
  6. memset(bag, , sizeof(bag));
  7. char data[];
  8. gets(data);
  9. int i, len, temp, ed;
  10. for (i = ; i < n; i += ){//枚举
  11. --i;
  12. for (len = , temp = ; i + len < n; ++len){
  13. temp |= bit[data[i + len] - 'a'];
  14. bag[temp] = std::max(bag[temp], len);
  15. }
  16. }
  17. for (i = , ed = bit[k]; i < ed; ++i){//压缩状态
  18. for (temp = ; temp <= k; ++temp){
  19. if (i & bit[temp]) bag[i] = std::max(bag[i], bag[i^bit[temp]]);
  20. }
  21. }
  22. for (i = , len = , --ed; i < ed; ++i){//搜索
  23. len = std::max(len, bag[i] * bag[i^ed]);
  24. }
  25. return len;
  26. }
  27. int main(){
  28. for (int i = , j = ; i < ; ++i, j <<= ) bit[i] = j;
  29. int T;
  30. scanf("%d", &T);
  31. while (T--){
  32. scanf("%d%d", &n, &k), getchar();
  33. printf("%d\n", fid());
  34. }
  35. return ;
  36. }
  1. #include<cstdio>
  2. #include<cstring>
  3. short bag[];
  4. int n, k, letter[];
  5. int fid(){
  6. memset(bag, , );
  7. char data[];
  8. gets(data);
  9. int i, len, temp, ed = << k;
  10. --ed;
  11. for (i = ; i < n; i += ){
  12. --i;
  13. for (len = , temp = ; i + len < n && temp^ed; ++len){
  14. temp |= letter[data[i + len] - 'a'];
  15. if (bag[temp] < len) bag[temp] = len;
  16. }
  17. }
  18. for (i = ; i <= ed; ++i){
  19. for (temp = ; temp <= k; ++temp)
  20. if (i & letter[temp] && bag[i] < bag[i^letter[temp]])
  21. bag[i] = bag[i^letter[temp]];
  22. }
  23. for (i = , len = ; i < ed; ++i){
  24. temp = bag[i] * bag[i^ed];
  25. if (len < temp) len = temp;
  26. }
  27. return len;
  28. }
  29. int main(){
  30. for (n = , k = ; n < ; ++n, k <<= ) letter[n] = k;
  31. int T;
  32. scanf("%d", &T);
  33. while (T--){
  34. scanf("%d%d", &n, &k), getchar();
  35. printf("%d\n", fid());
  36. }
  37. return ;
  38. }

FZU 2218

FZU 2218 Simple String Problem(简单字符串问题)的更多相关文章

  1. FZU - 2218 Simple String Problem(状压dp)

    Simple String Problem Recently, you have found your interest in string theory. Here is an interestin ...

  2. FZU - 2218 Simple String Problem 状压dp

    FZU - 2218Simple String Problem 题目大意:给一个长度为n含有k个不同字母的串,从中挑选出两个连续的子串,要求两个子串中含有不同的字符,问这样的两个子串长度乘积最大是多少 ...

  3. (比赛)A - Simple String Problem

    A - Simple String Problem Time Limit:10000MS     Memory Limit:65536KB     64bit IO Format:%lld & ...

  4. hdu------(1757)A Simple Math Problem(简单矩阵快速幂)

    A Simple Math Problem Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Ot ...

  5. FZU 2215 Simple Polynomial Problem(简单多项式问题)

    Description 题目描述 You are given an polynomial of x consisting of only addition marks, multiplication ...

  6. fzu2218 Simple String Problem

    Accept: 2    Submit: 16 Time Limit: 2000 mSec    Memory Limit : 32768 KB  Problem Description Recent ...

  7. O - String Problem KMP 字符串最小表示法

    Give you a string with length N, you can generate N strings by left shifts. For example let consider ...

  8. HDU 5284 wyh2000 and a string problem(字符串,水)

    题意:比如给你一个串,要求判断wyh是不是它的子序列,那么你只需要找一个w,找一个y,再找一个h,使得w在y前面,y在h前面即可.有一天小学生拿着一个串问他“wyh是不是这个串的子序列?”.但是wyh ...

  9. FZU2218 Simple String Problem(状压DP)

    首先,定义S,表示前k个字符出现的集合,用二进制来压缩. 接下来,推出dp1[S],表示集合为S的子串的最长长度. 然后根据dp1[S]再推出dp2[S],表示集合为S或S的子集的子串的最长长度. 最 ...

随机推荐

  1. java总结第二次(剩余内容)//类和对象1

    7.成员变量和局部变量 成员变量:在类中定义,用来描述对象将要有什么 局部变量:在类的方法中定义,在方法中保存临时数据 区别:作用域不同 局部变量的作用域仅限于定义它的方法 成员变量的作用域在整个类内 ...

  2. NXP QN9020

    NXP的这个BLE蓝牙方案也很有趣, 一起研究. 这个函数在app_gpa_task.c里面 ***************************************************** ...

  3. SuperSocket架构设计示意图【转】

    转自:http://docs.supersocket.net/v1-6/zh-CN/Architecture-Diagrams 中文(中国)Toggle Dropdown v1.6Toggle Dro ...

  4. centos6.5-64bit安装htop

    首先启用 EPEL Repository: yum -y install epel-release 启用 EPEL Repository 後, 可以用 yum 直接安裝 Htop: yum -y in ...

  5. Java 使用 Redis | 菜鸟教程

    入门教程: http://www.runoob.com/redis/redis-java.html 中文手册: http://redis.readthedocs.io/en/2.4/index.htm ...

  6. ArrayList集合的实现原理

    一. ArrayList概述: ArrayList是基于数组实现的,是一个动态数组,其容量能自动增长,类似于C语言中的动态申请内存,动态增长内存. ArrayList不是线程安全的,只能用在单线程环境 ...

  7. [C++][重载]

    运算符重载 C++中预定义的运算符的操作对象只能是基本数据类型,实际上,对于很多用户自定义类型,也需要有类似的运算操作.例如: class complex {  public:   complex(d ...

  8. 每日一九度之 题目1030:毕业bg

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:2046 解决:894 题目描述:     每年毕业的季节都会有大量毕业生发起狂欢,好朋友们相约吃散伙饭,网络上称为“bg”.参加不同团体的b ...

  9. 测试-关于Unity获取子层级内容的几种接口(Transform FindChild, Component GetComponentInChildren,...)

    测试常用的层级内组件查找接口,但一些需求还是需要扩展 比如按照名称批量查找节点,查找接口对象等 1.Transform - Transform Find(string name) 可以直接根据名称搜索 ...

  10. java 反编译和文档生成器

    挺有趣的东西. 代码在上一篇中有. 反编译器javap.exe javadoc.exe制作源文件类结构的html格式文档