回 帖 发 新 帖 刷新版面

主题:计单词个数 (30分)

计单词个数 (30分)
[问题描述]
  给出一个长度不超过200的由小写英文字母组成的字符串(约定:该字串以每行20个字母的方式输入,且保证每行一定为20个)。要求将此字串分成k份(1<k≤40),且每份中包含的单词个数加起来总数最大(每份中包含的单词可以部分重叠。当选用一个单词之后,其第一个字母不能再用。例如字符串this中可以包含this和is,选用this之后就不能包含th)。
  单词在给出的一个不超过6个单词的字典中。
  要求输出最大的个数。
[输入格式]
  全部输入数据存放在文本文件input3.dat中,其格式如下:
  第一行为一个正整数(0<n≤5)表示有n组测试数据
  每组的第一行有二个正整数:(p,k)
    p表示字串的行数;
    k表示分为k个部分。
  接下来的p行,每行均有20个字符。
  再接下来有一个正整数s,表示字典中单词个数。(1≤s≤6)
  接下来的s行,每行均有一个单词。
[输出格式]
  结果输出至屏幕,每行一个整数,分别对应每组测试数据的相应结果。
[样例]
输入:1
   1 3
   thisisabookyouareaoh
   4
   is
   a
   ok
   sab
输出:     //说明:(不必输出)
   7     // this/isabookyoua/reaoh
             sab

回复列表 (共1个回复)

沙发

var
   list:array[0..200,0..3] of integer;
   word:array[1..6] of string;
   str,ss:string;
   n,t,s:integer;

function work(str:string):integer;
var
   tot,i,j,l:integer;
   st:string;
   used:array[1..200] of boolean;
begin
   tot:=0;
   fillchar(used,sizeof(used),false);
   for i:=1 to s do begin
      st:=str;
      l:=0;
      j:=pos(word[i],st);
      while j<>0 do begin
         if not used[l+j] then inc(tot);
         delete(st,1,j);
         used[l+j]:=true;
         l:=l+j;
         j:=pos(word[i],st);
         end;
      end;
   work:=tot;
end;

procedure solve;
var
   i,j,p,k,l,u:integer;
begin
   readln(p,k);
   str:='';
   for i:=1 to p do begin
      readln(ss);
      str:=str+ss;
      end;
   readln(s);
   for i:=1 to s do readln(word[i]);
   fillchar(list,sizeof(list),0);
   for i:=1 to 20*p do
      for j:=1 to k do
         for u:=j to i do begin
            l:=work(copy(str,u,i-u+1));
            if list[u-1,j-1]+l>list[i,j] then list[i,j]:=list[u-1,j-1]+l;
            end;
   writeln(list[20*p,k]);
end;

begin
   assign(input,'input3.dat'); reset(input);
   readln(n);
   for t:=1 to n do solve;
end.

我来回复

您尚未登录,请登录后再回复。点此登录或注册