Форум программистов, компьютерный форум, киберфорум
Наши страницы

С++ для начинающих

Войти
Регистрация
Восстановить пароль
 
Рейтинг: Рейтинг темы: голосов - 10, средняя оценка - 4.60
Bulding
0 / 0 / 0
Регистрация: 13.12.2012
Сообщений: 47
#1

Рекурсия (проверка на предмет соответствия закрывающих и открывающих скобок) - C++

28.02.2013, 22:46. Просмотров 1559. Ответов 7
Метки нет (Все метки)

Пусть в алгебраической записи выражения имеется всего одна операция, умножение, обозначаемое обычным образом ( 2 сомножителя следуют непосредственно друг за другом). Выражение состоит из строки символов и скобок - ограничителей: (), [], {}. Напишите программу, которая выполняет проверку на предмет соответствия закрывающих и открывающих скобок. Например, запрещены выражения типа (ab] или a(b[c)d]. Язык программирования с++

не могу разобраться помогите
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
28.02.2013, 22:46
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Рекурсия (проверка на предмет соответствия закрывающих и открывающих скобок) (C++):

Проверить в тексте файла правильность расстановки открывающих и закрывающих скобок - C++
Помогите, пожалуйста написать программу C++ Дан текстовый файл INPUT.TXT. Проверить в тексте файла правильности расстановки открывающих и...

Применение открывающих и закрывающих тегов в С++ - C++
Здравствуйте! Подскажите, пожалуйста, для чего при описании класса применяются слова, заключенные в теги, например, ///<summary> ...

Проверка правильности расстановки скобок в строке (рекурсия) - C++
Помогите написать рекурсивную функцию, проверяющую правильность расстановки скобок в строке. Правильные скобочные структуры: () ({}) ...

