Глава 24. Практически задачи за изпит по програмиране – тема 1

В тази тема...

В настоящата тема ще разгледаме условията и ще предложим решения на три задачи от примерен изпит по програмиране. При решаването им ще приложим на практика описаната методология в главата "Как да решаваме задачи по програмиране".

Съдържание

Мисловни карти


Задача 1: Извличане на текста от HTML документ

Даден е HTML файл с име Problem1.html. Да се напише програма, която отстранява от него всички HTML тагове и запазва само текста вътре в тях. Изходът да се изведе във файла Рroblem1.txt.     

Примерен входен файл Рroblem1.html:

<html>

<head><title>Welcome to our site!</title></head>

<body>

<center>

<img src="/en/img/logo.gif" width="130" height="70" alt="Logo">

<br><br><br>

<font size="-1"><a href="/index.html">Home</a>

<a href="/contacts.html">Contacts</a>

<a href="/about.html">About</a></font><p>

</center>

</body>

</html>

Примерен изходен файл Problem1.txt:

Welcome to our site!

Home

Contacts

About

Измисляне на идея за решение

Първото, което ни хрумва като идея за решение на тази задача, е да четем последователно (примерно ред по ред) входния файл и да махаме всички тагове. Лесно се вижда, че всички тагове започват със символа "<" и завършват със символа ">". Това се отнася и за отварящите и за затварящите тагове. Това означава, че от всеки ред във файла трябва да се премахнат всички поднизове, започващи с "<" и завършващи с ">".

Проверка на идеята

Имаме идея за решаване на задачата. Дали идеята е вярна? Първо трябва да я проверим. Можем да я проверим дали е вярна за примерния входен файл, а след това да помислим дали няма някакви специални случаи, за които идеята би могла да е некоректна.

Взимаме лист и химикал и проверяваме на ръка идеята дали е вярна. Задраскваме всички поднизове от текста, които започват със символа "<" и завършват със символа ">". Като го направим, виждаме, че остава само чистият текст и всички тагове изчезват:

<html>

<head><title>Welcome to our site!</title></head>

<body>

<center>

<img src="/en/img/logo.gif" width="130" height="70" alt="Logo">

<br><br><br>

<font size="-1"><a href="/index.html">Home</a>

<a href="/contacts.html">Contacts</a>

<a href="/about.html">About</a></font><p>

</center>

</body>

</html>

Сега остава да измислим някакви по-специални случаи. Нали не искаме да напишем 200 реда код и чак тогава да се сетим за тях и да трябва да преправяме цялата програма? Затова е важно да проверим проблемните ситуации, още сега, преди да сме почнали да пишем кода на решението.

Можем да се сетим за следния специален пример:

<html><body>

Click<a href="info.html">on this

link</a>for more info.<br />

This is<b>bold</b>text.

</body></html>

В него има две особености:

-    Има тагове, съдържащи текст, които се отварят и затварят на различни редове. Такива тагове в нашия пример са <html>, <body> и <a>.

-    Има тагове, които съдържат текст и други тагове в себе си. Например  <body> и <html>.

Какъв трябва да е резултатът за този пример? Ако директно махнем всички тагове, ще получим нещо такова:

Clickon this

linkfor more info.

This isboldtext.

Или може би трябва да следваме правилата на езика HTML и да получим следния текст:

Click on this link for more info.

This is bold text.

Има и други варианти, например да слагаме всяко парче текст, което не е таг, на нов ред:

Click

on this

link

for more info.

This is

bold

text.

Ако махнем всичкия текст в таговете и долепим останалия текст, ще получим думи, които са залепени една до друга. От условието на задачата не става ясно дали това е исканият резултат или трябва, както в езика HTML, да получим по един интервал между отделните тагове. В езика HTML всяка поредица от разделители (интервали, нов ред, табулации и др.) се визуализира като един интервал. Това, обаче, не е споменато в условието на задачата и не става ясно от примерния вход и изход.

Не става ясно още дали трябва да отпечатваме думите, които са в таг, съдържащ в себе си други тагове или да ги пропускаме. Ако отпечатваме само съдържанието на тагове, в които има единствено текст, ще получим нещо такова:

on this

link

bold

От условието не става ясно още как се визуализира текст, който е разположен на няколко реда във вътрешността на някой таг.

Изясняване на условието на задачата

Първото, което трябва да направим, когато открием неясен момент в условието на задачата, е да го прочетем внимателно. В случая условието наистина не е ясно и не ни дава отговор на въпросите. Най-вероятно не трябва да следваме HTML правилата, защото те не са описани в усло­вието, но не става ясно дали долепяме думите в съседни тагове или да ги разделяме с нов ред.

Остава ни само едно: да питаме. Ако сме на изпит, ще питаме този, който ни е дал задачите. Ако сме в реалния живот, то все някой е поръчител на софтуера, който разработваме, и той би могъл да отговори на възникналите въпроси. Ако никой не може да отговори, избираме един от вариантите, който ни се струва най-правилен съгласно условието на задачата и действаме по него.

Приемаме, че трябва да се отпечата текста, който остава като премахнем всички отварящи и затварящи тагове, като използваме за разделител празен ред. Ако в текста има празни редове, запазваме ги. За нашия пример трябва да получим следния изход:

Click

on this

link

for more info.

This is

bold

text.

Нова идея за решаване на задачата

И така, нашата адаптирана към новите изисквания идея е следната: четем файла ред по ред и във всеки ред заместваме таговете с нов ред. За да избегнем дублирането на нови редове в резултатния файл, заместваме всеки два последова­телни нови реда от резултата с един нов ред.

Проверяваме новата идея с оригиналния пример от условието на задачата и с нашия пример и се убеждаваме, че идеята този път е вярна. Остава да я реализираме.

Разбиваме задачата на подзадачи

Задачата лесно можем да разбием на подзадачи:

-    Прочитане на входния файл.

-    Обработка на един ред от входния файл: заместване на таговете със символ за нов ред.

-    Записване на резултата в изходния файл.

Какво структури от данни да използваме?

В тази задача трябва да извършваме проста текстообработка и работа с файлове. Въпросът какви структури от данни да ползваме не стои пред нас – за текстообработката ще ползваме string и ако се наложи – StringBuilder.

Да помислим за ефективността

Ако четем редовете един по един, това няма да е бавна операция. Самата обработка на един ред може да се извърши чрез заместване на символи с други – също бърза операция. Не би трябвало да имаме проблеми с производителността.

Може би проблеми ще създаде изчистването на празните редове. Ако събираме всички редове в някакъв буфер (StringBuilder) и след това премахваме двойните празни редове, този буфер ще заеме много памет при големи входни файлове (примерно при 500 MB входен файл).

За да спестим памет, ще се опитаме да чистим излишните празни редове още след заместване на таговете със символа за празен ред.

Вече разгледахме внимателно идеята за решаване на задачата, уверихме се, че е добра и покрива специалните случаи, които могат да възникнат, и смятаме, че няма да имаме проблеми с производителността. Сега вече можем спокойно да преминем към имплементация на алгоритъма. Ще пишем кода стъпка по стъпка, за да откриваме грешките възможно най-рано.

Стъпка 1 – прочитане на входния файл

Първата стъпка от решението на поставената задача е прочитането входния файл. В нашия случай той е HTML файл. Това не трябва да ни притеснява, тъй като HTML е текстов формат. Затова, за да го прочетем, ще използваме класа StreamReader. Ще обходим входния файл ред по ред и за всеки ред ще извличаме (засега не ни интересува как) нужната ни информация (ако има) и ще я записваме в обект от тип StringBuilder. Извли­чането ще реализираме в следващата стъпка (стъпка 2), а записването в някоя от по-следващите стъпки. Да напишем нужния код за реализацията на нашата първа стъпка:

string line = string.Empty;

StreamReader reader = new StreamReader("Problem1.html");

 

while ((line = reader.ReadLine()) != null)

{

      // Find what we need and save it in the result

}

 

reader.Close();

Чрез написания код ще прочетем входния файл ред по ред. Да помислим дали сме реализирали добре първата стъпка. Сещате ли се какво пропус­нахме?

С написаното ще прочетем входния файл, но само ако съществува. Ами ако входния файл не съществува или не може да бъде отворен по някаква причина? Сегашното ни решение няма да се справи с тези проблеми. В кода има и още един проблем: ако настъпи грешка при четенето или обработката на данните от файла, той няма да бъде затворен.

Чрез File.Еxists(…) ще проверяваме дали входния файл съществува. Ако не съществува ще извеждаме подходящо съобщение и ще прекратяваме изпълнението на програмата. За да избегнем втория проблем ще изпол­зваме конструкцията try-catch-finally. Така, ако възникне изклю­чение ще го обработим и накрая винаги ще затваряме файла, с които сме работили. Не трябва да забравяме, че обекта от StreamReader трябва да е деклариран извън try блока, защото иначе ще е недостъпен във finally блока. Това не е фатална грешка, но често се допуска от начинаещите програмисти.

Добре е да дефинираме името на входния файл като константа, защото вероятно ще го използваме на няколко места.

Още нещо: при четене от текстов файл е редно да зададем кодирането на файла. В случая ще използваме кодиране windows-1251, тъй като искаме да поддържаме коректно работа с български уеб сайтове на кирилица.

Да видим до какво стигнахме:

using System.IO;

using System.Text;

 

class HtmlTagRemover

{

      private const string InputFileName = "Problem1.html";

      private const string Charset = "windows-1251";

 

      static void Main()

      {

            StreamReader reader = null;

            Encoding encoding = Encoding.GetEncoding(Charset);

            string line = string.Empty;

            StringBuilder result = new StringBuilder();

 

            if (!File.Exists(InputFileName))

            {

                  Console.WriteLine(

                        "File " + InputFileName + " not found.");

                  return;

            }

 

            try

            {

                  reader = new StreamReader(InputFileName, encoding);

                  while ((line = reader.ReadLine()) != null)

                  {

                        // Find what we need and save it in the result

                  }

            }

            catch (IOException ioex)

            {

                  Console.WriteLine(

                        "Can not read file " + InputFileName + ".");

            }

            finally

            {

                  if (reader != null)

                  {

                        reader.Close();

                  }

            }

      }

}

