Ich würde bevorzugen, dass Beispiele in einer Lisp-Variante (Bonuspunkte für Clojure oder Scheme) sind, da das ist, was ich am vertrautesten bin, aber jede Rückmeldung bezüglich DBC in funktionalen Sprachen wäre natürlich wertvoll für die größere Gemeinschaft.Wie könnten Sie Design-by-Contract in Clojure speziell oder funktionale Sprachen im Allgemeinen implementieren?
Hier ist offensichtlich, dass die Art und Weise:
(defn foo [action options]
(when-not (#{"go-forward" "go-backward" "turn-right" "turn-left"} action)
(throw (IllegalArgumentException.
"unknown action")))
(when-not (and (:speed options) (> (:speed options) 0))
(throw (IllegalArgumentException.
"invalid speed")))
; finally we get to the meat of the logic)
Ich mag an nicht diese Implementierung ist, dass der Vertrag Logik die Kernfunktionalität verschleiert; Der wahre Zweck der Funktion geht bei bedingten Prüfungen verloren. Dies ist das gleiche Problem, das ich in this question angesprochen habe. In einer imperativen Sprache wie Java kann ich Anmerkungen oder Metadaten/Attribute verwenden, die in der Dokumentation eingebettet sind, um den Vertrag aus der Methodenimplementierung zu entfernen.
Hat schon jemand versucht, Verträge mit Metadaten in Clojure hinzuzufügen? Wie würden Funktionen höherer Ordnung verwendet? Welche anderen Möglichkeiten gibt es?
Haben Sie sehen, wie Verträge in PLT-Schema umgesetzt werden? Schau mal. http://docs.plt-scheme.org/guide/contracts.html –
@Alexey - Das ist eine spektakuläre Ressource! Ich bin ziemlich neu in Scheme (arbeite durch The Little/Seasoned Bücher) und ich wusste nicht, dass das existiert, also danke. – rcampbell
Nicht direkt eine Antwort auf Ihre Frage, aber werfen Sie einen Blick auf QuickCheck und seine Derivate (ClojureCheck). Es ist im Wesentlichen eigenschaftsbasiertes Testen, und in Verträgen definieren Sie Eigenschaften, so dass Sie die erzeugten Tests leicht erhalten können. – Masse