Как отключить автодобавление закрывающих скобок - C++
Как отключить автодобавление закрывающих скобок в С++? (Например, после символа { или соответственно.) В окне сервис - параметры...

Проверка соответствия типу int - C++
Привет всем Вот кусок кода int kolvoChel=0; int m=0; int w=0; int k=0; cout << "Vvedite kol-vo mugchin v semie:...

Проверка баланса скобок - C++
Как задать условие в проверке баланса скобок, что если скобки окажутся НЕ пустыми, тое сть внутри них еще что-то будет (символы или ...

7
Croessmah
Ушел
Эксперт CЭксперт С++
13557 / 7707 / 872
Регистрация: 27.09.2012
Сообщений: 18,996
Записей в блоге: 3
Завершенные тесты: 1
01.03.2013, 09:31 #2
Рекурсия и рекурсивные алгоритмы
0
anmartex
...
1708 / 1201 / 496
Регистрация: 12.02.2013
Сообщений: 1,978
01.03.2013, 09:45 #3
Эх..., что то не очень мне нравиться моя реализация Но, работает.
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
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
#include <iostream>
#include <string>
#include <cstdlib>
 
bool IsRight(const std::string& text, size_t current = std::string::npos)
{
   static const std::string CBrackets = "{}[]()";
   static size_t i = 0;
 
   bool isOk = true;
 
   for (; (i < text.size()) && isOk; ++i)
   {
      size_t find = CBrackets.find_first_of(text[i]);
 
      if (find != std::string::npos)
      {
         if ((find % 2) == 0)
         {
            i++;
            isOk = IsRight(text, find);
         }
         else
         {
            isOk = ((find - current) == 1);
 
            if (current != std::string::npos)
            {
               return isOk;
            }
         }
      }
   }
 
   i = 0;
 
   return isOk;
}
 
int main(int argc, const char* argv[])
{
   std::string text;
   std::cout << "Input text: ";
   std::getline(std::cin, text);
 
   std::cout << (IsRight(text.c_str()) ? "right" : "not right") << std::endl;
 
   system("PAUSE");
 
   return 0;
}
Рекурсия (проверка на предмет соответствия закрывающих и открывающих скобок)

Исходник + бинарник: program.7z
0
Bulding
0 / 0 / 0
Регистрация: 13.12.2012
Сообщений: 47
01.03.2013, 16:15  [ТС] #4
Объясните пожалуйста с комментариями
0
kamre
126 / 130 / 4
Регистрация: 25.12.2011
Сообщений: 443
01.03.2013, 16:58 #5
Цитата Сообщение от Bulding Посмотреть сообщение
Выражение состоит из строки символов и скобок - ограничителей: (), [], {}. Напишите программу, которая выполняет проверку на предмет соответствия закрывающих и открывающих скобок. Например, запрещены выражения типа (ab] или a(b[c)d]. Язык программирования с++
Например, вот так можно на C++:
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
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
#include <boost/spirit/include/qi.hpp>
 
#include <iostream>
#include <string>
 
namespace qi = boost::spirit::qi;
 
template <typename Iterator>
struct grammar: qi::grammar<Iterator>
{
    grammar(): grammar::base_type(text)
    {
        text = +element;
        element = qi::char_('a', 'z')
                | '(' >> text >> ')'
                | '[' >> text >> ']'
                | '{' >> text >> '}'
                ;
    }
    qi::rule<Iterator> text, element;
};
 
void parse_string(const std::string & str)
{
    typedef std::string::const_iterator iter_t;
    iter_t iter = str.begin();
    iter_t end = str.end();
    grammar<iter_t> g;
    bool r = qi::parse(iter, end, g);
    bool success = r && (iter == end);
    std::cout << "parsing of string \"" << str << "\" "
              << (success ? "succeeded" : "failed") << std::endl;
}
 
int main(int argc, char **argv)
{
    parse_string("a");
    parse_string("abc");
    parse_string("(a)");
    parse_string("(ab]");
    parse_string("[a]");
    parse_string("[ab}");
    parse_string("{a}");
    parse_string("{aaaa{");
    parse_string("a(b[c)d]");
    parse_string("abc(ddd[e]ff{ggg})zzz");
    parse_string("abc(ddd[e]ff{ggg}]zzz");
    return 0;
}
Результат:
parsing of string "a" succeeded
parsing of string "abc" succeeded
parsing of string "(a)" succeeded
parsing of string "(ab]" failed
parsing of string "[a]" succeeded
parsing of string "[ab}" failed
parsing of string "{a}" succeeded
parsing of string "{aaaa{" failed
parsing of string "a(b[c)d]" failed
parsing of string "abc(ddd[e]ff{ggg})zzz" succeeded
parsing of string "abc(ddd[e]ff{ggg}]zzz" failed
0
anmartex
...
1708 / 1201 / 496
Регистрация: 12.02.2013
Сообщений: 1,978
01.03.2013, 19:46 #6
kamre, это конечно здорово но:
  1. А где у вас рекурсия? Вы название темы читали?
  2. Здорово, что вы знаете boost, но я уверен, что автор темы про него слыхом не слыхивал. И я уверен, что вы это прекрасно понимаете. Так зачем же тогда, измените за жаргон, эти понты?

Bulding, в предыдущем алгоритме есть баг. Когда сейчас начал комментировать, понял, что я немного ошибся. Переделал немного и по вашей просьбе прокомментировал основные моменты.
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
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
#include <iostream>
#include <string>
#include <cstdlib>
 
bool IsRight(const char*& text, size_t current)
{
   // Я думаю и так понятно. Закономерность такая:
   // чётные - открывающиеся, нечётные - закрывающиеся
   static const std::string CBrackets = "{}[]()";
 
   bool isOk = true;
 
   // Начинаем перебирать все символы до тех пор, пока
   // не встретим 0 символ или пока не найдётся ошибка
   for (; *text && isOk; ++text)
   {
      // Пытаемся найти текущий символ в нашем списке скобок
      size_t find = CBrackets.find_first_of(*text);
 
      if (find != std::string::npos)
      {
         // В том случае если у нас открывающаяся скобка
         // то нужно искать закрывающуюся, т.е. вызываем
         // заново эту же функцию.
         if ((find % 2) == 0)
         {
            isOk = IsRight(++text, find);
         }
         // Если это закрывающаяся скобка
         else
         {
            // Ошибка будет в том случае, если открывающаяся
            // и закрывающаяся стоят не рядом.
            isOk = ((find - current) == 1);
 
            // current может быть равным std::string::npos
            // только в одном случае, если это самый первый вызов
            // функции. Т.е. это основной цикл поиска скобок.
            // А его как раз нужно выполнить до конца, ну а если это какой-то
            // повторный вызов и мы нашли и открывающуюся и закрывающуюся,
            // то просто возвращаем результат нашего поиска
            if (current != std::string::npos)
            {
               return isOk;
            }
         }
      }
   }
 
   // Если это был не самый первый вызов функции,
   // значит пара не была найдена, т.е. перебрали всё
   // а закрывающейся на каком-то этапе не выявили, значит
   // всё - провал
   if (current != std::string::npos)
   {
      isOk = false;
   }
 
   return isOk;
}
 
bool IsRight(const std::string& text)
{
   const char* txt = text.c_str();
 
   return IsRight(txt, std::string::npos);
}
 
int main(int argc, const char* argv[])
{
   std::string text;
   std::cout << "Input text: ";
   std::getline(std::cin, text);
 
   std::cout << (IsRight(text) ? "right" : "not right") << std::endl;
 
   system("PAUSE");
 
   return 0;
}
0
kamre
126 / 130 / 4
Регистрация: 25.12.2011
Сообщений: 443
01.03.2013, 20:30 #7
Цитата Сообщение от anmartex Посмотреть сообщение
kamre, это конечно здорово но:
  1. А где у вас рекурсия? Вы название темы читали?
  2. Здорово, что вы знаете boost, но я уверен, что автор темы про него слыхом не слыхивал. И я уверен, что вы это прекрасно понимаете. Так зачем же тогда, измените за жаргон, эти понты?
  1. Boost Spirit is an object-oriented, recursive-descent parser and output generation library for C++.
  2. Да какие понты, просто пример немного другого подхода (более общего), где грамматика для языка выражений записана в явном виде. Если потребуется расширить грамматику (например, ввести */+- ) или извлечь имена переменных, то более общий подход будет удобнее.
0
dauren
0 / 0 / 0
Регистрация: 03.11.2014
Сообщений: 6
04.11.2014, 13:54 #8
люди можете облехчить код,, используя только iostream и без ::
0
04.11.2014, 13:54
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
04.11.2014, 13:54
Привет! Вот еще темы с ответами:

Проверка на расстановку скобок - C++
В общем задание такое. Программа (консольная) проверяет каждую строку файла для проверки на расстановку скобок 4 типов (&quot;(&quot; и &quot;)&quot;, &quot;&quot;,...

Проверка правильности расстоновки скобок - C++
Дана страка. Проверить правильность расстановки скобок (в страке приутствуют только круглые скобки). Вот мой текст, программа не правильно...

Работа со строками на С++ проверка на согласованность скобок. - C++
В арифметическом выражении проверить согласованность скобок, правильность их расположения и выдать соответствующее сообщение. Если можно...

Работа со строками в С++ проверка на согласованность скобок - C++
В арифметическом выражении проверить согласованность скобок, правильность их расположения и выдать соответствующее сообщение. Если можно...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
8
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.