Справихме се с описаните проблеми и изглежда вече имаме коректно реализирано четенето на входния файл. За да сме напълно сигурни можем да тестваме. Например да изпишем съдържанието на входния файл на конзолата, а след това да пробваме с несъществуващ файл. Изписването ще става в while цикъла чрез Console.WriteLine(…).

Ако тестваме с примера от условието на задачата, резултатът е следният:

<html>

<head>

<title>Welcome to our site!</title>

</head>

<body>

<center>

<img src="/en/img/logo.gif" width="130" height="70" alt="Logo">

<br><br><br>

<font size="-1"><a href="/index.html">Home</a> -

<a href="/contenst.html">Contacts</a> -

<a href="/about.html">About</a></font><p>

</center>

</body>

</html>

Да пробваме с несъществуващ файл. Да заменим името на файла Problem1.html с Problem2.html. Резултатът от това е следният:

File Problem2.html not found

Уверихме се, че дотук написаният код е верен. Да преминем към следва­щата стъпка.

Стъпка 2 – премахване на таговете

Сега трябва да измислим подходящ начин да премахнем всички тагове. Какъв да бъде начинът?

Един възможен начин е като проверяваме реда символ по символ. За всеки символ от текущия ред ще търсим символа "<". От него надясно ще знаем, че е имаме някакъв таг (отварящ или затварящ). Краят на тага символът ">". Така можем да откриваме таговете и да ги премахваме. За да не получим долепяне на думите в съседни тагове, ще заместваме всеки таг със символа за празен ред "\n".

Алгоритъмът не е сложен за имплементиране, но дали няма по-хитър начин? Можем ли да използваме регулярни изрази? С тях лесно можем да търсим тагове и да ги заместваме с "\n", нали? Същевременно кодът няма да е сложен и при възникване на грешки по–лесно ще бъдат отстранени. Ще се спрем на този вариант. Какво трябва да направим? Първо трябва да напишем регулярния израз. Ето как изглежда той:

<[^>]*>

Идеята е проста: всеки низ, който започва с "<", продължава с произволи символи, различни от ">" и завършва с ">", е HTML таг. Ето как можем да заместим таговете със символ за нов ред:

private static string RemoveAllTags(string str)

{

      string textWithoutTags = Regex.Replace(str, "<[^>]*>", "\n");

      return textWithoutTags;

}

След като написахме тази стъпка, трябва да я тестваме. За целта отново ще изписваме намерените низове на конзолата чрез Console.WriteLine(…). Да тестваме кода, който получихме:

HtmlTagRemover.cs

using System.IO;

using System.Text.RegularExpressions;

 

class HtmlTagRemover

{

      private const string InputFileName = "Problem1.html";

      private const string Charset = "windows-1251";

 

      static void Main()

      {

            StreamReader reader = null;

            Encoding encoding = Encoding.GetEncoding(Charset);

            string line = string.Empty;

            StringBuilder result = new StringBuilder();

 

            if (!File.Exists(InputFileName))

            {

                  Console.WriteLine(

                        "File " + InputFileName + " not found.");

                  return;

            }

 

            try

            {

                  reader = new StreamReader(InputFileName, encoding);

                  while ((line = reader.ReadLine()) != null)

                  {

                        line = RemoveAllTags(line);

                        Console.WriteLine(line);

                  }

            }

            catch (IOException ioex)

            {

                  Console.WriteLine(

                        "Can not read file " + InputFileName + ".");

            }

            finally

            {

                  if (reader != null)

                  {

                        reader.Close();

                  }

            }

      }

 

      private static string RemoveAllTags(string str)

      {

            string strWithoutTags = Regex.Replace(str, "<[^>]*>",

                  "\n");

            return strWithoutTags;

      }

}

Да стартираме програмата със следния входен файл:

<html><body>

Click<a href="info.html">on this

link</a>for more info.<br />

This is<b>bold</b>text.

</body></html>

Резултатът ще бъде е следният:

(празени редове)

Click

on this

link

for more info.

(празен ред)

This is

bold

text.

(празни редове)

Всичко работи отлично, само че имаме излишни празни редове. Можем ли да ги премахнем? Това ще е следващата ни стъпка.

Стъпка 3 – премахване на празните редове

Можем да премахнем излишните празни редове, като заменяме двоен празен ред "\n\n" с единичен празен ред "\n". Два символа за нов ред един след друг означават преминаване на нов ред и още едно преминаване на нов ред, при което се получава празен ред. Затова не трябва да имаме такива струпвания на повече от един символ за нов ред \n. Ето примерен метод, който извършва замяната:

private static string RemoveDoubleNewLines(string str)

{

      return str.Replace("\n\n", "\n");

}

Както, винаги, преди да продължим напред, тестваме метода дали работи коректно. Пробваме с текст, в който няма празни редове, а след това добавяме 2, 3, 4 и 5 празни реда, включително в началото и в края на текста.

Установяваме, че методът не работи коректно, когато има 4 празни реда един след друг. Например ако подадем като входни данни "ab\n\n\n\ncd", получаваме "ab\n\n\cd" вместо "ab\ncd". Този дефект се получава, защото Replace(…) намира и замества съвпаденията еднократно отляво надясно. Ако в резултат на заместване се появи отново търсеният низ, той бива прескочен.

Видяхте колко е полезно всеки метод да бъде тестван на момента, а не накрая да се чудим защо програмата не работи и да имаме 200 реда код, пълен с грешки и да се чудим от къде идва проблема. Ранното откриване на дефектите е много полезно и трябва да го правите винаги, когато е възможно. Ето поправения код:

private static string RemoveDoubleNewLines(string str)

{

      string pattern = "[\n]+";

      return Regex.Replace(str, pattern, "\n");

}

След серия тестове, се убеждаваме, че сега вече методът работи коректно. Готови сме да тестваме цялата програма дали отстранява коректно излишните нови редове. За целта правим следната промяна:

while ((line = reader.ReadLine()) != null)

{

      line = RemoveAllTags(line);

      line = RemoveDoubleNewLines(line);

      Console.WriteLine(line);

}

Изглежда пак има празни редове. От къде ли идват? Вероятно, ако имаме ред, който съдържа само тагове, той ще създаде проблем. Следователно трябва да предвидим този случай. Добавяме следната проверка:

if (!string.IsNullOrEmpty(line))

{

      Console.WriteLine(line);

}

Това премахва повечето празни редове, но не всички.

Ако се замислим, би могло да се случи така, че някой ред да започва или завършва с таг. Тогава този таг ще бъде заменен с единичен празен ред и така в началото или в края на реда може да има празен ред. Това означава, че трябва да чистим празните редове в началото и в края на всеки ред. Ето как можем да направим въпросното изчистване:

private static string TrimNewLines(string str)

{

      int start = 0;

      while (start < str.Length && str[start] == '\n')

      {

            start++;

      }

 

      int end = str.Length - 1;

      while (end >= 0 && str[end] == '\n')

      {

            end--;

      }

 

      if (start > end)

      {

            return string.Empty;

      }

 

      string trimmed = str.Substring(start, end - start + 1);

      return trimmed;

}

Методът работи много просто: преминава отляво надясно пред входния символен низ и прескача всички символи за празен ред. След това преминава отдясно наляво и отново прескача всички символи за празен ред. Ако лявата и дясната позиция са се разминали, това означава, че низът или е празен, или съдържа само символи за празен ред. Тогава връщаме празен низ. Иначе връщаме всичко надясно от стартовата позиция и наляво от крайната позиция.

Както винаги, тестваме въпросния метод дали работи коректно с няколко примера, сред които празен низ, низ без нови редове, низ с нови редове отляво или отдясно или и от двете страни и низ само с нови редове. Убеждаваме се, че методът работи коректно.

Сега остава да модифицираме логиката на обработката на входния файл:

while ((line = reader.ReadLine()) != null)

{

      line = RemoveAllTags(line);

      line = RemoveDoubleNewLines(line);

      line = TrimNewLines(line);

      if (!string.IsNullOrEmpty(line))

      {

            Console.WriteLine(line);

      }

}

Този път тестваме и се убеждаваме, че всичко работи коректно.

Стъпка 4 – записване на резултата във файл

Остава ни да запишем резултата в изходен файл. За да записваме резултата в изходния файл ще използваме StreamWriter. Тази стъпка е тривиална. Трябва да се съобразим само, че писането във файл може да предизвика изключение и затова трябва да променим леко логиката за обработка на грешки и за отварянето и затварянето на потоците за входния и изходния файл.

Ето какво се получава най-накрая като изходен код на програмата:

HtmlTagRemover.cs

using System.IO;

using System.Text.RegularExpressions;

 

class HtmlTagRemover

{

      private const string InputFileName = "Problem1.html";

      private const string OutputFileName = "Problem1.txt";

      private const string Charset = "windows-1251";

 

      static void Main()

      {

            StreamReader reader = null;

            StreamWriter writer = null;

            Encoding encoding = Encoding.GetEncoding(Charset);

            string line = string.Empty;

            StringBuilder result = new StringBuilder();

 

            if (!File.Exists(InputFileName))

            {

                  Console.WriteLine(

                        "File " + InputFileName + " not found.");

                  return;

            }

 

            try

            {

                  reader = new StreamReader(InputFileName, encoding);

                  writer = new StreamWriter(OutputFileName, false,

                        encoding);

 

                  while ((line = reader.ReadLine()) != null)

                  {

                        line = RemoveAllTags(line);

                        line = RemoveDoubleNewLines(line);

                        line = TrimNewLines(line);

                        if (!string.IsNullOrEmpty(line))

                        {

                              writer.WriteLine(line);

                        }

 

                  }

            }

            catch (IOException ioex)

            {

                  Console.WriteLine(

                        "Can not read file " + InputFileName + ".");

            }

            finally

            {

                  if (reader != null)

                  {

                        reader.Close();

                  }

 

                  if (writer != null)

                  {

                        writer.Close();

                  }

            }

      }

 

