Задача
После того как мама запретила Витэку заниматься неизвестным языком и забрала у него все кубики, не относящиеся к латинскому алфавиту, он нашел для себя новое занятие на детской железной дороге. Для начала он построил несколько депо, куда мог отправлять лишние вагончики, правда забирать их оттуда он не научился. И вот, имея некий запас кубиков с большими буквами латинского алфавита, он решил тайком от мамы заняться изучением английского языка. Но так как даже словарем с не родным английским языком мама запретила ему пользоваться, Витэк решил составить свой словарь.
В словарь он вначале заносил слово, образованное из начального расположения кубиков с буквами на вагончиках, а далее новые образовывал путем отцепления некоторого количества букв либо в начале состава, либо в конце, либо с обеих сторон, каждый раз используя за начальное первичное расположение кубиков.
Сколько всего разных слов может быть в таком необычном «английском» словаре Витэка?
Тесты
Входные данные
|
Выходные данные |
ALPHABET | 35 |
AAAA | 4 |
UNIVERSITY | 54 |
ABABABABC | 24 |
Код на C++
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 |
#include <iostream> #include <string> #include <vector> #include <algorithm> using namespace std; int main() { string a; vector <string> c; vector<string> :: iterator it;// задаем итератор cin>>a; for(int i=0;i<a.size();i++) { for(int j=1;j<a.size();j++) { c.push_back(a.substr(i,j)); } } sort(c.begin(), c.end()); // сортируем вектор для функции unique it = unique(c.begin(), c.end()); // задаем итератору значение функции unigue c.resize(it - c.begin());// производим resize вектора при помощи итератора cout<<c.size()+1; return 0; } |
Код на Java
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
import java.util.*; import java.lang.*; import java.io.*; public class Main { public static void main (String[] args) { Scanner in = new Scanner(System.in); Vector<String> c = new Vector<String>(); String a = in.nextLine(); for(int i=0;i<a.length();i++) { for(int j=i+1;j<=a.length();j++) { c.add(a.substring(i,j)); } } HashSet<String> unique = new HashSet<String>(c); Vector<String> d = new Vector<String>(unique); System.out.print(d.size()); } } |
Решение
Казалось бы, задачу можно решить простым циклом. Ведь в 6-буквенном слове: 6 однобуквенных слов, 5 двубуквенных и т.д. Однако очевидной становится проблема повторения,причем не только повторений букв, но и повторений подстрок, что изрядно усложняет наше решение. Создадим вектор, состоящий из строк, куда мы будем добавлять все подстроки строки из входного потока. Отсортируем полученный вектор. Это нужно для того чтобы одинаковые элементы шли рядом с друг другом. Тогда мы можем использовать функцию unique(), которая удалит из вектора все повторяющиеся элементы, кроме одного. Для того, чтобы провести resize() вектора нам понадобится итератор, который мы приравняем к функции unique(). После resize() останется только вывести размер вектора и прибавить к нему единицу, так как само слово не является подстрокой.