Modules in an F# project

F# supports grouping related code into namespaces and modules. Namespaces can only contain type declarations. They cannot contain values i.e. elements that are declared with the “let” keyword. Modules on the other hand can handle both types and values. Modules can also contain other modules, i.e. they can be nested.

Here’s a simple module with a type and a function:

module com.mycompany.domain.modules

type Address = {
    street: string;
    city: string;
    number: int;
}
with member this.evenNumber = this.number % 2 = 0

let sumFunction x y = x + y

Here’s how we can reference and use it from Program.fs using the “open” keyword:

namespace com.mycompany.entry

open System
open com.mycompany.domain.modules

module Main =    
    [<EntryPoint>]
    let main args =        
        let myAddress = {street = "Rue Bourbon"; city = "New Orleans"; number = 14}
        let isEven = myAddress.evenNumber
        printfn "Street name: %s, number is even: %b" myAddress.street isEven
        let sum = sumFunction 12 30
        let keepConsoleWindowOpen = Console.ReadKey()

        0

This will print…

Street name: Rue Bourbon, number is even: true

…in the console window and “sum” will of course be 42.

We can add a namespace to a module. The module name must be simple, i.e. not a path and followed by an equal sign. Furthermore, everything that’s included in the module must be indented:

namespace com.mycompany.domains.main

module AddressModule =
    type Address = {
        street: string;
        city: string;
        number: int;
    }
    with member this.evenNumber = this.number % 2 = 0

    let sumFunction x y = x + y

The full reference path in Program.cs will be com.mycompany.domains.main.AddressModule:

namespace com.mycompany.entry

open System
open com.mycompany.domains.main.AddressModule

module Main =    
    [<EntryPoint>]
    let main args =       
        let myAddress = {street = "Rue Bourbon"; city = "New Orleans"; number = 14}
        let isEven = myAddress.evenNumber
        printfn "Street name: %s, number is even: %b" myAddress.street isEven
        let sum = sumFunction 12 30
        let keepConsoleWindowOpen = Console.ReadKey()
        0

A namespace can include multiple modules:

namespace com.mycompany.domains.main

module AddressModule =
    type Address = {
        street: string;
        city: string;
        number: int;
    }
    with member this.evenNumber = this.number % 2 = 0

    let sumFunction x y = x + y

module BookModule =
    type Book = {
        title: string;
        numberOfPages:int;
        author: string
    } 
    with member this.takesLongTimeToRead = this.numberOfPages > 500

A module can include submodules but pay attention to the indentation:

namespace com.mycompany.domains.main

module AddressModule =
    type Address = {
        street: string;
        city: string;
        number: int;
    }
    with member this.evenNumber = this.number % 2 = 0

    let sumFunction x y = x + y

    module OrderModule = 
        type Order (product:string, value: int, address:Address) =
            member this.Product = product
            member this.Value = value
            member this.Address = address

The full reference path to the OrderModule will be…

open com.mycompany.domains.main.AddressModule.OrderModule

…, i.e. the namespace, followed by the top module name and then by the submodule name.

View all F# related articles here.

Advertisement

About Andras Nemes
I'm a .NET/Java developer living and working in Stockholm, Sweden.

Leave a Reply

Fill in your details below or click an icon to log in:

WordPress.com Logo

You are commenting using your WordPress.com account. Log Out /  Change )

Facebook photo

You are commenting using your Facebook account. Log Out /  Change )

Connecting to %s

Elliot Balynn's Blog

A directory of wonderful thoughts

Software Engineering

Web development

Disparate Opinions

Various tidbits

chsakell's Blog

WEB APPLICATION DEVELOPMENT TUTORIALS WITH OPEN-SOURCE PROJECTS

Once Upon a Camayoc

Bite-size insight on Cyber Security for the not too technical.

%d bloggers like this: