Hard coding
Hard-coding variables occurs when constants, configuration values, or logic are directly embedded into the code, making changes difficult. Hard-coding leads to rigid systems that require modifying the source code itself to change behavior, rather than adjusting parameters, settings, or external configurations.
Symptoms
- Magic numbers or string literals appear directly in the code.
- You find yourself searching the codebase for specific values to tweak behavior for different executions.
- The same constant value appears multiple times, making updates error-prone.
- The logic is less readable, since magic numbers donβt indicate what they represent.
- A small behavior change requires altering the core code, instead of adjusting an input parameters of config file.
Example - Hard coding and magic numbers
def calculate_area(radius):
# Hard-coded value of pi
return 3.14 * radius * radius # What if you need more precision?
def check_temperature(temperature):
# Hard-coded temperature values for thresholding
if temperature > 30: # What does 30 represent
print("It's too hot!")
elif temperature < 10:
print("It's too cold!")
Issues
- The value of pi is hard-coded as
3.14
, which can lead to precision issues. - The temperature thresholds (30, 10) are buried in the logic, making them difficult to modify.
- The meaning of 30 and 10 is unclear - are they for a specific region, season, or use case?
Solution
Using named constants and configurable parameters makes the code more readable, maintainable, and flexible.
import numpy as np # Use a library constant
= 30 # Defined constant for readability
HOT_THRESHOLD = 10 # Defined constant for readability
COLD_THRESHOLD
def calculate_area(radius): # Default parameter allows customization
return np.pi * radius * radius # Use library constant for pi
def check_temperature(temperature, hot_threshold=HOT_THRESHOLD, cold_threshold=COLD_THRESHOLD):
if temperature > hot_threshold: # Use named constants for readability
print("It's too hot!")
elif temperature < cold_threshold:
print("It's too cold!")
Example - Rigid code
This simulation hard-codes the time step and duration, making it rigid.
def run_simulation():
= 0.01 # Fixed timestep
step_size = 10 # Fixed total duration
total_time for t in range(0, total_time, step_size):
update_system(t)
Issues
- Change the step size of total duration required modifying the source code.
- The code is not reusable across different simulations.
Solution
Introduce function parameters or external configuration files for flexibility and reproducibility.
def run_simulation(step_size=0.01, total_time=10):
for t in range(0, total_time, step_size):
update_system(t)
# Calling with different configurations
=0.05, total_time=20) # Adjust without modifying the underlying code run_simulation(step_size
For larger projects, moving configuration values to a separate file or class can further improve reproducibility and maintainability. Users would then only need to adjust the (text-based) configuration file without touching the core code.
# config.yaml
simulation:
step_size: 0.01
total_time: 10
import yaml
def load_config(file_path="config.yaml"):
with open(file_path, 'r') as file:
return yaml.safe_load(file)
= load_config()
config 'simulation']['step_size'], config['simulation']['total_time']) run_simulation(config[
Key takeaways
- Use named constants for improved readability.
- Externalize configuration values to allow easy adjustments without modifying the source code.
- Configuration files or classes can further improve maintainability and reproducibility.