      /// <summary>

      /// Replaces every tag with new line

      /// </summary>

      private static string RemoveAllTags(string str)

      {

            string strWithoutTags = Regex.Replace(str, "<[^>]*>",

                  "\n");

            return strWithoutTags;

      }

 

      /// <summary>

      /// Replaces sequence of new lines with only one new line

      /// </summary>

      private static string RemoveDoubleNewLines(string str)

      {

            string pattern = "[\n]+";

            return Regex.Replace(str, pattern, "\n");

      }

 

      /// <summary>

      /// Removes new lines from start and end of string

      /// </summary>

      private static string TrimNewLines(string str)

      {

            int start = 0;

            while (start < str.Length && str[start] == '\n')

            {

                  start++;

            }

 

            int end = str.Length - 1;

            while (end >= 0 && str[end] == '\n')

            {

                  end--;

            }

 

            if (start > end)

            {

                  return string.Empty;

            }

 

            string trimmed = str.Substring(start, end - start + 1);

            return trimmed;

      }

}

Тестване на решението

Досега тествахме отделните стъпки от решението на задачата. Чрез извършените тестове на отделните стъпки намаляваме възможността за грешки, но това не значи, че не трябва да тестваме цялото решение. Може да сме пропуснали нещо, нали?

Тестваме с примерния входен файл от условието на задачата. Всичко работи коректно.

Тестваме с нашия "сложен" пример. Всичко работи добре.

Задължително трябва да тестваме граничните случаи и да пуснем тест за производителност.

Започваме с празен файл. Изходът е коректен – празен файл.

Тестваме с файл, който съдържа само една дума "Hello" и не съдържа тагове. Резултатът е коректен – изходът съдържа само думата "Hello".

Тестваме с файл, който съдържа само тагове и не съдържа текст. Резултатът е отново коректен – празен файл.

Пробваме да сложим празни редове на най-невероятни места във входния файл. Пускаме следния тест:

  Hello

 

<br><br>

 

 

<b>I<b> am here

 

I am not <b>here</b>

Изходът е следният:

  Hello

I

 am here

I am not

here

Изглежда открихме дребен дефект. Има един интервал в началото на два от редовете. Според условието не е много ясно дали това е дефект, но нека се опитаме да го оправим.

Добавяме следния код при обработката на поредния ред от входния файл:

line = line.Trim();

Дефектът се премахва, но само на първия ред. Пускаме дебъгера и забелязваме защо се получава така. Причината е, че отпечатваме в изходния файл символен низ със стойност "I\n am here" и така получаваме интервал след празен ред. Можем да поправим дефекта, като навсякъде заместим празен ред, следван от празно пространство (празни редове, интервали, табулации и т.н.), с единичен празен ред. Ето как изглежда поправ­ката:

private static string RemoveDoubleNewLines(string str)

{

      string pattern = "\n\\s+";

      return Regex.Replace(str, pattern, "\n");

}

Поправихме и тази грешка. Единствено трябва да му сменим името с някакво по-адекватно като например RemoveNewLinesWithWhiteSpace().

Сега трябва отново да тестваме упорито след поправката. Слагаме нови редове и интервали пръснати безразборно и се уверяваме се, че всичко работи вече коректно.

Остана един последен тест – за производителност. Лесно можем да създадем обемен входен файл. Отваряме някой сайт, примерно http://microsoft.com , взимаме сорс кода му и го копираме 1000 пъти. Получаваме достатъчно голям входен файл. В нашия случай се получи 44 MB файл с 947 000 реда. За обработката му бяха нужни под 10 секунди, което е напълно приемлива скорост. Когато тестваме решението не трябва да забравяме, че обработката на файла зависи от компютърната ни конфигурация.

Като надникнем в резултата, обаче, забелязваме много неприятен проб­лем. В него има части от тагове. По-точно виждаме следното:

<!--

var s_pageName="home page"

//-->

Бързо става ясно, че сме изпуснали един много интересен случай. В HTML може един таг да бъде затворен няколко реда след отварянето си, т.е. един таг може да е разположен на няколко последователни реда. Точно такъв е нашият случай: имаме таг с коментари, който съдържа JavaScript код. Ако програмата работеше коректно, щеше да отреже целия таг вместо да го запази в изходния файл.

Видяхте колко е полезно тестването и колко е важно. В някои сериозни фирми (като например Майкрософт) решение без тестове се счита за готово на 50%. Това означава, че ако пишете код 2 часа, трябва да отделите за тестване (ръчно или автоматизирано) поне още 2 часа! Само така можете да създадете качествен софтуер.

Колко жалко, че открихме проблема чак сега вместо в началото, когато проверявахме дали е правилна идеята ни за решение на задачата, преди да сме написали програмата. Понякога се случва така, няма как.

Как да оправим проблема с тагове на два реда?

Първата идея, която ни хрумва, е да заредим в паметта целия входен файл и да го обработваме като един голям стринг вместо ред по ред. Това е идея, която изглежда ще работи, но ще работи бавно и ще консумира голямо количество памет. Нека потърсим друга идея.

Очевидно не можем да четем файла ред по ред. Можем ли да го четем символ по символ? Ако можем, как ще обработваме таговете? Хрумва ни, че ако четем файла символ по символ, можем във всеки един момент да знаем дали сме в таг или сме извън таг и ако сме извън таг, можем да печатаме всичко, което прочетем. Ще се получи нещо такова:

bool inTag = false;

while (! <end of file is reached>)

{

      char ch = <read next character>;

      if (ch == '<')

      {

            inTag = true;

      }

      else if (ch == '>')

      {

            inTag = false;

      }

      else

      {

            if (!inTag)

            {

                  PrintBuffer(ch);

            }

      }

}

Идеята е много проста и лесна за реализация. Ако я реализираме директно, ще имаме проблема с празните редове и проблема със слива­нето на текст от съседни тагове. За да разрешим този проблем, можем да натрупваме текста в StringBuilder и да го отпечатваме при край на файла или при преминаване към таг. Ще се получи нещо такова:

bool inTag = false;

StringBuilder buffer = new StringBuilder();

while (! <end of file is reached>)

{

      char ch = <read next character>;

      if (ch == '<')

      {

            if (!inTag)

            {

                  PrintBuffer(buffer);

            }

            buffer.Remove(0, buffer.Length);

            inTag = true;

      }

      else if (ch == '>')

      {

            inTag = false;

      }

      else

      {

            if (!inTag)

            {

                  buffer.Append(ch);

            }

      }

}

PrintBuffer(buffer);

Ако добавим и логиката за избягване на празните редове, както и четенето на входа и писането на резултата, ще получим цялостно решение на задачата по новия алгоритъм:

using System.IO;

using System.Text.RegularExpressions;

 

public class SimpleHtmlTagRemover

{

      private const string InputFileName = "Problem1.html";

      private const string OutputFileName = "Problem1.txt";

      private const string Charset = "windows-1251";

 

      public static void Main()

      {

            StringBuilder buffer = new StringBuilder();

            bool inTag = false;

 

            if (!File.Exists(InputFileName))

            {

                  Console.WriteLine(

                        "File " + InputFileName + " not found.");

                  return;

            }

 

            StreamReader reader = null;

            StreamWriter writer = null;

            try

            {

                  Encoding encoding = Encoding.GetEncoding(Charset);

                  reader = new StreamReader(InputFileName, encoding);

                  writer = new StreamWriter(OutputFileName, false,

                        encoding);

 

                  Regex regex = new Regex("\n\\s+");

                  while (true)

                  {

                        int nextChar = reader.Read();

                        if (nextChar == -1)

                        {

                              // End of file reached

                              PrintBuffer(writer, buffer, regex);

                              break;

                        }

                        char ch = (char)nextChar;

                        if (ch == '<')

                        {

                              if (!inTag)

                              {

                                    PrintBuffer(writer, buffer, regex);

                              }

                              buffer.Length = 0;

                              inTag = true;

                        }

                        else if (ch == '>')

                        {

                              inTag = false;

                        }

                        else

                        {

                              // We have other character

                              // (not "<" or ">")

                              if (!inTag)

                              {

                                    buffer.Append(ch);

                              }

                        }

 

                  }

            }

            catch (IOException ioex)

            {

                  Console.WriteLine(

                        "Can not read file " + InputFileName + ".");

            }

            finally

            {

                  if (reader != null)

                  {

                        reader.Close();

                  }

 

                  if (writer != null)

                  {

                        writer.Close();

                  }

            }

      }

 

      /// <summary>

      /// Remove intervals and prints the buffer in file

      /// </summary>

      /// <param name="writer">store the result in file</param>

      /// <param name="buffer">keeps the current result from

      /// html file</param>

      /// <param name="regex">removes new lines followed by

      /// whitespaces</param>

      private static void PrintBuffer(StreamWriter writer,

                  StringBuilder buffer, Regex regex)

      {

            string str = buffer.ToString();

            string trimmed = str.Trim();

            string textOnly = regex.Replace(trimmed, "\n");

            if (!string.IsNullOrEmpty(textOnly))

            {

                  writer.WriteLine(textOnly);

            }

      }

}

Входният файл чете символ по символ с класа StreamReader.

Първоначално буферът за натрупване на текст е празен. В главния цикъл анализираме всеки прочетен символ. Имаме следните случаи:

-    Ако стигнем до края на файла, отпечатваме каквото има в буфера и алгоритъмът приключва.

