IPB
ЛогинПароль:

> Прочтите прежде чем задавать вопрос!

1. Заголовок темы должен быть информативным. В противном случае тема удаляется ...
2. Все тексты программ должны помещаться в теги [code=pas] ... [/code].
3. Прежде чем задавать вопрос, см. "FAQ", если там не нашли ответа, воспользуйтесь ПОИСКОМ, возможно такую задачу уже решали!
4. Не предлагайте свои решения на других языках, кроме Паскаля (исключение - только с согласия модератора).
5. НЕ используйте форум для личного общения, все что не относится к обсуждению темы - на PM!
6. Одна тема - один вопрос (задача)
7. Проверяйте программы перед тем, как разместить их на форуме!!!
8. Спрашивайте и отвечайте четко и по существу!!!

> Задача на проверку баланса скобок в строке, я решение знаю, предложи свое
Флогримм
сообщение 27.10.2004 6:49
Сообщение #1


Бывалый
***

Группа: Пользователи
Сообщений: 253
Пол: Мужской

Репутация: -  4  +


вот задачку недавно решил, так понравилась!

Задача.
В строке могут содержаться скобки '}',']',')','{','[','('. Прверить баланс скобок в строке. Считать, что он соблюдается, если: 1) для каждой открывающей скобки есть своя закрывающая и наоборот; 2) соблюдается вложеность; На выходе вывести 'false' или 'true' в зависимости от ответа.
пример:
Код
[(a+B)+c{b-a}]*{(a+c)(c-a)} - баланс скобок соблюден
[(a+B)+c{b-a] - баланс не соблюден: нет закрывающей скобки '}'
[(a+B)+c{b-a]} - баланс не соблюден: неправильная вложенность


какие предложения, господа?

Сообщение отредактировано: Pioneer - 27.10.2004 6:51


--------------------
Я не буду жить с этой злобой внутри / Я не буду частью смертельной цепи / Я не буду потребителем твоих идей / Я не буду никогда убивать зверей (Unconform)
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
 
 Ответить  Открыть новую тему 
Ответов(1 - 7)
xds
сообщение 27.10.2004 11:30
Сообщение #2


N337
****

Группа: Пользователи
Сообщений: 737
Пол: Мужской

Репутация: -  26  +


Цитата
вот задачку недавно решил, так понравилась!

Это круто! Положительные эмоции - один из важнейших инструментов разработки :D

1. Классика:

var
  s: String;
  Stack: array[0..127] of Char;
  sp, i: Integer;

procedure Push(c: Char);
begin
  if sp < 0 then    begin      Writeln(False);      Halt;    end;
  Stack[sp] := c;
  Dec(sp);
end;

begin
  Write('s>');
  Readln(s);
  Stack[127] := #0;
  sp := 126;
  for i := 1 to Length(s) do
    case s[i] of
    '(': Push(')');
    '[': Push(']');
    '{': Push('}');
    ')', ']', '}':
      begin
        Inc(sp);
        if (s[i] <> Stack[sp]) then     begin     Writeln(False);       Halt;          end;
      end;
    end;
  Writeln(sp = 126);
end.



2. С использованием динамической памяти:

type
  PItem = ^TItem;
  TItem = record
    Next: PItem;
    c: Char;
  end;

var
  s: String;
  sp, p: PItem;
  i: Integer;

procedure Push(c: Char);
begin
  New(p);
  p^.Next := sp;
  p^.c := c;
  sp := p;
end;

begin
  Write('s>');
  Readln(s);
  sp := nil;
  Push(#0);
  for i := 1 to Length(s) do
    case s[i] of
    '(': Push(')');
    '[': Push(']');
    '{': Push('}');
    ')', ']', '}':
      begin
        if (s[i] <> sp^.c) then
          begin
            Writeln(False);
            Halt;
          end;
        p := sp;
        sp := sp^.Next;
        Dispose(p);
      end;
    end;
  Writeln(sp^.c = #0);
end.



3.
Цитата("Справочник практического врача (под ред. проф. И. Г. Кочергина) @ издание второе, М.: Медицина, 1969")
БРЕДОВОЙ СИНДРОМ. Бред - ложное, возникшее на болезненной основе не соответствующее действительности, полностью овладевающее сознанием больного умозаключение (суждение), не поддающееся исправлению никакими доводами... Бред - несомненный признак психического заболевания...


type
  TCheck = procedure;

var
  s: String;
  Stack: array[0..127] of Char;
  sp, i: Integer;

procedure Push(c: Char);
begin
  if sp < 0 then    begin      Writeln(False);      Halt(0);    end;
  Stack[sp] := c;
  Dec(sp);
end;

procedure Push1;
begin
  Push(')');
end;

procedure Push2;
begin
  Push(']');
end;

procedure Push3;
begin
  Push('}');
end;

procedure Pop;
begin
  Inc(sp);
  if (s[i] <> Stack[sp]) then    begin      Writeln(False);      Halt(0);    end;
end;

procedure Null; begin end;

var
  Check: array[Char] of Pointer;

begin
  Write('s>');
  Readln(s);

  for i := 0 to 255 do
    Check[Chr(i)] := @Null;
  Check['('] := @Push1;
  Check['['] := @Push2;
  Check['{'] := @Push3;
  Check['}'] := @Pop;
  Check[')'] := @Pop;
  Check[']'] := @Pop;

  Stack[127] := #0;
  sp := 126;
  for i := 1 to Length(s) do
    TCheck(Check[s[i]]);
  Writeln(sp = 126);
end.



Сообщение отредактировано: Altair - 26.02.2006 15:21
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
APAL
сообщение 27.10.2004 11:44
Сообщение #3


Смотрю...
*****

Группа: Модераторы
Сообщений: 1 055
Пол: Мужской
Реальное имя: Пшеничный Алексей Анатольевич

Репутация: -  6  +


Вариант решения через работу со строкой:
Function CheckS(St : String) : Boolean;
Var i : Byte;
    s : String;
Begin
  CheckS:=False;
  s:='';
  For i:=1 to Length(St) do
    Case St[i] of
     '{','}','[',']','(',')' : s:=s+St[i];
    End;
  For i:=1 to Length(s) do
  Begin
    If Pos('{}',s)<>0 then Delete(s,Pos('{}',s),2);
    If Pos('[]',s)<>0 then Delete(s,Pos('[]',s),2);
    If Pos('()',s)<>0 then Delete(s,Pos('()',s),2);
  End;
  If s='' then CheckS:=True;
End;


P.S.: А вобще-то я хотел сделать через рекурсию... но что-то не пошло - надо немного подредактировать.
Но зато выше представленный вариант короче получился... :D

Сообщение отредактировано: Altair - 26.02.2006 15:22
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
xds
сообщение 27.10.2004 12:46
Сообщение #4


N337
****

Группа: Пользователи
Сообщений: 737
Пол: Мужской

Репутация: -  26  +


Немного тяжеловато, но оригинально lol.gif


--------------------
The idiots are winning.
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
xds
сообщение 27.10.2004 13:08
Сообщение #5


N337
****

Группа: Пользователи
Сообщений: 737
Пол: Мужской

Репутация: -  26  +


У меня с рекурсией получилось так:


var
  s: String;
  i: Integer;

procedure Rec(c: Char);
begin
  Inc(i);
  while i <= Length(s) do
    begin
      case s[i] of
      '(': Rec(')');
      '[': Rec(']');
      '{': Rec('}');
      ')', ']', '}':
        if s[i] = c then
          Exit
        else
          begin
            Writeln(False);
            Halt;
          end;
      end;
      Inc(i);
    end;
  Writeln(c = #0);
  Halt;
end;

begin
  Write('s>');
  Readln(s);
  i := 0;
  Rec(#0);
end.



По сути этот вариант аналогичен первому, только использует системный стек, причем тратит место на выравнивание параметра, адрес возврата и сохранение BP (по 7 дополнительных байт на каждую открывающуюся скобку). Лучше всего на ассемблере и без рекурсии smile.gif

Сообщение отредактировано: Altair - 26.02.2006 15:27


--------------------
The idiots are winning.
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
Altair
сообщение 27.10.2004 19:24
Сообщение #6


Ищущий истину
******

Группа: Модераторы
Сообщений: 4 824
Пол: Мужской
Реальное имя: Олег

Репутация: -  45  +


Так, так, так.... ИМХО тут в половине решений, не проверяется на вложенность!
(вообще-то я код не читал, просто пробежал глазами, так что если я не прав, можете поспорить smile.gif )

({} () <(){}> )


--------------------
Помогая друг другу, мы справимся с любыми трудностями!
"Не опускать крылья!" (С)
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
xds
сообщение 28.10.2004 3:18
Сообщение #7


N337
****

Группа: Пользователи
Сообщений: 737
Пол: Мужской

Репутация: -  26  +


Цитата
Код
[(a+B)+c{b-a]} - баланс не соблюден: неправильная вложенность
- единственное указание на то, что в данном случае называется "вложенностью": несоответствие парности скобок. Если под этим термином подразумевалось что-либо другое, например, порядок вложения пар "{}", "[]" и "()", то это следовало указать в условии.

Кстати, твой пример помог обнаружить ошибку в третьем ("бредовом") решении - в процедуре Push3: вместо '}' стояло ']'. После исправлений все четыре моих решения для указанного тобой случая возвращают TRUE. 10x smile.gif

Сообщение отредактировано: xds - 28.10.2004 3:31


--------------------
The idiots are winning.
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
APAL
сообщение 28.10.2004 9:55
Сообщение #8


Смотрю...
*****

Группа: Модераторы
Сообщений: 1 055
Пол: Мужской
Реальное имя: Пшеничный Алексей Анатольевич

Репутация: -  6  +


У меня тоже со вложенностью все в порядке. smile.gif


--------------------
Если что-то не делает того, что вы запланировали ему делать - это еще не означает, что оно бесполезно.
--------------------
Прежде, чем задать вопрос - Правила :: FAQ :: Поиск
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 

 Ответить  Открыть новую тему 
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 

- Текстовая версия 26.07.2025 1:12
Хостинг предоставлен компанией "Веб Сервис Центр" при поддержке компании "ДокЛаб"