In actual projects, we often need to load configuration files efficiently and threadably. In order to ensure that the configuration is loaded only once and the reading speed is as fast as possible in high concurrency scenarios, we often design some special loading solutions. Today, I will record the process of gradually abstracting advanced design concepts from specific implementation problems.
1. Requirements for efficient loading of configurations
In the early stages of the project, our requirements are simple:How to efficiently load configurations in multithreaded environments. To avoid duplicate loading and possible thread competition, we usually check if the configuration is already loaded. If it has been loaded, it will return directly; if not, it will enter the initialization stage. This ensures both performance and load uniqueness.
For example, a common implementation idea is as follows:
if (_config != null) return _config; lock (_lock) { if (_initialized) return _config ?? throw new InvalidOperationException($"Loading configuration failed {_path}"); _initialized = true; //Initialize the operation. _config = LoadConfig(); }
This code attempts to ensure the thread safety and efficiency of loading logic through the judgment outside the lock and the _initialized flag inside the lock.
2. Discussion about ?? throw
In the initialization code, we see a line:
return _config ?? throw new InvalidOperationException($"Loading configuration failed {_path}");
The purpose of this line of code is to immediately throw an exception when _config is null, prompting that the loading configuration failed. Then the question is:Is it necessary to make an empty judgment of _config here?
Design-wise, if initialization is successful, then _config should never be empty. But in actual programming, in order to defend against unexpected situations (such as exceptions during loading), developers may add such additional checks. This approach can capture possible unexpected situations to a certain extent, but it also reflects possible inconsistencies in state management.
If initialization fails, _initialized should not be set to true. That is, only after _config is ensured to be assigned correctly, set _initialized to true. This ensures consistency of the state, thus avoiding the situation where _initialized is true and _config is null during subsequent calls. Improve the code:
if (_config != null) return _config; lock (_lock) { if (_initialized) return _config ?? throw new InvalidOperationException($"Loading configuration failed {_path}"); //Initialize the operation. _config = LoadConfig(); _initialized = true;//Put this line after the _config assignment }
This will solve the problem.
3. Redundancy of _initialized variables
After further analysis, we may find that if our loading logic is rigorous enough to ensure that it enters the "loaded" state after _config is assigned correctly, then the _initialized variable actually seems redundant. In other words, you only need to rely on the non-empty judgment of _config to determine whether the configuration has been loaded.
This improvement not only simplifies the code, but also avoids potential problems caused by mismatch between the status tag and the actual value. The code can be refactored into a simpler form, focusing only on the state of _config, thus achieving more intuitive and reliable logic.
if (_config != null) return _config; lock (_lock) { if (_config != null) return _config; //Initialize the operation. _config = LoadConfig(); }
4. Classic double-decision empty check
Looking back carefully, our implementation is actually classicDouble verdict checkmodel. The outermost _config is used to improve efficiency (avoid unnecessary locks), and another check is performed inside the lock to ensure thread safety. This is a traditional way to solve the lazy loading problem.
5. Built-in packaging for programming languages
In response to the above problems, C# provides a built-in lazy loading mechanism—Lazy<T>, avoid manually managing locks and status marks.
6. Refining from concrete examples to abstract design morphemes
Looking back at the whole process, from initially to solve the performance and thread safety issues of configuration loading, to thinking about state synchronization and whether additional _initialized tags are needed, to identifying this is the classic double-determinalization check, we actually went through a refining process from concrete instances to abstract design morphemes.
This thinking pattern is very similar to the universal law extracted from specific experimental phenomena in scientific research. Through continuous in-depth understanding and abstraction of the problem, we can form a more general set of design principles that can be used repeatedly in other future scenarios and become the basic design morphemes in the hands of developers.
This method of summarizing experience from specific cases and extracting abstract design concepts not only improves the quality of the code, but also greatly improves the maintainability and scalability of the system. It is this process of continuous iteration and abstract improvement that has promoted the development of software design patterns and engineering practices.