Beyond the Random Class: Cryptographic Randomness in .NET 6+

We live in a naturally unpredictable world. Shuffling a deck of cards, picking lottery numbers, and weather patterns are all random. Software often needs to mimic this unpredictability, such as spawning enemies at random intervals in a game, selecting a random sample of data for training a machine learning model, or generating a random OTP code when authenticating a user. Computers, however, are deterministic machines. Given a particular input, the computer will always produce the same output. This makes replicating the unpredictability we live in a challenge in code.

C# offers a couple of APIs for generating random numbers, with the Random class being a common choice. However, while useful in many scenarios, its implementation isn't always appropriate, especially in security contexts where its predictability can be exploited. In those cases, alternatives like the RandomNumberGenerator, which offer a higher degree of unpredictability, are better suited.

In this article, you will learn how the Random class functions in C#, when it's appropriate to use, when and why it's not always suitable, and what alternatives to consider.

Why Computers Aren't Truly Random

Computers follow a strict set of instructions to perform a task. Given the same input, the process will yield the same output. This is great for computers, and it's what's needed for them to perform computations accurately.

When it comes to randomness, the computer has to cheat by relying on algorithms that generate seemingly random numbers. These algorithms are called Pseudo-Random Number Generators (PRNGs) and consist of mathematical formulas to produce a sequence of numbers that appear random but are still deterministic. PRNGs start with a seed value, which is the starting point for the sequence generation. Given the same seed value, a PRNG will always produce the same sequence of numbers. While not completely random, this predictability can sometimes be helpful where reproducibility is desired, like in unit tests or where reproducing the sequence carries little risk.

However, there are cases where you don't want this reproducibility. If an attacker can figure out the seed used to generate the number sequence, they can "predict" the next numbers. In specific contexts, like security, this can result in a system getting compromised. For example, if you use a PRNG to generate OTP codes for authentication, an attacker could compromise the system by figuring out the seed and generating the same codes. This is why true randomness, which is unpredictable, is essential for security-sensitive tasks.

Did you know?
Cloudflare, one of the world's leading CDN providers, generates completely random encryption keys using a wall of 100 lava lamps. A camera mounted to capture all the lava lamps takes regular pictures of the lamps and sends them to Cloudflare servers, where the image pixels are used to generate the next encryption key. Read more about it on their blog.

The Random Class: "Good-Enough" Randomness

The Random class is the de facto way of generating random numbers in most applications. It's a PRNG that's been around since the beginning of the .NET Framework. Although its underlying implementation has changed over the years, its API has remained relatively unchanged. It's intuitive, efficient, and works well in most cases.

Generating Random Integers

To use it, all you need to do is create a new instance of the class and then get the next random numbers:

var random = new Random();
Console.WriteLine(random.Next());
Console.WriteLine(random.Next());

// Output:
// 1472324096
// 2138638347

The Next() method generates a non-negative random integer. You can also pass in parameters to restrict the result if necessary:

var random = new Random();

// Generate random integer no larger than 9
Console.WriteLine(random.Next(10));

// Generate random integer between 10 and 19 (inclusive)
Console.WriteLine(random.Next(10, 20));

// Output:
// 7
// 14

When using the overloads, the start number is inclusive (if specified), and the end number is exclusive.

The Random class also has methods to generate random bytes, doubles, longs, and even pick random items from an array.

Seeding the Generator

You'll notice that no seed value was specified when instantiating the Random class above. When one isn't specified, .NET will generate the seed for you. How that seed is generated has changed over the years. In .NET Framework, the current time is used to create the seed. However, .NET Core and 5+ have improved on this by making an interop call to the operating system to generate a random seed using hardware, which is much less predictable.

You can pass in a seed using one of the constructor overloads. This is useful if you want to control the sequence of numbers generated. The snippet below demonstrates this:

// Use 78 as the seed
var random = new Random(78);

Console.WriteLine(random.Next());
Console.WriteLine(random.Next());

// Output:
// 1020951901
// 1511780237

You should get the same output as the snippet if you run the code below.

When To Use It

