Форум по Delphi программированию

Delphi Sources



Вернуться   Форум по Delphi программированию > Все о Delphi > [ "Начинающим" ]
Ник
Пароль
Регистрация <<         Правила форума         >> FAQ Пользователи Календарь Поиск Сообщения за сегодня Все разделы прочитаны

Ответ
 
Опции темы Поиск в этой теме Опции просмотра
  #1  
Старый 16.11.2012, 11:04
Аватар для morebeauty
morebeauty morebeauty вне форума
Начинающий
 
Регистрация: 21.06.2012
Сообщения: 106
Версия Delphi: Delphi 7
Репутация: 10
По умолчанию Остановка выполнения функции/процедуры из другой (дочерней) функции

Сей вопрос не особо важен и вызван чрезмерной ленью.

Суть:
Есть некоторая функция проверки условия. Вызывается из целого ряда других функций и процедур. Выдает сообщение об ошибке (если условие не выполнилось).

Вопрос:
Можно ли из этой функции проверки прервать выполнение той функции, из которой она была вызвана?

Мотивация:
На данный момент есть несколько функций проверки. Их результат критичен (то есть если результат отрицательный, дальше ничего выполняться не должно). В большинстве случаев они используются все, но местами требуется использовать их поодиночке. Выглядит это так:

Код:
function ФункцияПроверки1:boolean;
begin
******
end;

Function ФункцияПроверки2:boolean;
begin
******
end;

Function ФункцияПроверки3:boolean;
begin
******
end;

procedure ВыполняемаяПроцедура1;
begin
if ФункцияПроверки1 then
begin
***
if ФункцияПроверки2 then
begin
***
if ФункцияПроверки3 then
begin
*****
*****
*****
end;
end;
end;
end;

procedure ВыполняемаяПроцедура2;
begin
if ФункцияПроверки1 then
begin
***
if ФункцияПроверки2 then
begin
***
if ФункцияПроверки3 then
begin
*****
*****
*****
end;
end;
end;
end;
А хотелось бы упростить и привести к виду:

Код:
procedure ПроцедураПроверки1;
begin
if Условие=false then
begin
******
завершить подпрограмму, из которой была вызвана проверка;
end;
end;

procedure ПроцедураПроверки2;
begin
if Условие=false then
begin
******
завершить подпрограмму, из которой была вызвана проверка;
end;
end;

procedure ПроцедураПроверки3;
begin
if Условие=false then
begin
******
завершить подпрограмму, из которой была вызвана проверка;
end;
end;

procedure ВыполняемаяПроцедура1;
begin
ПроцедураПроверки1; //дальше не должно пойти, если условие не выполнено
***
ПроцедураПроверки2; //дальше не должно пойти, если условие не выполнено
***
ПроцедураПроверки3; //дальше не должно пойти, если условие не выполнено
***
***
***
end;

procedure ВыполняемаяПроцедура2;
begin
ПроцедураПроверки1; //дальше не должно пойти, если условие не выполнено
***
ПроцедураПроверки2; //дальше не должно пойти, если условие не выполнено
***
ПроцедураПроверки3; //дальше не должно пойти, если условие не выполнено
***
***
***
end;
Это возможно сделать?
Ответить с цитированием
  #2  
Старый 16.11.2012, 11:10
Аватар для poli-smen
poli-smen poli-smen вне форума
Профессионал
 
Регистрация: 06.08.2012
Адрес: Кривой Рог
Сообщения: 1,791
Версия Delphi: Delphi 7, XE2
Репутация: 4415
По умолчанию

Цитата:
Сообщение от morebeauty
Мотивация:
На данный момент есть несколько функций проверки. Их результат критичен (то есть если результат отрицательный, дальше ничего выполняться не должно).
Вызывай исключения. (см. raise)
Ответить с цитированием
  #3  
Старый 16.11.2012, 11:15
Аватар для M.A.D.M.A.N.
M.A.D.M.A.N. M.A.D.M.A.N. вне форума
Sir Richard Abramson
 
Регистрация: 05.04.2008
Сообщения: 5,505
Версия Delphi: XE10
Репутация: выкл
По умолчанию

Abort()

или raise EAbort.Create
__________________
— Как тебя понимать?
— Понимать меня не обязательно. Обязательно меня любить и кормить вовремя.


На Delphi, увы, больше не программирую.
Рекомендуемая литература по программированию
Ответить с цитированием
  #4  
