Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

WIP draft of randomness concept #816

Draft
wants to merge 2 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -4,3 +4,5 @@ bin/configlet
bin/configlet.exe
tmp/
Manifest.toml

.idea/
11 changes: 6 additions & 5 deletions concepts.wip/randomness/.meta/config.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
{
"blurb": "TODO",
"authors": [],
"contributors": []
}

"authors": [
"colinleach"
],
"contributors": [],
"blurb": "Julia has a wide range of ways to generate random values for modelling and simulations."
}
299 changes: 298 additions & 1 deletion concepts.wip/randomness/about.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,300 @@
# About

TODO: add information on randomness concept
Many programs need (apparently) random values to simulate real-world events.

Common and familiar examples include:
- A coin toss: a random value from `('H', 'T')`.
- The roll of a die: a random integer from 1 to 6.
- Shuffling a deck of cards: a random ordering of a card list.

Generating truly random values with a computer is a [surprisingly difficult technical challenge][truly-random], so you may see these results referred to as "pseudorandom".

***Important***: This Concept does _not_ cover cryptographically secure random numbers, which are a much more difficult challenge.

However, well-designed libraries like the [`Random`][Random] module in the Julia standard library are fast, flexible, and give results that are amply good enough for most applications in modelling, simulation and games.

Julia divides random functionality into multiple locations:

- Just a few basic but very versatile functions in `Base`, which are always available.
- A wider range of options in the `Random` module.
- More specialized functionality in packages which need to be installed before use (and are not available in Exercism).

`Random` is part of the standard library and likely to be pre-installed, but you will need to add `using Random` at the top of your program to bring its contents into the namespace.

## The `rand()` function

What this function does depends on the parameters you give it.
There are _many_ options.

With no parameters, it generates a float between 0 and 1.

A single integer parameter generates a vector of that length.

```julia-repl
julia> rand()
0.10261774967264703

julia> rand(5)
5-element Vector{Float64}:
0.24134501977563894
0.5664193284851202
0.9804412082089355
0.6229551330613335
0.47589221741904664
```

For a different range, just shift and scale the result appropriately.

The example below uses [broadcasting][broadcasting] for the subtraction, which will be covered in another Concept.
The `.-` simply applies this arithmetic to each vector element.

```julia-repl
# numbers between -1.0 and +1.0
julia> (rand(5) .- 0.5) * 2
5-element Vector{Float64}:
-0.5303906759076336
0.9635682226775855
-0.048823697086981754
0.465842804648374
0.9880834344780736
```

With a type as the only parameter, `rand` will use the `typemin` and `typemax` as limits.
This is probably not what you want!

For random integers, we can supply a range, plus optionally how many values to generate.

```julia-repl
julia> rand(Int64)
-9159538335234594326 # not very useful

julia> rand(1:10, 5)
5-element Vector{Int64}:
1
1
1
4
7
```

In the `rand(1:10, 5)` example above, notice that there are (coincidentally) repeating values, because each pick is independent.
This is "sampling with replacement", discussed in more detail below.

Alternatively, supply an array or tuple, and `rand` will return a random entry:

```julia-repl
julia> rand([4, 9, 16, 25])
16

# coin flip
julia> rand(['H', 'T'])
'H': ASCII/Unicode U+0048 (category Lu: Letter, uppercase)

# mixed types in tuple
julia> rand( (1, 3.2, "name"), 2 )
2-element Vector{Any}:
1
"name"
```

### Sampling with or without replacement

Imagine that we have a bag containing 3 red balls and 4 green balls, and we randomly pull a ball from the bag.
To get a second ball, there are two possibilities:

1. Replace the first ball in the bag and give everything a good shake before pulling out another.
The number of balls is now the same as before (7), and _the ratio of red to green is also the same_.
2. Put the first ball on the table before pulling out a second.
Now there are only 6 balls in the bag, and _the red:green ratio depends on the color of the first ball_.

Scenario 1 is with replacement, scenario 2 is without, and _they give different results_.

To simulate sampling without replacement in Julia, there are a couple of options.

Simplest (_and within Exercism the only option_), use `Random.shuffle()` to put the entries in random order, then use the first `n` elements.
This is fine for small problems but may not scale well to large collections: `shuffle` needs to generate the full array, even if you only want a small fraction of it.

To do sampling-with-replacement "properly", install the `StatsBase.jl` package.
That provide the `sample()` function with a full range of options.

We can reasonably hope that similar functionality will be added into `Random` in a future release, to make it part of the standard library (code samples in this document were tested with Julia 1.11).

## Working with Distributions

Until now, we have concentrated on cases where all outcomes are equally likely.
For example, `rand(1:100)` is equally likely to give any integer from 1 to 100.

Many real-world situations are far less simple than this.
As a result, statisticians have created a wide variety of [`distributions`][probability-distribution] to describe "real world" results mathematically.

### Uniform distributions

The `rand()` function described above is used when all probabilities are equal.
This is called a [`uniform`][uniform-distribution] distribution.

### Gaussian distribution

Also called the "normal" distribution or the "bell-shaped" curve, this is a very common way to describe imprecision in measured values.

For example, suppose the factory where you work has just bought 10,000 bolts which should be identical.
You want to set up the factory robot to handle them, so you weigh a sample of 100 and find that they have an average (or `mean`) weight of 4.731g.
This is extremely unlikely to mean that they all weigh exactly 4.731g.
Perhaps you find that values range from 4.627 to 4.794g but cluster around 4.731g.

This is the [`Gaussian distribution`][gaussian-distribution], for which probabilities peak at the mean and tails off symmetrically on both sides (hence "bell-shaped").
To simulate this in software, we need some way to specify the width of the curve (_typically, expensive bolts will cluster more tightly around the mean than cheap bolts!_).

By convention, this is done with the [`standard deviation`][standard-deviation]: small values for a sharp, narrow curve, large for a low, broad curve.
Mathematicians love Greek letters, so we use `μ` ('mu') to represent the mean and `σ` ('sigma') to represent the standard deviation.
Thus, if you read that "95% of values are within 2σ of μ" or "the Higgs boson has been detected with 5-sigma confidence", such comments relate to the standard deviation.

There will be more to say about this in the [`Statistics`][statistics] Concept.

## The `randn()`function

Short for "random normal", this is similar to the floating-point variant of `rand()` except that values are distributed as a Gaussian with mean 0 and standard deviation 1.

Again, you may want to scale the raw output from `randn` for standard deviation, and displace it for the mean.

```julia-repl
julia> randn(5)
5-element Vector{Float64}:
-2.072481073894565
-1.4138480399699225
0.004142896124356481
0.4670319074567224
0.554217625875170
```

It is hard to tell from looking at the output that these cluster closer to zero than for a uniform distribution.
If you doubt it, generate 1000 or more and plot them to make it more obvious.

## The `Random` module

[This module][randommod] contains the next tier of functionality, omitted from `Base` to help minimize the size of Julia's default configuration.

`Random` supplements `rand` and `randn` in `Base` with mutating versions, `rand!` and `randn!`.

A useful addition is [`randstring`][randstring], which generates a string of given length.
By default, this uses upper- and lowercase letters plus digits 0 to 9, but other choices can be specified.

```julia-repl
julia> using Random

julia> randstring(20)
"BoJnIxrS33pJiWggXZQV"
```

Additionally, there is a `bitrand` function to generate a random [`BitArray`][bitarray] of specified length.

```julia-repl
julia> bitrand(5)
julia> bitrand(5)
5-element BitVector:
1
1
0
0
1
```

### Shuffles and permutations

To randomly shuffle entries in a `Vector` we have [`shuffle`][shuffle]; also `shuffle!` to mutate the input vector in-place.

```julia-repl
julia> v = ['A', '1', '2', 'J', 'Q', 'K'];

julia> shuffle(v)
6-element Vector{Char}:
'K': ASCII/Unicode U+004B (category Lu: Letter, uppercase)
'1': ASCII/Unicode U+0031 (category Nd: Number, decimal digit)
'A': ASCII/Unicode U+0041 (category Lu: Letter, uppercase)
'J': ASCII/Unicode U+004A (category Lu: Letter, uppercase)
'2': ASCII/Unicode U+0032 (category Nd: Number, decimal digit)
'Q': ASCII/Unicode U+0051 (category Lu: Letter, uppercase)

# shuffles are random:
julia> shuffle(v)
6-element Vector{Char}:
'2': ASCII/Unicode U+0032 (category Nd: Number, decimal digit)
'K': ASCII/Unicode U+004B (category Lu: Letter, uppercase)
'A': ASCII/Unicode U+0041 (category Lu: Letter, uppercase)
'Q': ASCII/Unicode U+0051 (category Lu: Letter, uppercase)
'J': ASCII/Unicode U+004A (category Lu: Letter, uppercase)
'1': ASCII/Unicode U+0031 (category Nd: Number, decimal digit)
```

Sometimes it is useful to have the shuffled indices instead.
For this, use [`randperm(n)`][randperm] where n is the length of the sequence.

