- Feature Name: Finally for handled_sequence_of_statements
- Start Date: 2023-02-13
Add a finally
part to handled_sequence_of_statements
. The statements
will be executed unconditionally, whichever path the execution takes in the
handled_sequence_of_statements
.
Ada is a language with exceptions, but there is no way to have unconditional
finalization of objects, apart from using either goto_statements
, which are a
verbose workaround, and controlled objects, which impose performance penalties
and are not the most expressive way in some situations.
We propose to add a common construct in languages with exceptions, which is the
finally
block. Every statement in the finally
block of statements will be
executed, regardless of whether an exception has been caught by the exception
block or not, allowing to run finalization actions.
procedure Example is
F : File_Type;
File_Name : constant String := "simple.txt";
Bar : Integer := Function_That_Might_Raise;
begin
Create (F, Out_File, File_Name);
Put_Line (F, "Hello World #1");
Put_Line (F, "Hello World #2");
Put_Line (F, "Hello World #3");
finally
Close (F);
end Example;
The grammar is extended as follows:
handled_sequence_of_statements ::=
sequence_of_statements
[exception
exception_handler
{exception_handler}]
[finally
sequence_of_statements]
No specific amendment for name resolution, the sequence_of_statements
after
the finally block is resolved as any other sequence of statements.
-
Return statements in the
sequence_of_statements
attached to the finally are forbidden. -
Goto & continue where the target is outside of the finally's
sequence_of_statements
are forbidden
- Statements in the optional
sequence_of_statements
contained in thefinally
branch will be executed unconditionally, after the mainsequence_of_statements
is executed, and after any potentialexception_handler
is executed.
Note
This is added as a note, because it follows naturally from the syntactic nesting, but it is worth mentionning anyway:
The finally is executed before items in the enclosing declarative region
are finalized. Also, any exception raised in the enclosing declarative region
will happen before the handled sequence of statements, and hence the finally
block won't be executed. This is consistent with exception
handlers, but the consequence is that if one wants to use a finally
block
to ensure stuff is executed even if there is an exception raised during
initialization of related values, he must encapsluate the declarations inside
the finaly block:
begin
declare
F : Foo := Create_Foo;
begin
...;
end;
finally
...
end;
-- Using gnatX declaration syntax
begin
F : Foo := Create_Foo;
...;
finally
...
end;
-
If an exception is raised in the finally part, it cannot be caught by the
exception_handlers
. -
Abort/ATC (asynchronous transfer of control) cannot interrupt a finally block, nor prevent its execution, that is the finally block must be executed in full even if the containing task is aborted, or if the control is transferred out of the block.
We discussed in the working group the pros & cons of using a new reserved word, vs. using existing reserved words. The concrete alternative was "at end":
procedure Example is
F : File_Type;
File_Name : constant String := "simple.txt";
Bar : Integer := Function_That_Might_Raise;
begin
Create (F, Out_File, File_Name);
Put_Line (F, "Hello World #1");
Put_Line (F, "Hello World #2");
Put_Line (F, "Hello World #3");
at end
Close (F);
end Example;
The considered pros & cons were the following. Pros:
-
We have a very low migration cost, in this particular case, because naming a variable
Finally
is very unlikely. We didn't find an occurence in any codebase we have access to. -
finally
being the keyword used in other languages makes it very easy to discover/recognize, and is also one less thing to learn/remember for every multi-lingual programmer. -
We don't have a lot of people migrating language versions and those who migrate are OK to dedicate some resources to migrating, so we already decided a while back that we're not completely against breaking things, if it makes sense.
-
Very easy to do an automatic migrator if needed.
Cons:
- One more keyword in Ada which already has more than fifty
- It might be annoying for some people to have to change their code
Other designs such as something similar to Go's defer
were considered
#29.
In the end we're going with this design both because of its cognitive simplicity, because it's easy to implement in GNAT, and because it's familiar to programmers and thus harder to fall into a trap.
We think that in many cases, type based finalization is a better fit, and
something like Ada's controlled object, or a simple mechanism akin to what is
described in #65 is the best fit,
with finally
complementing this feature when needed.
Feel free to read from
here
for more discussion about the defer
-like proposal.
We decided in this RFC to make the declarations declared in the attached declarative region available to the finally block, and to only execute the finally block if the declarative region has been properly executed, in line with the design of exception handlers, and respecting the syntactic nesting.
This has the drawback of not allowing finalization in the case where the
elaboration of the declarative region was not done. One must nest declarations
inside the sequence_of_statements
for that.
declare
A : Integer := raise Constraint_Error;
begin
...
finally
Put_Line ("Hello"); -- Not printed
end;
begin
A : Integer := raise Constraint_Error;
finally
Put_Line ("Hello"); -- Printed
end;
Most languages with exceptions (Java, C#, C++, Python, ...) have a finally block.