Старый 16.11.2012, 11:16
Pyro Pyro вне форума
Так проходящий
 
Регистрация: 18.07.2011
Сообщения: 805
Версия Delphi: 7Lite
Репутация: 6063
По умолчанию

Код:
if not ФункцияПроверки1 then exit;
***
if not ФункцияПроверки1 then exit;
или просто по очереди вызывать и в случае чего ошибку кидать и ловить
__________________
>woweook<
Ответить с цитированием
  #5  
Старый 16.11.2012, 11:24
Аватар для morebeauty
morebeauty morebeauty вне форума
Начинающий
 
Регистрация: 21.06.2012
Сообщения: 106
Версия Delphi: Delphi 7
Репутация: 10
По умолчанию

Цитата:
Abort()

или raise EAbort.Create
Тогда ведь вместо блоков IF придется тыкать блоки TRY? или я не догоняю чего-то?
Ответить с цитированием
  #6  
Старый 16.11.2012, 11:28
Аватар для poli-smen
poli-smen poli-smen вне форума
Профессионал
 
Регистрация: 06.08.2012
Адрес: Кривой Рог
Сообщения: 1,791
Версия Delphi: Delphi 7, XE2
Репутация: 4415
По умолчанию

Цитата:
Сообщение от morebeauty
Тогда ведь вместо блоков IF придется тыкать блоки TRY? или я не догоняю чего-то?
Да, нужно использовать блоки try...except, но не вместо if.
Ответить с цитированием
  #7  
Старый 16.11.2012, 11:39
Аватар для NumLock
NumLock NumLock вне форума
Let Me Show You
 
Регистрация: 30.04.2010
Адрес: Северодвинск
Сообщения: 5,426
Версия Delphi: 7, XE5
Репутация: 59586
По умолчанию

Код:
procedure ВыполняемаяПроцедура1;
begin
if not ФункцияПроверки1 then Exit;
***
if not ФункцияПроверки2 then Exit;
***
if not ФункцияПроверки3 then Exit;
*****
*****
*****
end;
__________________
Пишу программы за еду.
__________________
Ответить с цитированием
  #8  
Старый 16.11.2012, 11:40
Аватар для morebeauty
morebeauty morebeauty вне форума
Начинающий
 
Регистрация: 21.06.2012
Сообщения: 106
Версия Delphi: Delphi 7
Репутация: 10
По умолчанию

Цитата:
Да, нужно использовать блоки try...except, но не вместо if.
Ну тогда лучше не станет. Я лишь хотел отказаться от одинакового каркаса для множества выполняемых процедур (или максимально его упростить).

Есть еще способы? Может, есть команда, которая разом прерывает выполнения всех функций (в том числе и рекурсивных) и переходит в режим ожидания действий пользователя? Ну, типа Application.Terminate, только без полного завершения программы, а лишь завершение активности.


А потоки в этом помогут, как думаете?

Последний раз редактировалось morebeauty, 16.11.2012 в 11:43.
Ответить с цитированием
  #9  
Старый 16.11.2012, 11:43
Pyro Pyro вне форума
Так проходящий
 
Регистрация: 18.07.2011
Сообщения: 805
Версия Delphi: 7Lite
Репутация: 6063
По умолчанию

Цитата:
Сообщение от morebeauty
Ну тогда лучше не станет.
станет, нужен один try..except на все вызовы
__________________
>woweook<
Ответить с цитированием
  #10  
Старый 16.11.2012, 11:47
Аватар для M.A.D.M.A.N.
M.A.D.M.A.N. M.A.D.M.A.N. вне форума
Sir Richard Abramson
 
Регистрация: 05.04.2008
Сообщения: 5,505
Версия Delphi: XE10
Репутация: выкл
По умолчанию

Если ф-ии с одинаковыми параметрами, можно ссылки на них свалить в массив и в одном цикле прогнать.
__________________
— Как тебя понимать?
— Понимать меня не обязательно. Обязательно меня любить и кормить вовремя.


На Delphi, увы, больше не программирую.
Рекомендуемая литература по программированию
Ответить с цитированием
  #11  
Старый 16.11.2012, 11:49
Аватар для morebeauty
morebeauty morebeauty вне форума
Начинающий
 
Регистрация: 21.06.2012
Сообщения: 106
Версия Delphi: Delphi 7
Репутация: 10
По умолчанию