```julia-repl
julia> randperm(6)
6-element Vector{Int64}:
6
2
4
1
3
5
```

In effect, the example above gives the same results as `shuffle(1:6)`.

Related functions include [`randsubseq`][randsubseq] for pulling out entries with fixed probability, and [`randcycle`][randcycle] for cyclic permutations.
These require some specialized knowledge, so check the documentation if they are of interest to you.

### Seeds and algorithms

Several random number generator (RNG) algorithms are built into `Random` as standard, and anyone with appropriate mathematical skills can add more.
Such things are well beyond the scope of this document!

A more common reason for working with RNGs is to specify a `seed`, which has the effect of making the sequence of "random" outputs reproducible from one run to the next.

Such reproducibility is not appropriate in production code, but it can help with testing and debugging.

## Other packages

Outside Exercism, there are many installable packages relating to randomness, probability and statistics.
For some more information, see the [`Statistics`][statistics] Concept.

### The `StatsBase.jl` package

Most of the [`StatsBase`][statsbase] functions are quite technical and not relevant to this document.

The exception is [`StatsBase.sample`][sbsample], which provides a full implementation of sampling with or without replacement (see an earlier section, above).
There are also functions for weighted (non-uniform) sampling.

### The `Distributions.jl` package

The `uniform` and `normal` (or Gaussian) distributions were described above.

The `Random` module also contains `randexp` to sample from the [Exponential Distribution][expdist], which is related to the (very common) [Poisson Distribution][poisson].

For a much wider range of options, there is the [`Distributions.jl`][distributions] package for those with an appropriate background in statistics.


[gaussian-distribution]: https://simple.wikipedia.org/wiki/Normal_distribution
[probability-distribution]: https://simple.wikipedia.org/wiki/Probability_distribution
[Random]: https://docs.julialang.org/en/v1/stdlib/Random/
[sampling-with-replacement]: https://www.youtube.com/watch?v=LnGFL_A6A6A
[standard-deviation]: https://simple.wikipedia.org/wiki/Standard_deviation
[truly-random]: https://www.malwarebytes.com/blog/news/2013/09/in-computers-are-random-numbers-really-random
[uniform-distribution]: https://www.investopedia.com/terms/u/uniform-distribution.asp#:~:text=In%20statistics%2C%20uniform%20distribution%20refers,a%20spade%20is%20equally%20likely.
[reproducibility]: https://docs.julialang.org/en/v1/stdlib/Random/#Reproducibility
[statsbase]: https://juliastats.org/StatsBase.jl/stable/
[sbsample]: https://juliastats.org/StatsBase.jl/stable/sampling/#StatsBase.sample
[broadcasting]: https://docs.julialang.org/en/v1/manual/arrays/#Broadcasting
[bitrand]: https://docs.julialang.org/en/v1/stdlib/Random/#Random.bitrand
[bitarray]: https://docs.julialang.org/en/v1/base/arrays/#Base.BitArray-Tuple{Any}
[randperm]: https://docs.julialang.org/en/v1/stdlib/Random/#Random.randperm
[shuffle]: https://docs.julialang.org/en/v1/stdlib/Random/#Random.shuffle
[randommod]: https://docs.julialang.org/en/v1/stdlib/Random/#Random-numbers-module
[randstring]: https://docs.julialang.org/en/v1/stdlib/Random/#Random.randstring
[randsubseq]: https://docs.julialang.org/en/v1/stdlib/Random/#Random.randsubseq
[randcycle]: https://docs.julialang.org/en/v1/stdlib/Random/#Random.randcycle
[seed]: https://en.wikipedia.org/wiki/Random_seed
[expdist]: https://en.wikipedia.org/wiki/Exponential_distribution
[poisson]: https://en.wikipedia.org/wiki/Poisson_distribution
[distributions]: https://juliastats.org/Distributions.jl/latest/
[statistics]: https://exercism.org/tracks/julia/concepts/statistics
2 changes: 0 additions & 2 deletions concepts.wip/randomness/introduction.md
Original file line number Diff line number Diff line change
@@ -1,3 +1 @@
# Introduction

<!-- TODO: add introduction for randomness concept -->
8 changes: 4 additions & 4 deletions concepts.wip/randomness/links.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[
{
"url": "https://docs.julialang.org/en/v1/stdlib/Random/",
"description": "Julia Manual"
}
{
"url": "https://docs.julialang.org/en/v1/stdlib/Random/",
"description": "Manual page for random functionality in the Base and Random modules."
}
]