Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
samples.docx
Скачиваний:
3
Добавлен:
09.11.2018
Размер:
118.23 Кб
Скачать

Задача 11: в строке все слова, которые начинаются и заканчиваются одной буквой, выделить квадратными скобками.

Пример: из строки «шалаш широкий и просторный» должно получиться «[шалаш] широкий [и] просторный».

Размышления:

  1. Очевидно, что задача представима в виде следующей последовательности действий:

    1. Ввод данных

    2. Обработка данных

    3. Вывод результата

  2. Так же логично предположить, что пользователь вводит строку с клавиатуры. Т.е. в качестве входных данных у нас есть строка.

  3. Из предыдущего примера ясно, что, если мы научимся находить начало слова и его длину, это решит проблему нахождения слова.

    1. Операции нахождения начала и длинны слова так же полностью аналогичны тем, которые были в предыдущем примере.

    2. Теперь мы умеем находить слово (потому что умеем находить начало слова и его длину). Поэтому вернемся к изначальному заданию: выделить слова, которые начинаются и заканчиваются одной и той же буквой. Очевидно, что выделение слова происходит по условию. Возникает вопрос – как реализовать это условие? Если у нас есть слово в виде строки, то первый символ в нем имеет индекс 0, а последний – Length - 1. Поэтому в виде кода условие необходимости выделения слова будет выглядеть следующим образом:

public static bool needMarkWord(string word) {

if (word[0] == word[word.Length - 1]) {

return true;

} else {

return false;

}

}

    1. Как видно из этого кода, нам нужно получить слово в виде строки. Получение слова из строки полностью аналогично примеру из предыдущей задачи.

    2. Что бы верно реализовать алгоритм, вспомним, что по сути строка является одномерным массивом символов. Поэтому вставка в нее ведет к изменению индексации (в данном случае в случае выделения слова вставляются квадратные скобки). Поэтому очевидно, что в результате обработки строки строка будет изменяться. И для обработки строки необходима только сама строка. Поэтому в функции обработки будет только один строковый параметр с ключевым словом ref.

    3. Псевдокодовый алгоритм решения задачи будет выглядеть следующим образом:

Для каждого символа строки

НачалоЦикла

Если текущийСимвол – начало слова, то

НачалоЕсли

Объявить длину слова

Найти длину слова, которое начинается в текущем символе

Объявить переменную, в которой будет храниться текущее слово

Получить слово в виде строки.

Если слово необходимо выделить, то

НачалоЕсли

Удалить слово

Выделить слово

Вставить выделенное слово на позицию старого

Увеличить текущую позицию на длину выделенного слова

ОкончаниеЕсли

ОкончаниеЕсли

ОкончаниеЦикла

    1. Обратите внимание на увеличение позиции после вставки слова. Это принципиально важно. Для того, что бы это продемонстрировать, рассмотрим предложение «я». Первая буква (с индексом 0) является началом слова, которое надо выделить. После выделения получится «[я]». При этом позиция текущего символа была равна 0. После выделения мы переместимся на символ с индексом 1. «я» является словом, которое надо выделить, поэтому после выделения получится «[[я]]». Как видно, без увеличения текущей позиции в данном случае алгоритм «зациклится».

3.7. Оптимизации кода так же аналогичны оптимизациям в предыдущей задаче. Приведем полный код решения:

using System;

namespace ConsoleApplication17 {

class Program {

public static bool needMarkWord(string word) {

if (word[0] == word[word.Length - 1]) {

return true;

} else {

return false;

}

}

private static bool isSeparator(char c) {

string oneCharStr = "" + c;

return " .,?!".Contains(oneCharStr);

}

public static bool isBeginOfWord(string str, int index) {

if (isSeparator(str[index])) {

return false;

}

if (index == 0) {

return true;

}

if (isSeparator(str[index - 1])) {

return true;

} else {

return false;

}

}

public static int getWordLength(string str, int index) {

int res = 0;

for (int i = index; i < str.Length; i++) {

if (isSeparator(str[i])) {

break;

}

res = res + 1;

}

return res;

}

public static void processString(ref string str) {

for (int i = 0; i < str.Length; i++) {

if (isBeginOfWord(str, i) == false) {

continue;

}

int len = getWordLength(str, i);

string word = str.Substring(i, len);

if (needMarkWord(word)) {

str = str.Remove(i, len);

word = "[" + word + "]";

str = str.Insert(i, word);

i = i + 2 + len;

} else {

i = i + len;

}

}

}

static void Main(string[] args) {

string inputStr = Console.ReadLine();

Console.WriteLine("Строка до обработки: " + inputStr);

processString(ref inputStr);

Console.WriteLine("Строка после обработки: " + inputStr);

Console.ReadKey();

}

}

}

Соседние файлы в предмете [НЕСОРТИРОВАННОЕ]