Julia Interfaces Tutorial
last modified March 3, 2025
In Julia, interfaces are informal contracts that define a set of methods a type must implement to satisfy a particular behavior. Unlike other languages, Julia does not enforce interfaces explicitly. Instead, they are conventions.
Interfaces are crucial for polymorphism, allowing different types to be used interchangeably if they implement the required methods.
Example 1: Iteration Interface
The iteration interface requires implementing iterate
and
Base.IteratorSize
.
struct MyRange start::Int stop::Int end Base.iterate(r::MyRange, state=r.start) = state > r.stop ? nothing : (state, state + 1) Base.IteratorSize(::Type{MyRange}) = Base.HasLength()
This example defines a custom range type that supports iteration.
Example 2: Indexing Interface
The indexing interface requires implementing getindex
and
setindex!
.
struct MyArray data::Vector{Int} end Base.getindex(a::MyArray, i::Int) = a.data[i] Base.setindex!(a::MyArray, v::Int, i::Int) = a.data[i] = v
This example defines a custom array type that supports indexing.
Example 3: AbstractArray Interface
The AbstractArray
interface requires implementing methods like
size
and getindex
.
struct MyMatrix data::Matrix{Int} end Base.size(m::MyMatrix) = size(m.data) Base.getindex(m::MyMatrix, i::Int, j::Int) = m.data[i, j]
This example defines a custom matrix type that behaves like an array.
Example 4: AbstractFloat Interface
The AbstractFloat
interface requires implementing methods like
+
, -
, and *
.
struct MyFloat value::Float64 end Base.:+(a::MyFloat, b::MyFloat) = MyFloat(a.value + b.value) Base.:-(a::MyFloat, b::MyFloat) = MyFloat(a.value - b.value) Base.:*(a::MyFloat, b::MyFloat) = MyFloat(a.value * b.value)
This example defines a custom floating-point type with arithmetic operations.
Example 5: AbstractString Interface
The AbstractString
interface requires implementing methods like
ncodeunits
and codeunit
.
struct MyString data::String end Base.ncodeunits(s::MyString) = ncodeunits(s.data) Base.codeunit(s::MyString, i::Int) = codeunit(s.data, i)
This example defines a custom string type that behaves like a standard string.
Example 6: AbstractDict Interface
The AbstractDict
interface requires implementing methods like
get
and haskey
.
struct MyDict data::Dict{String, Int} end Base.get(d::MyDict, key::String, default) = get(d.data, key, default) Base.haskey(d::MyDict, key::String) = haskey(d.data, key)
This example defines a custom dictionary type that supports key-value lookups.
Example 7: AbstractSet Interface
The AbstractSet
interface requires implementing methods like
in
and length
.
struct MySet data::Set{Int} end Base.in(x::Int, s::MySet) = x in s.data Base.length(s::MySet) = length(s.data)
This example defines a custom set type that supports membership checks.
Example 8: AbstractChannel Interface
The AbstractChannel
interface requires implementing methods like
put!
and take!
.
struct MyChannel data::Channel{Int} end Base.put!(c::MyChannel, x::Int) = put!(c.data, x) Base.take!(c::MyChannel) = take!(c.data)
This example defines a custom channel type for inter-task communication.
Example 9: AbstractIO Interface
The AbstractIO
interface requires implementing methods like
read
and write
.
struct MyIO data::IOBuffer end Base.read(io::MyIO, n::Int) = read(io.data, n) Base.write(io::MyIO, x::UInt8) = write(io.data, x)
This example defines a custom I/O type for reading and writing data.
Example 10: AbstractTask Interface
The AbstractTask
interface requires implementing methods like
schedule
and wait
.
struct MyTask task::Task end Base.schedule(t::MyTask) = schedule(t.task) Base.wait(t::MyTask) = wait(t.task)
This example defines a custom task type for asynchronous execution.
Best Practices for Interfaces
- Follow Conventions: Adhere to Julia's informal interface conventions.
- Document Behavior: Clearly document expected behavior for custom types.
- Test Thoroughly: Ensure custom types work with generic functions.
- Use Abstract Types: Leverage abstract types for flexible design.
Source
Julia Interfaces Documentation
In this tutorial, we explored Julia interfaces with practical examples, covering iteration, indexing, arrays, and more. Interfaces enable flexible and reusable code in Julia.
Author
List all Julia tutorials.