The Random class is simple and very efficient since all the numbers are generated by software (and not hardware, as you'll see in the next section). The generated numbers are spread across your desired range and will seem random enough for most applications. An end user shouldn't pick up any patterns.

You'll want to avoid it, though, if being able to predict values will compromise the security of your application or the integrity of your data. For example, generating password salts, nonces, and OTPs should not use the Random class since predicting those values will be disastrous. Similarly, you might not want to use the Random class for applications like gambling and picking winners for a draw since predicting the numbers could ruin the integrity of the game.

Warning on Instantiation and Thread Safety

The official Microsoft documentation advises not to re-instantiate the Random class every time you need a number. This makes sense, especially if you're working with .NET Framework, since seeds are generated using the system time. Two generators can have the same seed value if they are instantiated close together.

However, even if you're working in modern .NET (i.e., .NET 5 and up), re-instantiating the generator doesn't make sense. The generator is guaranteed to choose from a range of numbers with equal probability, meaning no number is more likely to be generated than another. This is good! But you lose that guarantee if you re-instantiate the Random class every time.

Also, keep in mind that the Random class is not thread-safe. To overcome this challenge, the documentation demonstrates how you can use locks to make sure the Random object is never called illegally.

The RandomNumberGenerator Class: Cryptographically Secure Randomness

Since computers are deterministic, generating truly random numbers is difficult. You need to incorporate entropy into the algorithm. This entropy typically comes from something in real life that exhibits random, unpredictable behaviour. Cloudflare gets entropy from a wall of lava lamps; C# gets entropy from the operating system, which, in turn, relies on underlying hardware for randomness. You can access this more secure random number generator using the RandomNumberGenerator class.

Microsoft recommends using the static implementation of the class, which will be demonstrated in this article.

Generating Random Integers

Unlike the Random class above, you need to specify the upper bound for the number you want to generate with the RandomNumberGenerator class. Another overload also lets you choose the lower bound:

// Get a cryptographically random number between 0 and 9
Console.WriteLine(RandomNumberGenerator.GetInt32(10));

// Get a cryptographically random number between 10 and 19
Console.WriteLine(RandomNumberGenerator.GetInt32(10, 20));

// Output:
// 8
// 16

Notice how the output is similar to our previous example, where we used the Random class. However, these are far less predictable, and because of entropy, it's practically impossible to repeat the sequence of numbers generated.

Generating Random Strings (.NET 8+)

The release of .NET 8 introduced several new helper methods to the RandomNumberGenerator class. One such method lets you provide a source string and desired length and then generates a random string of that length using the characters in the source string:

var availableCharacters = "ABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890";

// Get a cryptographically random string 20 characters long
Console.WriteLine(RandomNumberGenerator.GetString(availableCharacters, 20));

// Output:
// "5LZFSEVZTPI7LTFTPIJJ"

This is useful if you need to create passwords or OTPs.

Generating Random Hexadecimal Strings (.NET 8+)

The hexadecimal number system lets you represent numbers in base 16. It does this using characters 0-9 and A-F. From .NET 8 and up, you can generate random hexadecimal numbers using the GetHexString static method on the RandomNumberGenerator class:

// Get a cryptographically random hex string 5 characters long with 
// uppercase characters
Console.WriteLine(RandomNumberGenerator.GetHexString(5));

// Get a cryptographically random hex string 8 characters long with
// lowercase characters.
Console.WriteLine(RandomNumberGenerator.GetHexString(8, lowercase: true));

// Output:
// "F23BB"
// "7c92dfa5"

When To Use It

After seeing RandomNumberGenerator in action, you might wonder why you would use the Random class. But wait! As great as the RandomNumberGenerator class is, it has additional overhead. Whenever it generates a number, it calls the operating system to add entropy. This call does make the RandomNumberGenerator much slower, which you'll see in this benchmark.

The additional overhead means you should only use the RandomNumberGenerator when you actually need cryptographically random numbers. For any other use case, stick to the Random class.

Conclusion

This article explored the intricacies of generating random numbers in code, focusing specifically on the Random and RandomNumberGenerator classes in C#. You first learned how computers are deterministic and need to use PRNGs to generate seemingly random numbers. You then saw how to pick random numbers from the Random class, a PRNG generator. While simple and efficient, the Random class has limitations, especially regarding cryptographic operations.

You were then introduced to the RandomNumberGenerator, which uses entropy from the underlying operating system to generate cryptographically secure numbers which are practically impossible to predict. The article also demonstrated newer methods added to the class in .NET 8 and up, simplifying generating random strings and hexadecimal numbers.

As I looked through the source code for these classes, it's clear that very clever engineers have done much of the heavy lifting for us. However, it's still essential to understand how these classes work to use them effectively in your code base.

If you have any comments or feedback, please leave a comment below.