LISP - Gestione degli errori

Nella terminologia Common LISP, le eccezioni sono chiamate condizioni.

In effetti, le condizioni sono più generali delle eccezioni nei linguaggi di programmazione tradizionali, perché a condition rappresenta qualsiasi occorrenza, errore o meno, che potrebbe influenzare vari livelli dello stack di chiamate di funzione.

Il meccanismo di gestione delle condizioni in LISP gestisce tali situazioni in modo tale che le condizioni vengano utilizzate per segnalare un avviso (ad esempio stampando un avviso) mentre il codice di livello superiore sullo stack di chiamate può continuare il suo lavoro.

Il sistema di gestione delle condizioni in LISP ha tre parti:

  • Segnalazione di una condizione
  • Gestire la condizione
  • Riavvia il processo

Gestire una condizione

Prendiamo un esempio della gestione di una condizione derivante dalla condizione di divisione per zero, per spiegare i concetti qui.

È necessario eseguire i seguenti passaggi per gestire una condizione:

  • Define the Condition - "Una condizione è un oggetto la cui classe indica la natura generale della condizione e i cui dati di istanza contengono informazioni sui dettagli delle circostanze particolari che portano alla segnalazione della condizione".

    La macro di definizione delle condizioni viene utilizzata per definire una condizione, che ha la seguente sintassi:

(define-condition condition-name (error)
   ((text :initarg :text :reader text))
)
  • I nuovi oggetti condizione vengono creati con la macro MAKE-CONDITION, che inizializza gli slot della nuova condizione in base al file :initargs discussione.

Nel nostro esempio, il codice seguente definisce la condizione:

(define-condition on-division-by-zero (error)
   ((message :initarg :message :reader message))
)
  • Writing the Handlers- un condition handler è un codice utilizzato per gestire la condizione segnalata su di esso. È generalmente scritto in una delle funzioni di livello superiore che chiamano la funzione di errore. Quando viene segnalata una condizione, il meccanismo di segnalazione cerca un gestore appropriato in base alla classe della condizione.

    Ogni gestore è composto da:

    • Identificatore di tipo, che indica il tipo di condizione che può gestire
    • Una funzione che accetta un singolo argomento, la condizione

    Quando viene segnalata una condizione, il meccanismo di segnalazione trova il gestore stabilito più di recente compatibile con il tipo di condizione e chiama la sua funzione.

    La macro handler-casestabilisce un gestore delle condizioni. La forma base di un caso del gestore -

(handler-case expression error-clause*)

Dove, ogni clausola di errore ha la forma -

condition-type ([var]) code)
  • Restarting Phase

    Questo è il codice che ripristina effettivamente il programma dagli errori e i gestori di condizioni possono quindi gestire una condizione invocando un riavvio appropriato. Il codice di riavvio è generalmente inserito nelle funzioni di livello medio o basso e i gestori delle condizioni sono posizionati nei livelli superiori dell'applicazione.

    Il handler-bindmacro consente di fornire una funzione di riavvio e consente di continuare a funzioni di livello inferiore senza srotolare lo stack di chiamate di funzione. In altre parole, il flusso di controllo sarà ancora nella funzione di livello inferiore.

    La forma base di handler-bind è il seguente -

(handler-bind (binding*) form*)

Dove ogni associazione è un elenco di quanto segue:

  • un tipo di condizione
  • una funzione gestore di un argomento

Il invoke-restart macro trova e richiama la funzione di riavvio associata più di recente con il nome specificato come argomento.

Puoi avere più riavvii.

Esempio

In questo esempio, dimostriamo i concetti di cui sopra scrivendo una funzione chiamata funzione-divisione, che creerà una condizione di errore se l'argomento divisore è zero. Abbiamo tre funzioni anonime che forniscono tre modi per uscirne: restituendo un valore 1, inviando un divisore 2 e ricalcolando, o restituendo 1.

Crea un nuovo file di codice sorgente denominato main.lisp e digita il codice seguente.

(define-condition on-division-by-zero (error)
   ((message :initarg :message :reader message))
)
   
(defun handle-infinity ()
   (restart-case
      (let ((result 0))
         (setf result (division-function 10 0))
         (format t "Value: ~a~%" result)
      )
      (just-continue () nil)
   )
)
     
(defun division-function (value1 value2)
   (restart-case
      (if (/= value2 0)
         (/ value1 value2)
         (error 'on-division-by-zero :message "denominator is zero")
      )

      (return-zero () 0)
      (return-value (r) r)
      (recalc-using (d) (division-function value1 d))
   )
)

(defun high-level-code ()
   (handler-bind
      (
         (on-division-by-zero
            #'(lambda (c)
               (format t "error signaled: ~a~%" (message c))
               (invoke-restart 'return-zero)
            )
         )
         (handle-infinity)
      )
   )
)

(handler-bind
   (
      (on-division-by-zero
         #'(lambda (c)
            (format t "error signaled: ~a~%" (message c))
            (invoke-restart 'return-value 1)
         )
      )
   )
   (handle-infinity)
)

(handler-bind
   (
      (on-division-by-zero
         #'(lambda (c)
            (format t "error signaled: ~a~%" (message c))
            (invoke-restart 'recalc-using 2)
         )
      )
   )
   (handle-infinity)
)

(handler-bind
   (
      (on-division-by-zero
         #'(lambda (c)
            (format t "error signaled: ~a~%" (message c))
            (invoke-restart 'just-continue)
         )
      )
   )
   (handle-infinity)
)

(format t "Done."))

Quando esegui il codice, restituisce il seguente risultato:

error signaled: denominator is zero
Value: 1
error signaled: denominator is zero
Value: 5
error signaled: denominator is zero
Done.

Oltre al "Sistema delle condizioni", come discusso sopra, Common LISP fornisce anche varie funzioni che possono essere chiamate per segnalare un errore. La gestione di un errore, quando segnalato, è tuttavia dipendente dall'implementazione.

Funzioni di segnalazione degli errori in LISP

La tabella seguente fornisce le funzioni di uso comune che segnalano avvisi, interruzioni, errori non irreversibili e fatali.

Il programma utente specifica un messaggio di errore (una stringa). Le funzioni elaborano questo messaggio e possono / non possono visualizzarlo all'utente.

I messaggi di errore dovrebbero essere costruiti applicando il format funzione, non deve contenere un carattere di nuova riga né all'inizio né alla fine e non è necessario indicare errori, poiché il sistema LISP si prenderà cura di questi secondo il suo stile preferito.

Sr.No. Funzione e descrizione
1

error formato-stringa e rest args

Segnala un errore fatale. È impossibile continuare da questo tipo di errore; quindi l'errore non tornerà mai al suo chiamante.

2

cerror continue-stringa-formato stringa-formato-errore e rest args

Segnala un errore ed entra nel debugger. Tuttavia, consente di continuare il programma dal debugger dopo aver risolto l'errore.

3

warn formato-stringa e rest args

stampa un messaggio di errore ma normalmente non entra nel debugger

4

break& stringa di formato opzionale & rest args

Stampa il messaggio e entra direttamente nel debugger, senza consentire alcuna possibilità di intercettazione da parte delle strutture di gestione degli errori programmate

Esempio

In questo esempio, la funzione fattoriale calcola fattoriale di un numero; tuttavia, se l'argomento è negativo, genera una condizione di errore.

Crea un nuovo file di codice sorgente denominato main.lisp e digita il codice seguente.

(defun factorial (x)
   (cond ((or (not (typep x 'integer)) (minusp x))
      (error "~S is a negative number." x))
      ((zerop x) 1)
      (t (* x (factorial (- x 1))))
   )
)

(write(factorial 5))
(terpri)
(write(factorial -1))

Quando esegui il codice, restituisce il seguente risultato:

120
*** - -1 is a negative number.