Код:
procedure ВыполняемаяПроцедура1;
begin
if not ФункцияПроверки1 then Exit;
***
if not ФункцияПроверки2 then Exit;
***
if not ФункцияПроверки3 then Exit;
*****
*****
*****
end;
Такое тоже рассматривал. Не подходит. Все дело в том, что на данный момент мне подойдет просто Exit, завтра будут другие задачи, послезавтра третьи. Представь, сколько придется переделывать ВыполняемыхПроцедурХ каждый раз.

Вот поэтому я и хочу свести все действия на невыполнение условия к процедуре проверки. Тогда код будет более гибким ИМХО.


Цитата:
станет, нужен один try..except на все вызовы
Я тогда не совсем представляю как это будет выглядеть. можешь изобразить?

Цитата:
Если ф-ии с одинаковыми параметрами, можно ссылки на них свалить в массив и в одном цикле прогнать.
это как?
Ответить с цитированием
  #12  
Старый 16.11.2012, 11:54
Аватар для Aristarh Dark
Aristarh Dark Aristarh Dark вне форума
Модератор
 
Регистрация: 07.10.2005
Адрес: Москва
Сообщения: 2,907
Версия Delphi: Delphi XE
Репутация: выкл
По умолчанию

Вот рабочий метод из софтины которую я сейчас пишу. Используется куча Exit'ов, вполне нормальный код, ИМХО. Ничего криминального я тут не вижу.
Код:
function TTestThread.CheckLoops: Boolean;
{$DEFINE new_alg}
{.$DEFINE old_alg}
const
  LoopWait = 5000;
  LoopsState:array [1..3,1..2] of Boolean =
    (
      (True,False),(False,True),(True,True)
    );
var
  Points:Integer;
  i: Integer;
begin
  AddToLog('Вход в процедуру CheckLoops');
  Result:=False;
  if not (Loop1(False) and  Loop2(False)) then
    begin
      AddToLog('Ошибка управления шлейфами.');
      AddToLog(Format('Выход из процедуры CheckLoops, с результатом %s',[BoolToStr(Result,True)]));
      Exit;
    end;
  Sleep(LoopWait);
  //Залить настройку
  if not WriteSettingForLoops then
    begin
      AddToLog('Ошибка настройки шлейфов');
      AddToLog(Format('Выход из процедуры CheckLoops, с результатом %s',[BoolToStr(Result,True)]));
      Exit;
    end;
  Points:=GetPoint;
  if Points<0 then
    begin
      AddToLog('Ошибка пролучения значения +point');
      AddToLog(Format('Выход из процедуры CheckLoops, с результатом %s',[BoolToStr(Result,True)]));
      Exit;
    end;
  //Сформировать события
  for i := Low(LoopsState) to High(LoopsState) do
    begin
      if not (Loop1(LoopsState[i,1]) and Loop2(LoopsState[i,2])) then
        begin
          AddToLog('Ошибка управления шлейфами.');
          AddToLog(Format('Выход из процедуры CheckLoops, с результатом %s',[BoolToStr(Result,True)]));
          Exit;
        end;
      Inc(Points);
      if i=3 then
        Inc(Points);
      Sleep(LoopWait);
{$IFDEF new_alg}
      if not (Loop1(False) and  Loop2(False)) then
        begin
          AddToLog('Ошибка управления шлейфами.');
          AddToLog(Format('Выход из процедуры CheckLoops, с результатом %s',[BoolToStr(Result,True)]));
          Exit;
        end;
{$ENDIF}
      if not CheckPointsCount(Points) then
        begin
          AddToLog('Текущее состояние истории не соответствует ожидаемому');
          FTestResult.LoopsTests[i]:=False;
        end
      else
        begin
          FTestResult.LoopsTests[i]:=True;
          AddToLog(Format('Для состояния шлейфов [%s,%s] тест прошел успешно',[BoolToStr(LoopsState[i,1]),BoolToStr(LoopsState[i,2])]));
        end;
{$IFDEF old_alg}
      if not (Loop1(False) and  Loop2(False)) then
        begin
          AddToLog('Ошибка управления шлейфами.');
          AddToLog(Format('Выход из процедуры CheckLoops, с результатом %s',[BoolToStr(Result,True)]));
          Exit;
        end;
{$ENDIF}
      Sleep(LoopWait);
    end;
  //Очистить настройки шлейфов
  if not ClearSettingsForLoops then
    begin
      AddToLog('Ошибка очистки настроек шлейфов');
      AddToLog(Format('Выход из процедуры CheckLoops, с результатом %s',[BoolToStr(Result,True)]));
      Exit;
    end;
  Result:=True;
  AddToLog(Format('Выход из процедуры CheckLoops, с результатом %s',[BoolToStr(Result,True)]));