-    При срещане на символ "<" (начало на таг) първо отпечатваме буфера (ако установим, че преминаваме от текст към таг). След това зачистваме буфера и установяваме inTag = true.

-    При срещане на символ ">" (край на таг) установяваме inTag = false. Това ще позволи следващите след тага символи да се натрупват в буфера.

-    При срещане на някой друг символ (текст или празно пространство), той се добавя към буфера, ако сме извън таг. Ако сме в таг, символът се игнорира.

Печатането на буфера се грижи да премахва празните редове в текста и да изчиства празното пространство в началото и в края на текста. Как точно извършваме това, вече разгледахме в предходното решение на задачата, което се оказа грешно.

Във второто решение обработката на буфера е много по-лека и по-кратка, затова буфера се обработва непосредствено преди отпечатването му.

В предишното решение на задачата използваме регулярни изрази за заместване чрез статичните методи на класа Regex, а сега създаваме  обект от същия клас. При статичния вариант шаблонът се компилира за машината на регулярните изрази всеки път, иначе само веднъж и просто се подава като аргумент на нужните методи.

Тестване на новото решение

Остава да тестваме задълбочено новото решение. Изпълняваме всички тестове, които проведохме за предното решение. Добавяме тест с тагове, които се разпростират на няколко реда. Отново тестваме за производител­ност със сайта на Java 1000 пъти. Уверяваме се, че и за него програмата работи коректно и дори е по-бърза.

Нека да пробваме с някой друг сайт, например сайта на книгата: https://introprogramming.info/. Отново взимаме сорс кода му и пускаме решението на нашата задача. След внимателно преглеждане на входните данни (сорс кода на сайта на книгата) и изходния файл, забелязваме че отново има проблем. Част от съдържанието от този таг се отпечатва в изходния файл:

<!--

<br />

<br />

Прочетете безплатната книга на Светлин Наков и колектив за програмиране на Java.

...

...

-->

Къде е проблемът?

Проблемът изглежда възниква, когато в един таг се среща друг таг преди първият да бъде затворен. Това може да се случи при HTML коментарите. Ето как се стига до грешката:

clip_image002

Както знаем в решението на задачата използваме булева променлива (inTag), за да знаем дали текущия символ се намира  в таг или не. На картинката сме показали, че в момент 1 установяваме inTag = true. Дотук изпълнението на задачата е нормално. Настъпва в момент 2, където текущия прочетен символ  е ">". В този момент установяваме inTag = false. Проблема е, че тагът, който е отворен от момент 1 все още не е затворен, а булевата променлива показва, че вече не сме в таг и следващите символи се записват в буфера. Ако между двата тага за нов ред (<br />) имаше текст, той също щеше да бъде записан в буфера.

Как да оправим проблема?

Оказа се, че и във второто решение има грешка. Програмата не работи коректно при  наличието на вложени тагове в таг с коментари. Чрез булева променлива може да знаем само дали сме в таг или не, но не и да помним дали все още дали сме в предходния. Това ни подсказва, че вместо да използваме булева променлива може да запомняме броя на таговете, в които се намираме (променлива от тип int). Ще модифици­раме предходното решение:

int openedTags = 0;

StringBuilder buffer = new StringBuilder();

while (! <end of file is reached>)

{

      char ch = <read next character>;

      if (ch == '<')

      {

            if (openedTags == 0)

            {

                  PrintBuffer(buffer);

            }

            buffer.Remove(0, buffer.Length);

            openedTags++;

      }

      else if (ch == '>')

      {

            openedTags--;

      }

      else

      {

            if (openedTags == 0)

            {

                  buffer.Append(ch);

            }

      }

}

PrintBuffer(buffer);

В главния цикъл анализираме всеки прочетен символ. Имаме следните случаи:

-    Ако стигнем до края на файла, отпечатваме каквото има в буфера и алгоритъмът приключва.

-    При срещане на символ "<" (начало на таг) първо отпечатваме буфера (ако установим, че преминаваме от текст към таг). След това зачистваме буфера и увеличаваме брояча с единица.

-    При срещане на символ ">" (край на таг) намаляваме брояча с единица. Затварянето на вложен таг няма да позволи натрупване в буфера. Ако след затварянето на таг сме  извън тагове символите ще започнат да се натрупват в буфера.

-    При срещане на някой друг символ (текст или празно пространство), той се добавя към буфера, ако сме извън таг. Ако сме в таг, символът се игнорира.

Остава ни да напишем цялото решение и след това да тестваме. Логиката по четенето на входния файл и печатането на буфера остава същата:

using System.IO;

using System.Text.RegularExpressions;

 

public class SimpleHtmlTagRemover

{

      private const string InputFileName = "Problem1.html";

      private const string OutputFileName = "Problem1.txt";

      private const string Charset = "windows-1251";

 

      public static void Main()

      {

            StringBuilder buffer = new StringBuilder();

            int openedTags = 0;

 

            if (!File.Exists(InputFileName))

            {

                  Console.WriteLine(

                        "File " + InputFileName + " not found.");

                  return;

            }

 

            StreamReader reader = null;

            StreamWriter writer = null;

            try

            {

                  Encoding encoding = Encoding.GetEncoding(Charset);

                  reader = new StreamReader(InputFileName, encoding);

                  writer = new StreamWriter(OutputFileName, false,

                        encoding);

 

                  Regex regex = new Regex("\n\\s+");

                  while (true)

                  {

                        int nextChar = reader.Read();

                        if (nextChar == -1)

                        {

                              // End of file reached

                              PrintBuffer(writer, buffer, regex);

                              break;

                        }

                        char ch = (char)nextChar;

                        if (ch == '<')

                        {

                              if (openedTags == 0)

                              {

                                    PrintBuffer(writer, buffer, regex);

                                    buffer.Length = 0;

                              }

                              openedTags++;

                        }

                        else if (ch == '>')

                        {

                              openedTags--;

 

                        }

                        else

                        {

                              // We aren't in tags (not "<" or ">")

                              if (openedTags == 0)

                              {

                                    buffer.Append(ch);

                              }

                        }

                  }

            }

            catch (IOException ioex)

            {

                  Console.WriteLine(

                        "Can not read file " + InputFileName + ".");

            }

            finally

            {

                  if (reader != null)

                  {

                        reader.Close();

                  }

 

                  if (writer != null)

                  {

                        writer.Close();

                  }

            }

      }

 

      /// <summary>

      /// Rеmove intervals and prints the buffer in file

      /// </summary>

      /// <param name="writer">store the result in file</param>

      /// <param name="buffer">keeps the current result from

      /// html file</param>

      /// <param name="regex">removes new lines followed by

      /// whitespaces</param>

      private static void PrintBuffer(StreamWriter writer,

                  StringBuilder buffer, Regex regex)

      {

            string str = buffer.ToString();

            string trimmed = str.Trim();

            string textOnly = regex.Replace(trimmed, "\n");

            if (!string.IsNullOrEmpty(textOnly))

            {

                  writer.WriteLine(textOnly);

            }

      }

}

Тестване на новото решение

