Skip to content

default ​

Applies to: struct fields function arguments method arguments

Makes the member optional and assigns a default value to it. The default value is lazily computed inside of the finishing function.

FormHow default value is computed
#[builder(default)]Default::default()
#[builder(default = expression)]expression

If combined with #[builder(into)], the default expression is additionally converted via Into::into.

Setters ​

Two setter methods are generated for the member with #[builder(default)] just like for members of type Option<T>:

NameInputDescriptionConfiguration attribute
{member}TAccepts a non-None value.some_fn
maybe_{member}Option<T>Accepts an Option directly.option_fn

If None is passed to the maybe_{member} setter, then the default value is used.

TIP

Switching between #[builder(default)] and Option<T> is compatible.

rust
use bon::Builder;

#[derive(Builder)]
struct Example {
    #[builder(default)] 
    x1: u32,

    #[builder(default = "anon".to_owned())] 
    x2: String,

    // No need for `.to_owned()`. Into is applied to the expression
    #[builder(default = "bon", into)] 
    x3: String,
}

let value = Example::builder().build();

assert_eq!(value.x1, 0);
assert_eq!(value.x2, "anon");
assert_eq!(value.x3, "bon");

let value = Example::builder()
    .x1(99)
    .maybe_x2(None) // None means the default will be used
    .maybe_x3(Some("lyra"))
    .build();

assert_eq!(value.x1, 99);
assert_eq!(value.x2, "anon");
assert_eq!(value.x3, "lyra");
rust
use bon::builder;

#[builder]
fn example(
    #[builder(default)] 
    x1: u32,

    #[builder(default = "anon".to_owned())] 
    x2: String,

    // No need for `.to_owned()`. Into is applied to the expression
    #[builder(default = "bon", into)] 
    x3: String,
) -> (u32, String, String) {
    (x1, x2, x3)
}

let value = example().call();

assert_eq!(value.0, 0);
assert_eq!(value.1, "anon");
assert_eq!(value.2, "bon");

let value = example()
    .x1(99)
    .maybe_x2(None) // None means the default will be used
    .maybe_x3(Some("lyra"))
    .call();

assert_eq!(value.0, 99);
assert_eq!(value.1, "anon");
assert_eq!(value.2, "lyra");
rust
use bon::bon;

struct Example {
    x1: u32,
    x2: String,
    x3: String,
}

#[bon]
impl Example {
    #[builder]
    fn new(
        #[builder(default)] 
        x1: u32,

        #[builder(default = "anon".to_owned())] 
        x2: String,

        // No need for `.to_owned()`. Into is applied to the expression
        #[builder(default = "bon", into)] 
        x3: String,
    ) -> Self {
        Self { x1, x2, x3 }
    }
}

let value = Example::builder().build();

assert_eq!(value.x1, 0);
assert_eq!(value.x2, "anon");
assert_eq!(value.x3, "bon");

let value = Example::builder()
    .x1(99)
    .maybe_x2(None) // None means the default will be used
    .maybe_x3(Some("lyra"))
    .build();

assert_eq!(value.x1, 99);
assert_eq!(value.x2, "anon");
assert_eq!(value.x3, "lyra");

Evaluation Context ​

You can reference other members as variables in the default expression. All members are initialized in the order of their declaration, and thus only members that are declared earlier (higher) in the code are available for the default expression.

rust
use bon::Builder;

#[derive(Builder)]
struct Example {
    x1: u32,

    // Note that here we don't have access to `x3`
    // because it's declared (and thus initialized) later
    #[builder(default = 2 * x1)]
    x2: u32,

    #[builder(default = x2 + x1)]
    x3: u32,
}

let value = Example::builder()
    .x1(3)
    .build();

assert_eq!(value.x1, 3);
assert_eq!(value.x2, 6);
assert_eq!(value.x3, 9);
rust
use bon::builder;

#[builder]
fn example(
    x1: u32,

    // Note that here we don't have access to `x3`
    // because it's declared (and thus initialized) later
    #[builder(default = 2 * x1)]
    x2: u32,

    #[builder(default = x2 + x1)]
    x3: u32,
) -> (u32, u32, u32) {
    (x1, x2, x3)
}

let value = example()
    .x1(3)
    .call();

assert_eq!(value, (3, 6, 9));
rust
use bon::bon;

struct Example;

#[bon]
impl Example {
    #[builder]
    fn example(
        x1: u32,

        // Note that here we don't have access to `x3`
        // because it's declared (and thus initialized) later
        #[builder(default = 2 * x1)]
        x2: u32,

        #[builder(default = x2 + x1)]
        x3: u32,
    ) -> (u32, u32, u32) {
        (x1, x2, x3)
    }
}

let value = Example::example()
    .x1(3)
    .call();

assert_eq!(value, (3, 6, 9));

Caveats ​

The self parameter in associated method syntax is not available to the default expression. If you need the self context for your defaulting logic, then set your member's type to Option<T> and handle the defaulting in the function's body manually.

Compile Errors ​

This attribute is incompatible with members of Option type, since Option already implies the default value of None. However, it can be used together with #[builder(required)].

Generated Docs ​

The documentation for the setters will include the default value expression. If you use #[builder(default)], then the default value expression will be inferred for primitive and some well-known types.

See examples of the generated docs with #[builder(default)].