Ада-подобные типы в Нимроде

Сегодня я спросил в списке рассылки D, можно ли определять и использовать пользовательские типы данных способом, подобным, например, пример из страницы Ada wiki:

type Day_type   is range    1 ..   31;
type Month_type is range    1 ..   12;
type Year_type  is range 1800 .. 2100;
type Hours is mod 24;
type Weekday is (Monday, Tuesday, Wednesday, Thursday, Friday, Saturday, Sunday); 


type Date is
   record
     Day   : Day_type;
     Month : Month_type;
     Year  : Year_type;
   end record;

subtype Working_Hours is Hours range 0 .. 12;
subtype Working_Day is Weekday range Monday .. Friday;
Work_Load: constant array(Working_Day) of Working_Hours 
   := (Friday => 6, Monday => 4, others => 10); 

а ответ продемонстрировал что-то вроде:

import std.typecons;
import std.exception;

struct Limited(T, T lower, T upper)
{
    T _t;
    mixin Proxy!_t; //Limited acts as T (almost)
    invariant()
    {
        enforce(_t >= lower && _t <= upper);
    }
    this(T t)
    {
        _t = t;
    }
}

auto limited(T, T lower, T upper)(T init = T.init)
{
    return Limited!(T, lower, upper)(init);
}

unittest
{
    enum l = [-4,9];
    auto a = limited!(int, l[0], l[1])();
    foreach(i; l[0] .. l[1]+1)
    {
        a = i;
    }

    assertThrown({a = -5;}());
    assertThrown({a = 10;}());
}

который показывает, что это возможно, но, вероятно, не хватает элегантности Ada.

Теперь, после прочтения о Nimrod в последнее время, мне интересно, как он может справиться с аналогичной задачей с предоставлением обеспечения такой же безопасности типа Ada?

Ответ 1

Nimrod поддерживает их довольно прямо:

type
  Day = range[1..31]
  Month = range[1..12]

  WeekDay = enum
    Monday, Tuesday, Wednesday, Thursday, Friday, Saturday, Sunday

  WorkingDays = range[Monday..Friday]
  WorkingHours = range[0..12]

  WorkSchedule = array[WorkingDays, WorkingHours]

Ошибки выполняются либо во время компиляции:

var x: Day
x = 40 # conversion from int literal(40) to Day is invalid

.. или во время выполнения

var x: Day
var y = unknownInt() # let say it returns 100

x = y # unhandled exception: value 100 out of range [EOutOfRange]

Кроме того, различные типы могут использоваться, если требуется еще более безопасная защита.

Ответ 2

Далее, экземпляр становится одним аргументом проще с использованием

import std.traits;

/*! Instantiator for \c Limited. */
template limited(alias min, alias max, bool Exceptional = true)
    if (!is(CommonType!(typeof(min), typeof(max)) == void)) {
    auto limited(CommonType!(typeof(min), typeof(max)) value) {
        return Limited!(typeof(value), min, max)(value);
    }
}

См. Функция Instantiator для шаблона Bound не компилируется