Skip to content

CarloLucibello/AutoStructs.jl

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

10 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

AutoStructs

Build Status

This package provides the macro @structdef to automatically define a struct with a given name and fields starting from its constructor.

The package has two goals:

  • Combine the definition of a struct and its constructor in a single concise step.
  • Allow to redefine a struct without restarting the REPL.

Usage

julia> using AutoStructs, Random

# Define a struct from a constructor function.
# The struct is automatically named Layer and has two fields: weight and bias.
julia> @structdef function Layer(din::Int, dout::Int)
           weight = randn(dout, din)
           bias = zeros(dout)
           return Layer(weight, bias)
       end
var"##Layer#230"

# Now we can create an instance of the struct
julia> layer = Layer(2, 4)
Layer(weight = [-1.422950752982445 1.6242590842562115; -1.3393896739857631 0.8191382347282851; 0.3944420481119003 0.5955417101440335; 1.3944705999832914 1.1224997165166155], bias = [0.0, 0.0, 0.0, 0.0])

julia> layer isa Layer
true

julia> typeof(layer) # Layer is an alias for var"##Layer#230"
var"##Layer#230"{Matrix{Float64}, Vector{Float64}}

julia> layer isa Layer{Matrix{Float64}, Vector{Float64}} # Layer is a parametric type
true

# You can redefine the struct without restarting the REPL
julia> @structdef function Layer(din::Int, dout::Int, activation = identity)
           weight = randn(dout, din)
           bias = zeros(dout)
           return Layer(weight, bias, activation)
       end
var"##Layer#231"

julia> layer = Layer(2, 4, relu)
Layer(weight = [-0.7682741444223932 0.6155740231067407; -1.506126153999598 -0.7804554207069556; -0.10944649893432226 1.782291543052865; 0.26095648405623756 1.7713201612872245], bias = [0.0, 0.0, 0.0, 0.0], activation = tanh)

# Define any new method for the struct
julia> predict(l::Layer, x) = l.activation.(l.weight * x .+ l.bias)
predict (generic function with 1 method)

Documentation

help?> @structdef
  @structdef function MyType(args...; kws...); ...; return MyType(f1, f2, ...); end

  This is a macro for easily defining new types.

  Typically the steps to define a new struct are:

    1. Define a struct MyType with the desired fields.

    2. Define a constructor function like MyType(arg1, arg2, ...), which initialises the fields.

  This macro combines these steps into one, by taking the constructor definition of step 2 and using the return line
  to automatically infer the field names and define the struct.

  Moreover, if you change the name or types of the fields, then the struct definition is automatically replaced. This
  works because this definition uses an auto-generated name, and MyType becomes an alias to it. Thanks to this, you
  can easily experiment with different field names and types, overcoming a major limitation of a Revise.jl workflow.

  Subtyping is also supported, by adding a <: Supertype to the return line.

  Examples
  ========

  @structdef function Layer(din::Int, dout::Int)
      weight = randn(dout, din)
      bias = zeros(dout)
      return Layer(weight, bias)
  end
  
  layer = Layer(2, 4)
  layer isa Layer  # true

  The @structdef definition above is equivalent to the following code:

  @kwdef struct Layer001{T1, T2}
      weight::T1
      bias::T2
  end
  
  function Layer001(din::Int, dout::Int)
      weight = randn(dout, din)
      bias = zeros(dout)
      return Layer001(weight, bias)
  end
  
  Layer = Layer001
  
  Base.show(io::IO, x::Layer) = ... # we do some pretty printing
  Base.show(io::IO, ::MIME"text/plain", x::Layer) = ... 

  Since Layer001{T1, T2} can hold any objects, even Layer("hello", "world"), there should never be an ambiguity
  between the struct's own constructor, and your constructor function. If the two have the same number of arguments,
  you can avoid the ambiguity by using type restrictions in the input arguments (as in the example above) or in the
  return line:

  @structdef function Layer(din, dout)
      weight = randn(dout, din)
      bias = zeros(dout)
      return Layer(weight::AbstractMatrix, bias::AbstractVector)
  end
  
  layer = Layer(2, 4)

  This creates a struct like this:

  @kwdef struct Layer002{T1<:AbstractMatrix, T2<:AbstractVector}
      weight::T1
      bias::T2
  end

  and reassigns Layer = Layer002.

  Finally, we can also define a struct that is a subtype of another type:

  abstract type AbstractLayer end
  
  @structdef function Layer(din, dout)
      weight = randn(dout, din)
      bias = zeros(dout)
      return Layer(weight, bias) <: AbstractLayer
  end
  
  layer = Layer(din=2, dout=4)
  @assert layer isa AbstractLayer

  The corresponding struct definition becomes

  @kwdef struct Layer003{T1, T2} <: AbstractLayer
      weight::T1
      bias::T2
  end

Credits

This package was inspired by conversations among Flux.jl developers on how to compactly define models which resulted in the defintion of the @autostruct macro in Fluxperimental. @structdef here is a version of @autostruct for generic usage instead of being tied to Flux.

Similar Packages

About

Compact struct definition and prototyping

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages