References
References are like an alias for an lvalue, but this lvalue is no ordinary lvalue. It should always be a variable-based expression. You can think of them as pointers but they are safer because of compiler's safety obsessions. Actually, references are raw pointers with safety mask.
They are used with &
operator in syntax. You can't have nested references, for example you can have a pointer pointing to a pointer but not a reference referencing to a reference.
References point to the value from which they were initialized and must receive an initialize expression. The lvalue pointing to later references cannot be changed. Any assignment statement is always about changing the data they refer to.
Type safety accepts when a reference is the exact same data-type by value. Implicit cast or etc. cannot be used to pass references.
Reference Variables
Reference variables are variables that reference an lvalue. Any assignment made affects the referenced lvalue. Declared with &
operator.
For example:
fn main() {
let mut a = 20
let mut &b = a
b += 20
println(a) // 40
}
Also you can use reference types in mult-declarative assignments.
For example:
fn main() {
let mut a = 20
let (mut &x, mut y) = a, 20
x += y
println(a) // 40
}
Static Variables
Static variables cannot be reference due to safety reasons. After assigned to static storage, reference may be dangling, so pointer is dangling. Tracking lifetime of lvalues is expensive, therefore Jule is not allows reference static variables.
If you need reference for static storage, use raw pointer instead. Raw pointers are part of the Unsafe Jule and references are safety masks for raw pointers.
Anonymous Functions with References
Anonymous functions copies instead of referencing the definitions of the scope in which they are defined, for safety reasons. Thus, a possible danger of dangling is prevented. But some copied things can be undsgr, one of them being references. Even if the references are copied, they will still continue to point to the same address as it is an address alias in nature. Therefore, there is a danger of dangling the reference if it goes out of scope. To avoid this, Safe Jule does not allow you to use references from parent scopes.
If you're sure it's safe to do so, Unsafe Jule lets you access such dangerous references.
Reference Parameters
Reference parameters must take an lvalue as an argument. To specify a reference parameter, the parameter identifier must be preceded by the &
operator.
For example:
fn add_20(mut &a: int) {
a += 20
}
fn main() {
let mut a = 20
add_20(a)
println(a) // 40
}
Why References Accepts only Lvalue
This is the result of the compiler trying to make sure things are safe. It always asks to reference a variable to keep a good watch on your reference and make sure it's safe. This is an effort to guarantee that your reference will never be dangling because the scope of your variable is traceable.
An expression which is pointed to always must be variable based. For example, variables, structure (stored in variable) field or arrays. Slices are not supported because they might deallocate internal buffer to grow, so references are not safe for slices. But arrays always supported because they are guaranteed to be fixed size at runtime always.
Concurrency
Concurrency imposes a number of process-intensive conditions that are difficult to trace at compile time and further increase compile times. Using references in a concurrent call means that the scope of the variable cannot be properly traced. Therefore, concurrent calls do not support functions with reference parameters.
Of course, if you want to do this even though you know it's unsafe, Unsafe Jule lets you do it.