Отново  тестваме решението на задачата. Изпълняваме всички тестове, направени за предното решение (вж. секцията "Тестване на решението"). Да пробваме със сайта на MSDN (http://msdn.microsoft.com/). Нека да проверим внимателно изходния файл. Може да се види, че в края на файла има грешни символи. След като внимателно прегледаме сорс кода на сайта на MSDN забелязваме, че има неправилно изобразяване на символа ">" (за да се визуализира този символ в HTML документ, трябва да се използва "&gt;", а не ">"). Грешката е на сайта на MSDN, а не в нашата програма.

Сега ни остава да тестваме за производител­ност със сайта на книгата (https://introprogramming.info), копиран 1000 пъти. Уверяваме се, че и за него програмата работи достатъчно бързо.

Най-сетне вече сме готови за следващата задача.

Задача 2: Лабиринт

Даден е лабиринт, който се състои от N x N квадратчета, всяко от които може да е проходимо (0) или не (x). В едно от квадратчетата се намира нашият герой Минчо (*):

x

x

x

x

x

x

0

x

0

0

0

x

x

*

0

x

0

x

x

x

x

x

0

x

0

0

0

0

0

x

0

x

x

x

0

x

Две квадратчета са съседни, ако имат обща стена. Минчо може на една стъпка да преминава от едно проходимо квадратче в съседно на него проходимо квадратче. Ако Минчо стъпи в клетка, която е на границата на лабиринта, той може с една стъпка да излезе извън него. Напишете програма, която по даден лабиринт отпе­чатва минималния брой стъпки, необходими на Минчо, за да излезе от лабиринта или -1 ако няма изход.

Входните данни се четат от текстов файл с име Problem2.in. На първия ред във файла стои числото N (2 < N < 100). На следващите N реда стоят по N символа, всеки от които е или "0" или "x" или "*". Изходът представлява едно число и трябва да се изведе във файла Problem2.out.

Примерен входен файл Problem2.in:

6

xxxxxx

0x000x

x*0x0x

xxxx0x

00000x

0xxx0x

Примерен изходен файл Problem2.out:

9

Измисляне на идея за решение

Имаме лабиринт и трябва да намерим най-краткия път в него. Това не е лесна задача и трябва доста да помислим или да сме прочели някъде как се решават такива задачи.

Нашият алгоритъм ще започва от работата си от началната точка, която ни е дадена. Знаем, че можем да се предвижваме в съседна клетка хори­зонтално и вертикално, но не и по диагонал. Нашият алгоритъм трябва да обхожда лабиринта по някакъв начин, за да намери в него най-късия път. Как да обхождаме клетките в лабиринта?

Един възможен начин за обхождане е следният: стартираме от началната клетка. Преместваме се в съседна клетка на текущата (която е прохо­дима), след това в съседна клетка на нея (която е проходима и все още непосетена), след това в съседна на последната посетена (която е проходима е все още непосетена) и така продължаваме рекурсивно напред, докато или стигнем изход от лабиринта, или стигнем до място, от където няма продължение (няма съседна клетка, която е свободна и непосетена). В този момент се връщаме от рекурсията (към предходната клетка, от която сме стигнали текущата) и посещаваме друга клетка на предходната клетка. Ако няма продължение, се връщаме още назад. Описаният рекурсивен процес представлява обхождане на лабиринта в дълбочина (спомнете си главата "Рекурсия").

Възниква въпросът "Нужно ли е да минаваме през една клетка повече от един път"? Ако минаваме през една клетка най-много веднъж, то бързо ще обходим целия лабиринт и ако има изход, ще го намерим. Обаче мини­мален ли ще е този път. Ако си нарисуваме процеса на хартия, бързо ще се убедим, че намереният път няма да е минимален.

Ако при връщане от рекурсията отбелязваме като свободна клетката, която напускаме, ще позволим до една и съща клетка да стигаме много­кратно, идвайки по различен път. Пълното рекурсивно обхождане на лабиринта на практика ще намери всички възможни пътища от началната клетка до всяка други клетка. От всички тези пътища можем да вземем най-късия път до клетка на границата на лабиринта (изход) и така ще намерим решение на задачата.

Проверка на идеята

Изглежда имаме идея за решаване на задачата: с рекурсивно обхождане намираме всички пътища в лабиринта от началната клетка до клетка на границата на лабиринта и измежду всички тези пътища избираме най-късия. Нека да проверим идеята.

Взимаме лист хартия и си правим един примерен лабиринт. Пробваме алгоритъма. Вижда се, че той намира всички пътища от началната клетка до някой от изходите, като доста обикаля напред-назад. В крайна сметка намира всички изходи и измежду всички пътища може да се избере най-краткият.

Дали идеята работи, ако няма изход? Правим си втори лабиринт, който е без изход. Пробваме алгоритъма върху него, отново на лист хартия. Виж­даме, че след доста обикаляне напред-назад алгоритъмът не намира нито един изход и приключва.

Изглежда имаме правилна идея за решаване на задачата. Да преминем напред и да помислим за структурите от данни.

Какви структури от данни да използваме?

Първо трябва да преценим как да съхраняваме лабиринта. Съвсем естест­вено е да ползваме матрица от символи, точно като тази на картинката. Ще считаме, че една клетка е проходима и можем да влезем в нея, ако съдържа символ, различен от символа 'x'. Може да пазим лабиринта и в матрица с числа или булеви стойности, но разликата не е съществена. Матрицата от символи е удобна за отпечатване, а това ще ни помогне докато дебъгваме. Няма много възможности за избор. Ще съхраняваме лабиринта в матрица от символи.

След това трябва да решим в каква структура да запомняме обходените до момента клетки по време на рекурсията (текущия път). На нас ни трябва винаги последната обходена клетка. Това ни навежда на мисълта за структура, която спазва "последен влязъл, пръв излязъл", тоест стек. Можем да ползваме Stack<Cell>, където Cell е клас, съдържащ коорди­натите на една клетка (номер на ред и номер на колона).

Остава да помислим в какво да запомняме намерените пътища, за да можем да извлечем накрая най-късия от тях. Ако се замислим малко, не е нужно да пазим всички пътища. Достатъчно е да помним текущия път и най-късият път за момента. Дори не е необходимо да пазим най-късия път за момента, ами само неговата дължина. Всеки път, когато намерим път до изход от лабиринта, можем да взимаме неговата дължина и ако тя е по-малка от най-късата дължина за момента, да я запомняме.

Изглежда намерихме ефективни структури от данни. Според препоръките за решаване на задачи, още не трябва да се втурваме да пишем кода на програмата, защото трябва да помислим за ефективността на алгоритъма.

Да помислим за ефективността

Нека да проверим идеята си от следна точка на ефективността? Какво правим ние? Намираме всички възможни пътища и от тях взимаме най-късия. Няма спор, че алгоритъмът ще работи, но ако лабиринтът стане много голям, дали ще работи бързо?

За да отговорим на този въпрос, трябва да помислим колко са пътищата. Ако вземем празен лабиринт, то на всяка стъпка на рекурсията ще имаме средно по 3 свободни продъл­жения (като изключим клетката, от която идваме).

Така, ако имаме примерно лабиринт 10 на 10, пътят може да стане дълъг до 100 клетки и по време на обхождането на всяка стъпка ще имаме по 3 съседни клетки. Изглежда броят пътища е число от порядъка на 3 на степен 100. Очевидно алгоритъмът ще "приспи" компютъра много бързо.

Намерихме сериозен проблем на алгоритъма. Той ще работи много бавно, дори при малки лабиринти, а при големи изобщо няма да работи! Хубавото е, че още не сме написали нито един ред код и генералната смяна на подхода към задачата няма да ни струва много пропиляно време.

Да измислим нова идея

Разбрахме, че обхождането на всички пътища в лабиринта е неправилен подход, затова трябва да измислим друг.

Нека започнем от началната клетка и да обходим всички нейни съседни и да ги маркираме като обходени. За всяка обходена клетка ще запомняме едно число, което е равно на броя клетки, през които сме преминали, за да достигнем до нея (дължина на пътя от началната клетка до текущата).

За началната клетка дължината на пътя е 0. За нейните съседи дължината на пътя трябва да е 1, защото с 1 движение можем да ги достигнем от началната клетка. За съседните клетки на съседите на началната клетка дължината на пътя е 2. Можем да продължим да разсъждаваме по този начин и ще стигнем до следния алгоритъм:

1.  Записваме дължина на пътя 0 за началната клетка. Отбелязваме я като посетена.

2.  За всяка съседна клетка на началната отбелязваме, че пътят до нея е с дължина 1. Отбелязваме тези клетки като посетени.

3.  За всяка клетка, която е съседна на клетка с дължина 1 и не е посетена, записваме, че е дължината на пътя до нея е 2. Отбелязваме въпросните клетки като посетени.

4.  Продължавайки аналогично, на N-тата стъпка намираме всички непосетени все още клетки, които са на разстояние N премествания от началната клетка и ги отбелязваме като посетени.

Можем да визуализираме процеса по следния начин (взимаме друг лабиринт, за да покажем по-добре идеята):

Стъпка 0 – отбелязваме разстоянието от началната клетка до нея самата с 0 (за удобство на картинката отбелязваме свободните клетки с "-"):

x

x

x

x

x

x

-

x

-

-

-

x

x

0

-

x

-

x

x

-

-

x

-

x

x

-

-

-

-

x

-

x

x

x

-

x

Стъпка 1 – отбелязваме с 1 всички проходими съседи на клетки със стойност 0:

x

x

x

x

x

x

-

x

-

-

-

x

x

0

1

x

-

x

x

1

-

x

-

x

x

-

-

-

-

x

-

x

x

x

-

x

Стъпка 2 – отбелязваме с 2 всички проходими съседи на клетки с 1:

x

x

x

x

x

x

-

x

2

-

-

x

x

0

1

x

-

x

x

1

2

x

-

x

x

2

-

-

-

x

-

x

x

x

-

x

Стъпка 3 – отбелязваме с 3 всички проходими съседи на клетки със стойност 2:



x

x

x

x

x

x

-

x

2

3

-

x

x

0

1

x

-

x

x

1

2

x

-

x

x

2

3

-

-

x

-

x

x

x

-

x

Продължавайки така, в един момент или ще достигнем клетка на границата на лабиринта (т.е. изход) или ще установим, че такава не е достижима.

Проверяване производителността на новия алгоритъм

Понеже никога не посещаваме повече от веднъж една и съща клетка, броят стъпки, които извършва този алгоритъм, не би трябвало да е голям. Примерно, ако имаме лабиринт с размери 100 на 100, той ще има 10 000 клетки, всяка от които ще посетим най-много веднъж и за всяка посетена клетка ще проверим всеки неин съсед дали е свободен, т.е. ще извършим по 4 проверки. В крайна сметка ще извършим най-много 40 000 проверки и ще обходим най-много 10 000 клетки. Общо ще направим около 50 000 операции. Това означава, че алгоритъмът ще работи мигновено.

Проверяване коректността на новия алгоритъм

Изглежда този път нямаме проблем с производителността. Имаме бърз алгоритъм.

Да проверим дали е коректен. За целта си рисуваме на лист хартия някой по-голям и по-сложен пример, в който има много изходи и много пътища и започваме да изпълняваме алгоритъма. Изглежда работи коректно.

След това пробваме с лабиринт без изход. Изглежда алгоритъмът завър­шва, но не намира изход. Следователно работи коректно.

Пробваме още 2-3 примера и се убеждаваме, че този алгоритъм винаги намира най-краткия път до някой изход и винаги работи бързо, защото обхожда всяка клетка от лабиринта най-много веднъж.

Какви структури от данни да използваме?

С новия алгоритъм обхождаме последователно всички съседни клетки на началната клетка. Можем да ги сложим в някаква структура данни, примерно масив или по-добре списък(или списък от списъци), понеже в масива не можем да добавяме.

След това взимаме списъка с достигнатите на последната стъпка клетки и добавяме в друг списък техните съседи.

Така, ако индексираме списъците, получаваме списък0, който съдържа началната клетка, списък1, който съдържа проходимите съседни на началната клетка, след това списък2, който съдържа проходимите съседи на списък1 и т.н. На n-тата стъпка получаваме списъкn, който съдържа всички клетки, достижими за точно n стъпки, т.е. клетките на разстояние n от стартовата клетка.

Изглежда можем да ползваме списък от списъци, за да пазим клетките, получени на всяка стъпка. Ако се замислим, за да получим n-тия списък, ни е достатъчен (n-1)-вия. Реално не ни трябва списък от списъци, а само списъкът от последната стъпка.

Можем да достигнем и до по-генерален извод: клетките се обработват в реда на постъпване: когато свършват клетките от стъпка k, чак тогава се обработват клетките от стъпка k+1, а едва след тях – клетките от стъпка k+2 и т.н. Процесът прилича на опашка – по-рано постъпилите клетки се обработват по-рано.

За да реализираме алгоритъма, можем да използваме опашка от клетки. За целта трябва да дефинираме клас клетка (Cell), който да съдържа координатите на дадена клетка (ред и колона). Можем да пазим в матрицата за всяка клетка на какво разстояние се намира от началната клетка или -1, ако разстоянието още не е пресметнато.

Ако се замислим още малко, разстоянието от стартовата клетка може да се пази в самата клетка (в класа Cell) вместо да се прави специална матрица за разстоянията. Така ще се спести памет.

Вече имаме яснота за структурите данни. Остава да реализираме алго­ритъма – стъпка по стъпка.

Стъпка 1 – класът Cell

Можем да започнем от дефиницията на класа Cell. Той ще ни трябва, за да запазим стартовата клетка, от която започва търсенето на пътя. За да е по-кратък и прегледен кодът ще използваме автоматични свойства (automatic properties или auto-implemented properties). Благодарение на тях не се налага да декларираме поле за съответното свойство, нито да пишем код за извличане или промяна на стойността на полето. Това се извършва автоматично от компилатора по време на компилацията. Чрез инструментите за дисасемблиране JustDecompiler или ILSpy, споменати в глава 1, може да проверите какво e генерирал компилаторът при използва­нето на автоматични свойства във вашата програма. Ето го и класът Cell:

public class Cell

{

      public int Row { get; set; }

      public int Column { get; set; }

      public int Distance { get; set; }

}

Може да му добавим и конструктор за удобство:

public Cell(int row, int column, int distance)

{

      this.Row = row;

      this.Column = column;

      this.Distance = distance;

}

Стъпка 2 – прочитане на входния файл

Ще четем входния файл ред по ред чрез познатия ни клас StreamReader. На всеки ред ще анализираме символите и ще ги записваме в матрица от символи. При достигане на символ "*" ще запомним координатите му в инстанция на класа Cell, за да знаем от къде да започнем търсенето на най-краткия път за излизане от лабиринта.

Можем да дефинираме клас Maze и в него да пазим матрицата на лабиринта и стартовата клетка:

Maze.cs

public class Maze

{

      private char[,] maze;

      private int size;

      private Cell startCell = null;

 

      public void ReadFromFile(string fileName)

      {

            using (StreamReader reader = new StreamReader(fileName))

            {

                  // Read maze size and create maze

                  this.size = int.Parse(reader.ReadLine());

                  this.maze = new char[this.size, this.size];

 

                  // Read the maze cells from the file

                  for (int row = 0; row < this.size; row++)

                  {

                        string line = reader.ReadLine();

                        for (int col = 0; col < this.size; col++)

                        {

                              this.maze[row, col] = line[col];

                              if (line[col] == '*')

                              {

                                    this.startCell = new Cell(row, col, 0);

                              }

                        }

                  }

            }

      }

}

За простота ще пропуснем обработката на грешки при четене и писане във файл. При възникване на изключение го изхвърляме от главния метод и ще оставяме CLR да го отпечата в конзолата.

Вече имаме класа Maze и подходящо представяне на данните от входния файл. За да сме сигурни, че написаното дотук е вярно трябва да тестваме. Можем да проверим дали матрицата е вярно попълнена, като я отпечатаме на конзолата. Друг вариант е да разгледаме стойностите на полетата от класа Maze през дебъгера на Visual Studio.

След като тестваме написаното дотук продължаваме със следващата стъпка, а именно търсенето на най-краткия път.

Стъпка 3 – намиране на най-къс път

Можем директно да имплементираме алгоритъма, който вече дискути­рахме. Трябва да дефинираме опашка и в нея да сложим в началото стартовата клетка. След това в цикъл трябва да взимаме поредната клетка от опашката и да добавяме всичките й непосетени проходими съседи. На всяка стъпка има шанс да стъпим в клетка от границата на лабиринта, при което считаме, че сме намерили изход и търсенето приключва. Повтаряме цикъла докато опашката свърши. При всяко посещение на дадена клетка проверяваме дали клетката е свободна и ако е свободна, я маркираме като непроходима. Така избягваме повторно попадане в същата клетка. Ето как изглежда имплементацията на алгоритъма:

public int FindShortestPath()

{

      // Queue for traversing the cells in the maze

      Queue<Cell> visitedCells = new Queue<Cell>();

      VisitCell(visitedCells, this.startCell.Row,

            this.startCell.Column, 0);

 

      // Perform Breath-First-Search (BFS)

      while (visitedCells.Count > 0)

      {

            Cell currentCell = visitedCells.Dequeue();

            int row = currentCell.Row;

            int column = currentCell.Column;

            int distance = currentCell.Distance;

            if ((row == 0) || (row == size - 1)

                  || (column == 0) || (column == size - 1))

            {

                  // We are at the maze border

                  return distance + 1;

            }

            VisitCell(visitedCells, row, column + 1, distance + 1);

            VisitCell(visitedCells, row, column - 1, distance + 1);

            VisitCell(visitedCells, row + 1, column, distance + 1);

            VisitCell(visitedCells, row - 1, column, distance + 1);

      }

 

      // We didn't reach any cell at the maze border -> no path

      return -1;

}

 

private void VisitCell(Queue<Cell> visitedCells,

      int row, int column, int distance)

{

      if (this.maze[row, column] != 'x')

      {

            // The cell is free --> visit it

            maze[row, column] = 'x';

            Cell cell = new Cell(row, column, distance);

            visitedCells.Enqueue(cell);

      }

}

Проверка след стъпка 3

Преди да се захванем със следващата стъпка, трябва да тестваме, за да проверим нашия алгоритъм. Трябва да пробваме нормалния случай, както и граничните случаи, когато няма изход, когато се намираме на изход, когато входният файл не съществува или квадратната матрица е с размер нула. Едва след това може да започнем решаването на следващата стъпка.

Нека да пробваме случая, в който имаме дължина нула на квадратната матрица във входния файл:

Unhandled Exception: System.NullReferenceException: Object reference not set to an instance of an object.

 at Maze.FindShortestPath()

Допуснали сме грешка. Проблемът е в това, че при създаване на обект от класа Maze, променливата, в която ще помним началната клетка, се инициализира с null. Ако лабиринтът няма клетки (дължина 0) или липсва стартовата клетка, би трябвало програмата да връща резултат -1, а не да дава изключение. Можем да добавим проверка в началото на метода FindShortestPath():

public int FindShortestPath()

{

      if (this.startCell == null)

      {

            // Start cell is missing -> no path

            return -1;

      }

     

В останалите случаи изглежда, че алгоритъмът работи.

Стъпка 4 – записване на резултата във файл

Остава да запишем резултата от метода FindShortestWay() в изходния файл. Това е тривиална задача:

public void SaveResult(String fileName, int result)

{

      using (StreamWriter writer = new StreamWriter(fileName))

      {

            writer.WriteLine("The shortest way is: " + result);

      }

}

Ето как изглежда пълният код на решението на задачата:

Maze.cs

using System.IO;

using System.Collections.Generic;

 

public class Maze

{

      private const string InputFileName = "Problem2.in";

      private const string OutputFileName = "Problem2.out";

 

      public class Cell

      {

            public int Row { get; set; }

            public int Column { get; set; }

            public int Distance { get; set; }

 

            public Cell(int row, int column, int distance)

            {

                  this.Row = row;

                  this.Column = column;

                  this.Distance = distance;

            }

      }

 

      private char[,] maze;

      private int size;

      private Cell startCell = null;

 

      public void ReadFromFile(string fileName)

      {

            using (StreamReader reader = new StreamReader(fileName))

            {

                  // Read maze size and create maze

                  this.size = int.Parse(reader.ReadLine());

                  this.maze = new char[this.size, this.size];

 

                  // Read the maze cells from the file

                  for (int row = 0; row < this.size; row++)

                  {

                        string line = reader.ReadLine();

                        for (int col = 0; col < this.size; col++)

                        {

                              this.maze[row, col] = line[col];

                              if (line[col] == '*')

                              {

                                    this.startCell = new Cell(row, col, 0);

                              }

                        }

                  }

            }

      }

 

      public int FindShortestPath()

      {

            if (this.startCell == null)

            {

                  // Start cell is missing -> no path

                  return -1;

            }

 

            // Queue for traversing the cells in the maze

            Queue<Cell> visitedCells = new Queue<Cell>();

            VisitCell(visitedCells, this.startCell.Row,

                  this.startCell.Column, 0);

 

            // Perform Breath-First-Search (BFS)

            while (visitedCells.Count > 0)

            {

                  Cell currentCell = visitedCells.Dequeue();

                  int row = currentCell.Row;

                  int column = currentCell.Column;

                  int distance = currentCell.Distance;

                  if ((row == 0) || (row == size - 1)

                        || (column == 0) || (column == size - 1))

                  {

                        // We are at the maze border

                        return distance + 1;

                  }

 

                  VisitCell(visitedCells, row, column + 1, distance + 1);

                  VisitCell(visitedCells, row, column - 1, distance + 1);

                  VisitCell(visitedCells, row + 1, column, distance + 1);

                  VisitCell(visitedCells, row - 1, column, distance + 1);

            }

 

            // We didn't reach any cell at the maze border -> no path

            return -1;

      }

 

      private void VisitCell(Queue<Cell> visitedCells,

            int row, int column, int distance)

      {

            if (this.maze[row, column] != 'x')

            {

                  // The cell is free --> visit it

                  maze[row, column] = 'x';

                  Cell cell = new Cell(row, column, distance);

                  visitedCells.Enqueue(cell);

            }

      }

 

      public void SaveResult(String fileName, int result)

      {

            using (StreamWriter writer = new StreamWriter(fileName))

            {

                  writer.WriteLine("The shortest way is: " + result);

            }

      }

 

      public static void Main()

      {

            Maze maze = new Maze();

            maze.ReadFromFile(InputFileName);

            int pathLength = maze.FindShortestPath();

            maze.SaveResult(OutputFileName, pathLength);

      }

}

Тестване на решението на задачата

След като имаме решение на задачата трябва да тестваме. Вече тествахме граничните случаи и случаи като липса на изход или началната позиция да е на изхода. Видяхме, че алгоритъмът работи коректно.

Остава да тестваме с голям лабиринт, например 1000 на 1000. Можем да си направим такъв лабиринт много лесно – с copy/paste. Изпълняваме теста и се убеждаваме, че програмата работи коректно за големия тест и работи изключително бързо – не се усеща каквото и да е забавяне.

При тестването трябва да се опитваме по всякакъв начин да счупим нашето решение. Пускаме още няколко по-трудни примера (примерно лабиринт с проходими клетки във формата на спирала). Можем да сложим голям лабиринт с много пътища, но без изход. Можем да сложим и каквото още се сетим.

Накрая се убеждаваме, че имаме коректно решение и преминаваме към следващата задача.

Задача 3: Магазин за авточасти

Фирма планира създаване на система за управление на магазин за авто­части. Една част може да се използва при различни модели автомо­били и има следните характеристики:

Код, наименование, категория (за ходовата част, гуми и джанти, за двигателя, аксесоари и т.н.), покупна цена, продажна цена, списък с модели автомобили, за които може да се използва (даден автомобил се описва с марка, модел и година на производство, примерно Mercedes C320, 2008), фирма-производител.

Фирмите-производители се описват с наименование, държава, адрес, телефон и факс.

Да се проектира съвкупност от класове с връзки между тях, които моделират данните за магазина. Да се напише демонстрационна програма, която показва коректната работа на всички класове.

Измисляне на идея за решение

От нас се изисква да създадем съвкупност от класове и връзки между тях, които да описват данните за магазина. Трябва да разберем кои съществи­телни са важни за решаването на задачата. Те са обекти от реалния свят, на които съответстват класове.

Кои са тези съществителни, които ни интересуват? Имаме магазин, авточасти, автомобили и фирми-производители. Трябва да създадем клас описващ магазин. Той ще се казва Shop. Другите класове съответно са Part, Car и Manufacturer. В условието на задачата има и други съществи­телни, например код на една част или година на производство на дадена кола. За тези съществителни няма да създаваме отделни класове, а вместо това ще бъдат полета в създадените от нас класове. Например в класа Part ще има примерно поле code от тип string.

Вече знаем кои ще са нашите класове, както и полетата, които ги описват. Остава да си изясним връзките между обектите.

Каква структури от данни да използване, за да опишем връзката между два класа?

За да опишем връзката между два класа можем да използваме масив. При масива имаме достъп до елементите му по индекс, но веднъж след като го създадем не можем да му променяме дължината. Това го прави неудобен за нашата задача, понеже не знаем колко части ще имаме в магазина и по всяко време може да докарат още части или някой да купи някоя част и да се наложи да я изтрием или променим.

По-удобен е List<T>. Той притежава предим­ствата на масив, а освен това е с променлива дължина и с него лесно се реализира въвеж­дане и изтриване на елементи.

Засега изглежда, че List<T> е най-подходящ. За да се убедим ще разгледаме още няколко структури от данни. Например хеш-таблица – не е удобна в този случаи, понеже структурата "части" не от типа ключ-стойност. Тя би била подходяща, ако в магазина всяка част има уникален номер (например баркод). Тогава ще можем да ги търсим по този уникален номер. Структури като стек и опашка са неуместни.

Структурата "множество" и нейната имплементация HashSet се ползва, когато имаме уникалност по даден ключ. Може би на места ще е добра да ползваме тази структура, за да избегнем повторения. Трябва да имаме предвид, че ползването на HashSet<T> изисква да имаме методи GetHashCode() и Equals(), дефинирани коректно в типа T.

В крайна сметка избираме да ползваме List<T> и HashSet<T>.

Разделяне на задачата на подзадачи

Сега остава да си изясним въпроса от къде да започнем написването на задачата. Ако започнем да пишем класа Shop, ще се нуждаем от класа Part. Това ни подсеща, че трябва да започнем от клас, който не зависи от другите. Ще разделим написването на всеки клас на подзадача, като ще започнем от независещите от другите класове:

-    Клас описващ автомобил – Car

-    Клас описващ производител на части – Manufacturer

-    Клас описващ част за автомобили – Part

-    Клас за магазина – Shop

-    Клас за тестване на останалите класове с примерни данни – TestShop

Имплементиране: стъпка по стъпка

Започваме написването на класовете, които сме описали в нашата идея. Ще ги създаваме в реда, по който са изброени в списъка.

Стъпка 1: класът Car

Започваме решаването на задачата с дефинирането на класа Car. В дефиницията имаме три полета, които показват производителя, модела и годината на производство на една кола и стандартния метод ToString(), който връща низ с информация за дадена кола. Дефинираме го по след­ния начин:

Car.cs

public class Car

{

      private string brand;

      private string model;

      private string productionYear;

 

      public Car(string brand, string model, string productionYear)

      {

            this.brand = brand;

            this.model = model;

            this.productionYear = productionYear;

      }

 

      public override string ToString()

      {

            return "<" + this.brand + "," + this.model + ","

                  + this.productionYear + ">";

      }

}

Стъпка 2: класът Manufacturer

Следва да реализираме дефиницията на класа Manufacturer, който описва производителя на дадена част. Той ще има пет полета – име, държава, адрес, телефонен номер и факс. Ще предефинираме стандартния метод ToString(), с който ще представяме цялата информацията за дадена инстанция на класа Manufacturer.

Manufacturer.cs

public class Manufacturer

{

      private string name;

      private string country;

      private string address;

      private string phoneNumber;

      private string fax;

 

      public Manufacturer(string name, string country,

            string address, string phoneNumber, string fax)

      {

            this.name = name;

            this.country = country;

            this.address = address;

            this.phoneNumber = phoneNumber;

            this.fax = fax;

      }

 

      public override string ToString()

      {

            return this.name + " <" + this.country + "," + this.address

                  + "," + this.phoneNumber + "," + this.fax + ">";

      }

}

Стъпка 3: класът Part

Сега трябва да дефинираме класа Part. Дефиницията му ще включва следните полета – име, код, категория, списък с коли, с които може да се използва дадената част, начална и крайна цена и производител. Тук вече ще използваме избраната от нас структура от данни HashSet<T>. В случая ще бъде HashSet<Car>. Полето показващо производителя на частта ще бъде от тип Manufacturer, защото задача изисква да се помни допълнителна информация за производителя. Ако се искаше да се знае само името на производителя (както случая с класа Car) нямаше да има нужда от този клас. Щяхме да имаме поле от тип string. За полето, което описва категорията на частта ще използваме enum:

PartCategory.cs

public enum PartCategory

{

      Engine,

      Tires,

      Exhaust,

      Suspention,

      Brakes

}

Нужен ни е метод за добавяне на кола (обект от тип Car) в списъка с колите (в HashSet<Car>). Той ще се казва AddSupportedCar(Car car). Ето го и кода на класа Part:

Part.cs

public class Part

{

      private String name;

      private String code;

      private PartCategory category;

      private HashSet<Car> supportedCars;

      private double buyPrice;

      private double sellPrice;

      private Manufacturer manufacturer;

 

      public Part(string name, double buyPrice, double sellPrice,

            Manufacturer manufacturer, string code,

            PartCategory category)

      {

            this.name = name;

            this.buyPrice = buyPrice;

            this.sellPrice = sellPrice;

            this.manufacturer = manufacturer;

            this.code = code;

            this.category = category;

            this.supportedCars = new HashSet<Car>();

      }

 

      public void AddSupportedCar(Car car)

      {

            this.supportedCars.Add(car);

      }

 

      public override string ToString()

      {

            StringBuilder result = new StringBuilder();

            result.Append("Part: " + this.name + "\n");

            result.Append("-code: " + this.code + "\n");

            result.Append("-category: " + this.category + "\n");

            result.Append("-buyPrice: " + this.buyPrice + "\n");

            result.Append("-sellPrice: " + this.sellPrice + "\n");

            result.Append("-manufacturer: " +

                  this.manufacturer + "\n");

            result.Append("---Supported cars---" + "\n");

            foreach (Car car in this.supportedCars)

            {

                  result.Append(car);

                  result.Append("\n");

            }

            result.Append("----------------------\n");

            return result.ToString();

      }

}

Понеже в Part ползваме HashSet<Car> е необходимо да предефинираме методите GetHashCode() и Equals() за класа Car:

public override int GetHashCode()

{

      const int prime = 31;

      int result = 1;

      result = prime * result + ((this.brand == null) ? 0 :

            this.brand.GetHashCode());

      result = prime * result + ((this.model == null) ? 0 :

            this.model.GetHashCode());

      result = prime * result + ((this.productionYear == null) ? 0 :

            this.productionYear.GetHashCode());

      return result;

}

 

public override bool Equals(Object obj)

{

      if (this == obj)

      {

            return true;

      }

 

      if (obj == null)

      {

            return false;

      }

 

      if (this.GetType() != obj.GetType())

      {

            return false;

      }

 

      Car other = (Car)obj;

      if (this.brand == null)

      {

            if (other.brand != null)

            {

                  return false;

            }

      }

      else if (!this.brand.Equals(other.brand))

      {

            return false;

      }

 

      if (this.model == null)

      {

            if (other.model != null)

            {

                  return false;

            }

      }

      else if (!this.model.Equals(other.model))

      {

            return false;

      }

 

      if (this.productionYear == null)

      {

            if (other.productionYear != null)

            {

                  return false;

            }

      }

      else if (!this.productionYear.Equals(other.productionYear))

      {

            return false;

      }

 

      return true;

}

Стъпка 4: класът Shop

Вече имаме всички нужни класове за създаване на класа Shop. Той ще има две полета – име и списък от части, които се продават. Списъкът ще бъде List<Part>. Ще добавим метода AddPart(Part part), чрез който ще добавяме нова част. С предефинирания ToString() ще отпечатваме името на магазина и частите в него. Ето примерна реали­зация:

Shop.cs

public class Shop

{

      private string name;

      private List<Part> parts;

 

      public Shop(string name)

      {

            this.name = name;

            this.parts = new List<Part>();

      }

 

      public void AddPart(Part part)

      {

            this.parts.Add(part);

      }

 

      public override string ToString()

      {

            StringBuilder result = new StringBuilder();

            result.Append("Shop: " + this.name + "\n\n");

            foreach (Part part in this.parts)

            {

                  result.Append(part);

                  result.Append("\n");

            }

            return result.ToString();

      }

}

Стъпка 5: класът ТestShop

Създадохме всички нужни класове. Остава да създадем още един, с който да демонстрираме използването на всички останали класове. Той ще се казва ТestShop. В Main() метода ще създадем два производителя и няколко коли. Ще ги добавим към две части. Частите ще добавим към обект от тип Shop. Накрая ще отпечатаме всичко на конзолата. Ето при­мерния код:

TestShop.cs

public class TestShop

{

      public static void Main()

      {

            Manufacturer bmw = new Manufacturer("BWM",

                  "Germany", "Bavaria", "665544", "876666");

            Manufacturer lada = new Manufacturer("Lada",

                  "Russia", "Moscow", "653443", "893321");

 

            Car bmw316i = new Car("BMW", "316i", "1994");

            Car ladaSamara = new Car("Lada", "Samara", "1987");

            Car mazdaMX5 = new Car("Mazda", "MX5", "1999");

            Car mercedesC500 = new Car("Mercedes", "C500", "2008");

            Car trabant = new Car("Trabant", "super", "1966");

            Car opelAstra = new Car("Opel", "Astra", "1997");

 

            Part cheapPart = new Part("Tires 165/50/13", 302.36,

                  345.58, lada, "T332", PartCategory.Tires);

            cheapPart.AddSupportedCar(ladaSamara);

            cheapPart.AddSupportedCar(trabant);

 

            Part expensivePart = new Part("BMW Engine Oil",

                  633.17, 670.0, bmw, "Oil431", PartCategory.Engine);

            expensivePart.AddSupportedCar(bmw316i);

            expensivePart.AddSupportedCar(mazdaMX5);

            expensivePart.AddSupportedCar(mercedesC500);

            expensivePart.AddSupportedCar(opelAstra);

 

            Shop newShop = new Shop("Tunning shop");

            newShop.AddPart(cheapPart);

            newShop.AddPart(expensivePart);

 

            Console.WriteLine(newShop);

      }

}

Това е резултатът от изпълнението на нашата програма:

Shop: Tunning shop

 

Part: Tires 165/50/13

-code: T332

-category: TIRES

-buyPrice: 302.36

-sellPrice: 345.58

-manufacturer: Lada <Russia,Moscow,653443,893321>

---Supported cars---

<Lada,Samara,1987>

<Trabant,super,1966>

----------------------

 

Part: BMW Engine Oil

-code: Oil431

-category: ENGINE

-buyPrice: 633.17

-sellPrice: 670.0

-manufacturer: BWM <Germany,Bavaria,665544,876666>

---Supported cars---

<Opel,Astra,1997>

<BMW,316i,1994>

<Mazda,MX5,1999>

<Mercedes,C500,2008>

----------------------

Тестване на решението

Накрая остава да тестваме нашата задача. Всъщност ние направихме това с класа TestShop. Това обаче не означава, че сме изтествали напълно нашата задача. Трябва да се проверят граничните случаи, например когато някои от списъците са празни. Да променим малко кода в Main() метода, за да пуснем задачата с празен списък:

TestShop.cs

public class TestShop

{

      public static void Main()

      {

            Manufacturer bmw = new Manufacturer("BWM",

                  "Germany", "Bavaria", "665544", "876666");

            Manufacturer lada = new Manufacturer("Lada",

                  "Russia", "Moscow", "653443", "893321");

           

            Car bmw316i = new Car("BMW", "316i", "1994");

            Car ladaSamara = new Car("Lada", "Samara", "1987");

            Car mazdaMX5 = new Car("Mazda", "MX5", "1999");

            Car mercedesC500 = new Car("Mercedes", "C500", "2008");

            Car trabant = new Car("Trabant", "super", "1966");

            Car opelAstra = new Car("Opel", "Astra", "1997");

 

            Part cheapPart = new Part("Tires 165/50/13", 302.36,

                  345.58, lada, "T332", PartCategory.Tires);

           

            Part expensivePart = new Part("BMW Engine Oil",

                  633.17, 670.0, bmw, "Oil431", PartCategory.Engine);

            expensivePart.AddSupportedCar(bmw316i);

            expensivePart.AddSupportedCar(mazdaMX5);

            expensivePart.AddSupportedCar(mercedesC500);

            expensivePart.AddSupportedCar(opelAstra);

           

            Shop newShop = new Shop("Tunning shop");

            newShop.AddPart(cheapPart);

            newShop.AddPart(expensivePart);

           

            Console.WriteLine (newShop);

      }

}

Резултатът от този тест е следният:

Shop: Tunning shop

 

Part: Tires 165/50/13

-code: T332

-category: TIRES

-buyPrice: 302.36

-sellPrice: 345.58

-manufacturer: Lada <Russia,Moscow,653443,893321>

---Supported cars---

----------------------

 

Part: BMW Engine Oil

-code: Oil431

-category: ENGINE

-buyPrice: 633.17

-sellPrice: 670.0

-manufacturer: BWM <Germany,Bavaria,665544,876666>

---Supported cars---

<Opel,Astra,1997>

<BMW,316i,1994>

<Mazda,MX5,1999>

<Mercedes,C500,2008>

----------------------

От резултата се вижда, че списъкът от коли на евтината част е празен. Това е и правилният изход. Следователно нашата задача изпълнява коректно граничния случай с празен списък.

Упражнения

1.  Даден входен файл mails.txt, който съдържа имена на потребители и  техните email адреси. Всеки ред от файла изглежда така:

<first name> <last name> <username>@<host>.<domain>

Има изискване за имейл адресите – <username> може да е последова­телност от латински букви (a-z, A-Z) и долна черна (_), <host> е последователност от малки латински букви (a-z), а <domain> има огра­ничение от 2 до 4 малки латински букви (a-z). Да се напише програма, която намира валидните email адреси и ги записва заедно с имената на потребителите в изходен файл validMails.txt.

2.  Даден е лабиринт, който се състои от N x N квадратчета, всяко от които може да е проходимо (0) или не (x):

x

x

x

0

x

x

0

x

0

0

0

 

0

*

0

x

0

0

x

x

x

x

0

x

0

0

0

0

0

x

0

x

0

x

x

0

В едно от квадратчетата се намира отново нашият герой Минчо (*). Две квадратчета са съседни, ако имат обща стена. Минчо може на една стъпка да преминава от едно проходимо квадратче в съседно на него проходимо квадратче. Напишете програма, която по даден лабиринт отпечатва броя на възможните изходи от лабиринта.

Входните данни се четат от текстов файл с име Problem.in. На първия ред във файла стои числото N (2 < N < 1000). На следващите N реда стоят по N символа, всеки от които е или "0" или "x" или "*". Изходът представлява едно число и трябва да се изведе във файла Problem.out.

3.  Даден е лабиринт, който се състои от N x N квадратчета, всяко от които може да е проходимо или не. Проходимите клетки съдържат малка латинска буква между "а" и "z", а непроходимите – '#'. В едно от квадратчетата се намира Минчо. То е означено с "*".

Две квадратчета са съседни, ако имат обща стена. Минчо може на една стъпка да преминава от едно проходимо квадратче в съседно на него проходимо квадратче. Когато Минчо минава през проходимите квадрат­чета, той си записва буквите от всяко квадратче. На всеки изход получава дума. Напишете програма, която по даден лабиринт отпе­чатва думите, които се образуват при всички възможни изходи от лабиринта.

a

#

#

k

m

#

z

#

a

d

a

#

a

*

m

#

#

#

#

d

#

#

#

#

r

i

f

i

d

#

#

d

#

d

#

t

Входните данни се четат от текстов файл с име Problem.in. На първия ред във файла стои числото N (2 < N < 10). На следващите N реда стоят по N символа, всеки от които е или латинска буква между "а" и "z" или "#" или "*". Изходът трябва да се изведе във файла Problem.out.

4.  Фирма планира създаване на система за управление на звукозаписна компания. Звукозаписната компания има име, адрес, собственик и из­пъл­нители. Всеки изпълнител има име, псевдоним и създадени албуми. Албумите се описват с име, жанр, година на издаване, брой на прода­дените копия и списък от песни. Песните, от своя страна се описват с име и времетраене. Да се проектира съвкупност от класове с връзки между тях, които моделират данните за звукозаписната компания. Да се реализира тестов клас, който демонстрира работата на всички останали класове.

5.  Фирма планира създаване на система за управление на компания за недвижими имоти. Компанията има име, собственик, Булстат, служи­тели и разполага със списък от имоти за продажба. Служители се опис­ват с име, длъжност и стаж. Компанията продава няколко вида имоти – апартаменти, къщи, незастроени площи и магазини. Всички те се характеризират с площ, цена на квадратен метър и местоположение. За някои от тях има допълнителна информация. За апартамента има данни за номер на етажа, дали в блока има аса