I’m playing around with MLStyle but can’t seem to get a definition for a cons list to work.
If I try:
@data XX{T} begin
Nil
Cons(hd::T, tl::XX{T})
end
I get: ERROR: LoadError: Defining generic enum Nil <: XX{T} is invalid, as Julia does not support generic value.
If I try:
@data XX{T} begin
Nil()
Cons(hd::T, tl::XX{T})
end
I get: MethodError: no method matching List.Nil()
The type List.Nil exists, but no method is defined for this combination of argument types when trying to construct it.
OK, I’m not sure why, but the solution is less, not more …
@data XX begin
Nil
Cons(hd, tl::XX)
end
function sum_list(lst)
@match lst begin
Nil => 0
Cons(h, t) => h + sum_list(t)
end
end
function printer(lst)
@match lst begin
Nil => println("[]")
Cons(h, t) => begin
print("$(h) . ")
printer(t)
end
end
end
function make_test_1()
return Cons(1, Cons(2, Cons(3, Nil)))
end
function make_test_2()
return Cons("A", Cons("B", Cons("C", Nil)))
end
i = make_test_1()
s = make_test_2()
println("sum_list(i) = $(sum_list(i))")
#sum_list(i) = 6
printer(i)
#1 . 2 . 3 . []
printer(s)
#A . B . C . []
using Moshi.Data: @data
julia> @data XX{T} begin
Nil
Cons(T, XX{T})
end
julia> using .XX: Nil, Cons
julia> Cons('a', Cons('b', Nil{Char}()))
Main.XX.var"typeof(XX)"{Char}(Main.XX.var"##Storage#Cons"{Char}('a', Main.XX.var"typeof(XX)"{Char}(Main.XX.var"##Storage#Cons"{Char}('b', Main.XX.var"typeof(XX)"{Char}(Main.XX.var"##Storage#Nil"{Char}())))))
Really ugly representation but I think it’s the right value.
And heterogeneous lists fail like you want:
julia> Cons('a', Cons(1.2, Nil{Char}()))
ERROR: MethodError: no method matching Cons(::Float64, ::Main.XX.var"typeof(XX)"{Char})
The type `Cons` exists, but no method is defined for this combination of argument types when trying to construct it.
module List
using Moshi.Data: @data
using Moshi.Match: @match
export sum_list, make_test_1, make_test_2, printer
@data XX{T} begin
Nil
Cons(T, XX{T})
end
function sum_list(lst)
@match lst begin
Nil => 0
Cons(h, t) => h + sum_list(t)
end
end
function printer(lst)
@match lst begin
Nil => println("[]")
Cons(h, t) => begin
print("$(h) . ")
printer(t)
end
end
end
function make_test_1()
return Cons(1, Cons(2, Cons(3, Nil)))
end
function make_test_2()
return Cons("A", Cons("B", Cons("C", Nil)))
end
end # module List
my test program is just doing this
using Test
using List
@testset "Simple Cons" begin
s = make_test_2()
printer(s)
end
I get this error: ERROR: LoadError: UndefVarError: Cons not defined in List
OK, that seems strange to need a “using” inside the module that defines those things …
I put the using directly after the @data:
@data XX{T} begin
Nil
Cons(T, XX{T})
end
using ..XX: Nil, Cons
and now I get
MethodError: no method matching List.XX.Cons(::String, ::Type{List.XX.Nil{String}})
The type List.XX.Cons exists, but no method is defined for this combination of argument types when trying to construct it.
Closest candidates are:
List.XX.Cons(::T, ::List.XX.var"typeof(XX)"{T}) where T
I get a very similar error whether I have just Nil or Nil{String} in the example construction.
function make_test_1()
return Cons(1, Cons(2, Cons(3, Nil{Int}())))
end
function make_test_2()
return Cons("A", Cons("B", Cons("C", Nil{String}())))
end
Of course, this errors too because the code generated for @match in recursive ADTs seems to be not good, but that’s something you can open an issue about.