How can I get around the lack of a finally block in PHP? How can I get around the lack of a finally block in PHP? php php

How can I get around the lack of a finally block in PHP?


Solution, no. Irritating cumbersome workaround, yes:

$stored_exc = null;try {    // Do stuff} catch (Exception $exc) {    $stored_exc = $exc;    // Handle an error}// "Finally" here, clean up after yourselfif ($stored_exc) {    throw($stored_exc);}

Yucky, but should work.

Please note: PHP 5.5 finally (ahem, sorry) added a finally block: https://wiki.php.net/rfc/finally (and it only took a few years... available in the 5.5 RC almost four years to the date since I posted this answer...)


The RAII idiom offers a code-level stand-in for a finally block. Create a class that holds callable(s). In the destuctor, call the callable(s).

class Finally {    # could instead hold a single block    public $blocks = array();    function __construct($block) {        if (is_callable($block)) {            $this->blocks = func_get_args();        } elseif (is_array($block)) {            $this->blocks = $block;        } else {            # TODO: handle type error        }    }    function __destruct() {        foreach ($this->blocks as $block) {            if (is_callable($block)) {                call_user_func($block);            } else {                # TODO: handle type error.            }        }    }}

Coordination

Note that PHP doesn't have block scope for variables, so Finally won't kick in until the function exits or (in global scope) the shutdown sequence. For example, the following:

try {    echo "Creating global Finally.\n";    $finally = new Finally(function () {        echo "Global Finally finally run.\n";    });    throw new Exception;} catch (Exception $exc) {}class Foo {    function useTry() {        try {            $finally = new Finally(function () {                echo "Finally for method run.\n";             });            throw new Exception;        } catch (Exception $exc) {}        echo __METHOD__, " done.\n";    }}$foo = new Foo;$foo->useTry();echo "A whole bunch more work done by the script.\n";

will result in the output:

Creating global Finally.Foo::useTry done.Finally for method run.A whole bunch more work done by the script.Global Finally finally run.

$this

PHP 5.3 closures can't access $this (fixed in 5.4), so you'll need an extra variable to access instance members within some finally-blocks.

class Foo {    function useThis() {        $self = $this;        $finally = new Finally(            # if $self is used by reference, it can be set after creating the closure            function () use ($self) {               $self->frob();            },            # $this not used in a closure, so no need for $self            array($this, 'wibble')        );        /*...*/    }    function frob() {/*...*/}    function wibble() {/*...*/}}

Private and Protected Fields

Arguably the biggest problem with this approach in PHP 5.3 is the finally-closure can't access private and protected fields of an object. Like accessing $this, this issue is resolved in PHP 5.4. For now, private and protected properties can be accessed using references, as Artefacto shows in his answer to a question on this very topic elsewhere on this site.

class Foo {    private $_property='valid';    public function method() {        $this->_property = 'invalid';        $_property =& $this->_property;        $finally = new Finally(function () use (&$_property) {                $_property = 'valid';        });        /* ... */    }    public function reportState() {        return $this->_property;    }}$f = new Foo;$f->method();echo $f->reportState(), "\n";

Private and protected methods can be accessed using reflection. You can actually use the same technique to access non-public properties, but references are simpler and more lightweight. In a comment on the PHP manual page for anonymous functions, Martin Partel gives an example of a FullAccessWrapper class that opens up non-public fields to public access. I won't reproduce it here (see the two previous links for that), but here is how you'd use it:

class Foo {    private $_property='valid';    public function method() {        $this->_property = 'invalid';        $self = new FullAccessWrapper($this);        $finally = new Finally(function () use (&$self) {                $self->_fixState();        });        /* ... */    }    public function reportState() {        return $this->_property;    }    protected function _fixState() {        $this->_property = 'valid';    }}$f = new Foo;$f->method();echo $f->reportState(), "\n";

try/finally

try blocks require at least one catch. If you only want try/finally, add a catch block that catches a non-Exception (PHP code can't throw anything not derived from Exception) or re-throw the caught exception. In the former case, I suggest catching StdClass as an idiom meaning "don't catch anything". In methods, catching the current class could also be used to mean "don't catch anything", but using StdClass is simpler and easier to find when searching files.

try {   $finally = new Finally(/*...*/);   /* ... */} catch (StdClass $exc) {}try {   $finally = new Finally(/*...*/);   /* ... */} catch (RuntimeError $exc) {    throw $exc}


Here is my solution to the lack of finally block. It not only provides a work around for the finally block, it also extends the try/catch to catch PHP errors (and fatal errors too). My solution looks like this (PHP 5.3):

_try(    //some piece of code that will be our try block    function() {        //this code is expected to throw exception or produce php error    },    //some (optional) piece of code that will be our catch block    function($exception) {        //the exception will be caught here        //php errors too will come here as ErrorException    },    //some (optional) piece of code that will be our finally block    function() {        //this code will execute after the catch block and even after fatal errors    });

You can download the solution with documentation and examples from git hub - https://github.com/Perennials/travelsdk-core-php/tree/master/src/sys