Multidollar Common Programming Language Mistakes and Better Approaches

Volodymyr Pavlyshyn
2 min readJul 22, 2024

--

Modern programming languages have evolved considerably, yet certain design flaws and pitfalls remain pervasive, causing significant challenges and costs in software development. This article explores some of these common mistakes and presents alternative approaches to mitigate their impact, as discussed in a recent insightful video.

The Null Problem

One of the most notorious issues in programming is the concept of null references. Tony Hoare, the inventor of the null reference, even referred to it as his “billion-dollar mistake” due to the numerous errors and system crashes it has caused. To avoid null-related issues, several strategies can be employed:

- Option and Maybe Types: Languages like Java and C# offer options or monads that encapsulate values that may be null, providing a safer way to handle optional values.
- Dependent Types: These types allow more logic to be moved to the type system, ensuring null-free code.
Libraries and Optional Chains: JavaScript has adopted optional chaining, which, while not perfect, significantly reduces null-related errors.

Handling Errors

Exception handling in programming languages has often been criticized for its complexity and propensity to introduce bugs. Traditional try-catch blocks can lead to cumbersome and error-prone code. Alternative approaches include:

- Monadic Interfaces: Using monads to handle errors as values can improve code readability and safety. Languages like Haskell and Scala employ this method effectively.
- Algebraic Effects: This advanced technique separates error handling from the main code logic, allowing for more flexible and maintainable error management.

Coloring Functions

The concept of “coloring” functions refers to the practice of marking functions as asynchronous (async). This can lead to a cascade effect where many functions must be marked async, complicating codebases. Algebraic effects offer a solution:

  • Algebraic Effects and Handlers:By using algebraic effects, asynchronous operations and error handling can be integrated more seamlessly, reducing the need for widespread async function declarations.

Concurrent Programming and Locks

Concurrency introduces significant complexity, often leading to difficult-to-diagnose bugs. Traditional methods like locks and threads can be error-prone. Better alternatives include:

- Actor Model: This model, used in languages like Erlang and frameworks like Akka, encapsulates state and behavior within actors, making concurrency more manageable.
- Go-like Channels: Go uses channels to facilitate communication between goroutines, simplifying concurrent programming by abstracting the complexities of thread management.

Fragile Classes and Overuse of Inheritance

Object-oriented programming (OOP) and inheritance can lead to fragile base class problems and excessive complexity. Alternatives to traditional class-based inheritance include:

- Prototypal Inheritance: JavaScript utilizes prototypes, where objects inherit directly from other objects. This approach can be more flexible and less error-prone than classical inheritance.
- Data-Oriented Programming: Focusing on data and its transformations rather than the objects can lead to more maintainable and understandable code.

Conclusion

By acknowledging and addressing these common pitfalls in programming languages, developers can write more robust, maintainable, and error-free code. Embracing advanced techniques such as monads, algebraic effects, actor models, and prototypal inheritance can significantly enhance the quality and reliability of software systems. As the field of software development continues to evolve, it’s crucial to stay informed about these approaches and incorporate them into your programming practices.

--

--

Volodymyr Pavlyshyn

I believe in SSI, web5 web3 and democratized open data.I make all magic happens! dream & make ideas real, read poetry, write code, cook, do mate, and love.