在Chicken Scheme中模拟typeclass

Mistivia aa535dd51c add tests hace 4 meses
tests aa535dd51c add tests hace 4 meses
.gitignore eae9dcdba5 update readme hace 4 meses
LICENSE 98e8c71983 init hace 4 meses
README.md 19d0723b1c add example with modules hace 4 meses
release-info 98e8c71983 init hace 4 meses
trait.egg aa535dd51c add tests hace 4 meses
trait.scm 4760023130 make trait impl lexically scoped hace 4 meses

README.md

Trait

A trait/typeclass system for Chicken Scheme, inspired by Type Classes Without Types. But currently my implementation is in a very early stage, so it's much more simpler and inferior than the one described in the paper.

I prefer the name typeclass, but there has been an egg named typeclass already. So I borrowed the word trait from Rust.

Example

Eq

(import trait)

(define-record point x y)

(define-trait Eq
  (==)
  ;; with a default implementation
  (/= (lambda (a b) (not (== a b)))))

(define-trait-impl (Eq number?)
  (== =))

(define-trait-impl (Eq symbol?)
  (== eq?))

(define-trait-impl (Eq list?)
  (== equal?))

(define-trait-impl (Eq point?)
  (== (lambda (a b) (and (point? b)
                         (= (point-x a) (point-x b))
                         (= (point-y a) (point-y b))))))

(display (list (== 'a 'a)
               (/= 'a 'a)

               (== 1 1)
               (/= 1 1)

               (== 1 2)
               (/= 1 2)

               (== (list 1 2) (list 1 2))
               (/= (list 1 2) (list 1 2))

               (== (make-point 3 4) (make-point 3 4))
               (/= (make-point 3 4) (make-point 3 4))))

Monad

(import trait)

(define-trait Monad
  (>>=)
  (return))

(define-record nullable is-null value)
(define (make-some value)
  (make-nullable #f value))
(define (make-null)
  (make-nullable #t '()))
(define (nullable-type x)
  (if (nullable-is-null x)
      'null
      'some))

(define-trait-impl (Monad nullable?)
  (>>= (lambda (m f)
         (let ((type (nullable-type m)))
           (cond ((eq? type 'null)
                  (make-null))
                 ((eq? type 'some)
                  (f (nullable-value m)))))))
  (return make-some))

(define (*2 a)
  (make-some (* 2 a)))

(display (nullable-value (>>= (make-some 3) *2)))
(newline)

;; For a function in trait, the implementation is selected by
;; applying the predicates to the first argument.
;; When the varaible of related type is the return value or
;; the 2nd/3rd/... argument, use (with-type-of var trait function)
;; to get the right function.
(define x (make-some 42))
(let ((return (with-type-of x Monad return)))
  (display (nullable-value (return 99))))

With Multiple Modules

(module define-macro (define-macro)
  ;; import
  (import scheme
      (chicken syntax))
  ;; body
  (define-syntax define-macro
    (er-macro-transformer
     (lambda (exp r c)
       (let ((def (cadr exp))
       (body (cddr exp)))
         `(define-syntax ,(car def)
      (er-macro-transformer
       (lambda (e2 r2 c2)
         (define (transform-func ,@(cdr def))
           ,@body)
         (apply transform-func (cdr e2))))))))))

(module Eq (Eq == /=)
  (import scheme
          (chicken base)
          trait)

  (define-trait Eq
    (==)
    (/= (lambda (a b)
          (display ==) (newline)
          (not (== a b))))))

(module point (point?
               make-point
               point==?
               impl-Eq-point)
  (import scheme
          (chicken base)
          Eq
          trait
          define-macro)

  (define-record point x y)

  (define (point==? a b)
    (and (point? a)
         (point? b)
         (= (point-x a) (point-x b))
         (= (point-y a) (point-y b))))

  ;; trait implementations should be defined as macros
  ;; to export to other modules
  (define-macro (impl-Eq-point)
    '(define-trait-impl (Eq point?)
       (== point==?))))

(module mymodule ()
  (import scheme
          (chicken base)
          trait
          Eq
          point)

  ;; import trait implementation
  (impl-Eq-point)

  (display (== (make-point 1 2) (make-point 1 2)))
  (newline)
  (display (/= (make-point 1 2) (make-point 1 2)))
  (newline))