end;
__________________
Некоторые программисты настолько ленивы, что сразу пишут рабочий код.

Если вас наказали ни за что - радуйтесь: вы ни в чем не виноваты.
Ответить с цитированием
  #13  
Старый 16.11.2012, 11:56
Аватар для NumLock
NumLock NumLock вне форума
Let Me Show You
 
Регистрация: 30.04.2010
Адрес: Северодвинск
Сообщения: 5,426
Версия Delphi: 7, XE5
Репутация: 59586
По умолчанию

Код:
unit Unit1;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls;

type
  TForm1 = class(TForm)
    Button1: TButton;
    procedure FormCreate(Sender: TObject);
    procedure Button1Click(Sender: TObject);
  private
    { Private declarations }
  public
    { Public declarations }
    procedure ExceptionEvent(Sender: TObject; E: Exception);
    procedure DoCheck;
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

procedure TForm1.FormCreate(Sender: TObject);
begin
  Application.OnException:=ExceptionEvent;
end;

procedure TForm1.ExceptionEvent(Sender: TObject; E: Exception);
begin
//  ShowMessage(E.Message);
end;

procedure TForm1.DoCheck;
begin
  if True then raise Exception.Create('Oops');
end;

procedure TForm1.Button1Click(Sender: TObject);
begin
  DoCheck;
end;

end.

но с Exit красивее
__________________
Пишу программы за еду.
__________________
Ответить с цитированием
  #14  
Старый 16.11.2012, 12:00
Аватар для morebeauty
morebeauty morebeauty вне форума
Начинающий
 
Регистрация: 21.06.2012
Сообщения: 106
Версия Delphi: Delphi 7
Репутация: 10
По умолчанию

Код:
procedure TForm1.ExceptionEvent(Sender: TObject; E: Exception);
begin
//  ShowMessage(E.Message);
end;
И как мне из этой процедуры завершить ту, которая в процессе выполнения?

Вы извините меня, я не очень шарю. Мануалы не всегда помогают. В этом, например, случае. Поэтому и спрашиваю.
Ответить с цитированием
  #15  
Старый 16.11.2012, 12:01
Аватар для poli-smen
poli-smen poli-smen вне форума
Профессионал
 
Регистрация: 06.08.2012
Адрес: Кривой Рог
Сообщения: 1,791
Версия Delphi: Delphi 7, XE2
Репутация: 4415
По умолчанию

Цитата:
Сообщение от morebeauty
Есть еще способы? Может, есть команда, которая разом прерывает выполнения всех функций (в том числе и рекурсивных) и переходит в режим ожидания действий пользователя?
Исключение "разом прерывает выполнения всех функций (в том числе и рекурсивных)" вплоть до последнего обработчика исключения except.

В примере ниже "Beep" никогда не вызовется, так как первый же "Abort" прервёт одним махом все вложенные (рекурсивные) вызовы и перейдёт "в режим ожидания действий пользователя" (если конечно не встретится другого обработчика except):
Код:
procedure TForm1.Button4Click(Sender: TObject);

  procedure Recursion(Level: Integer);
  begin
    if Level = 100 then Abort;
    Recursion(Level + 1);
    Beep;
  end;

begin
  Recursion(0);
end;
Ответить с цитированием
Этот пользователь сказал Спасибо poli-smen за это полезное сообщение:
morebeauty (19.11.2012)
Ответ


Delphi Sources

Опции темы Поиск в этой теме
Поиск в этой теме:

Расширенный поиск
Опции просмотра

Ваши права в разделе
Вы не можете создавать темы
Вы не можете отвечать на сообщения
Вы не можете прикреплять файлы
Вы не можете редактировать сообщения

BB-коды Вкл.
Смайлы Вкл.
[IMG] код Вкл.
HTML код Выкл.
Быстрый переход


Часовой пояс GMT +3, время: 13:49.


 

Сайт

Форум

FAQ

Соглашения

Прочее

 

Copyright © Форум "Delphi Sources" by BrokenByte Software, 2004-2025