Tony Marston's Blog About software development, PHP and OOP

Programmer Productivity takes Precedence over Paradigm Purity

Posted on 10th July 2024 by Tony Marston

Amended on 28th November 2024

Introduction
Some Universal practices
Not-so-best practices
Better practices
Previous experience
Switching Languages
Learning Object Oriented Programming
My implementing of OOP
Adding more reusable components
The results of this volume of reusability
References
Amendment History
Comments

Introduction

Let me start with some universal truths:

While there are some practices which I regard as universally applicable this is because they are not tied to any any particular programming language or paradigm, nor do they have any preferred or even suggested implementations. They are general philosophies rather than itemised rules and all they require is sensible interpretations from people who can think for themselves rather than those who can only follow the direction set by others.

The topic of "best practices" is often talked about as if it were a set of rules that is cast in stone and universally accepted by every programmer on the planet, but that is not the case. What some people regard as being best for them others will dismiss as second-best or even nowhere-near-best. In the 1980s and 90s I worked in several teams for different organisations, and each team had its own set of programming standards. Each of these had areas which I liked and some which I disliked. The idea of a single set of universal standards simply did not exist, mainly because getting a group of programmers to reach a consensus on the definition of "best practice" is like herding cats, with the likelihood of success being inversely proportional to the size of the herd.

During my time as a junior programmer I began to formulate my own set of standards by sifting through what I had encountered and filtering out any practices which I saw as getting in the way of productivity. I did not follow a rule as if it was cast in stone, I examined it, and if it did not measure up to my expectations I discarded it in favour of something better. By "better" I mean something which which increased my levels of productivity in the most cost-effective manner. This means that I am results-oriented, not rules-oriented. I am a pragmatist, not a dogmatist.

In those decades I was mostly employed by software houses where we designed and built bespoke solutions for different clients, and as we had to compete against other software houses for business we had to demonstrate that we were the most cost-effective, that we had higher rates of productivity. When I became team leader in Prolog System Limited there were no formal standards, so it was a free-for-all. This was unacceptable to me, so I made my personal standards (which can be viewed on my COBOL page) available to everyone in my team. After very little training they all saw a decrease in program bugs and an increase in productivity. This pleased my bosses so much they formally adopted my personal standards as the company standards.

I have built enterprise applications in COBOL, UNIFACE and PHP, and because they are different what works in one language may not work in another. While the objectives remained the same, the means of achieving those objectives were different in each language. In other words "what needs to be done" was consistent, but "how can it be done" has an infinite number of variations, and each team has its own version of "how" based on their set of experiences. It is one thing to learn the capabilities of the language, but it is another to learn how to utilise those capabilities to best effect. Whenever a new version of the language was released I took the time to examine any new features, and if any looked as if they could add value to the code I would evaluate them to compare the costs against their benefits. Anything which enabled me to replace duplicated code with reusable code, or enabled me to do something with less code, and hence increased my productivity, was always beneficial.

When I switched to PHP with its Object Oriented capabilities at the start of this century I carried on with this habit. After learning the mechanics of Encapsulation, Inheritance and Polymorphism I began writing code which incorporated high cohesion and loose coupling to produce the best results with the least amount of effort, thus contributing to higher levels of productivity. After completing several modules I would often see repeating patterns that could be turned into code that could be reused instead of being duplicated. Another practice I follow to reduce the amount of code which I write is not to write code that is not necessary, to always choose the simplest solution instead of a complex one which invariably ends up by resembling a Rube Goldberg machine.

I didn't follow what was later pointed out to be "best practices" for the simple reason that I didn't know that they existed. Fellow developers began to ridicule my work by saying You are not following *this* principle or *that* principle, therefore your work is wrong and inferior. When I examined these so-called "superior" techniques I detected a huge flaw in their arguments - the results which they achieved were not as good as mine, for reasons such as:

By following only those practices which actively contribute to my high levels of productivity (what used to take me 1 week in COBOL and then 1 day in UNIFACE I reduced to 5 minutes in PHP) I have found myself discarding quite a number of practices and principles which other programmers regard as being sacrosanct. They criticise me for having the audacity to break their precious rules while totally ignoring the results which I have achieved. I single-handedly designed and built the RADICORE framework which I released as open source in 2006, and I used this framework to single-handedly design and build TRANSIX, my first ERP package application, in 2008. This has now grown into the GM-X Application Suite which has been sold on three continents.


Some Universal practices

The following ideas are not tied to any any particular programming language or paradigm, nor do they have any preferred or even suggested implementations. They are general philosophies rather than itemised rules and all they require is sensible interpretations from people who can think for themselves rather than those who can only follow the direction set by others.


Not-so-best practices

Ever since I started publishing articles regarding my experiences with PHP other programmers have told me that the code which I write and the methods which I use are completely wrong simply because I am not following "best practices". By this they mean that I am not following the same set of practices as they are, but I see no reason why I should. I have worked with many teams in many organisations, and the only thing in common with what they called "programming standards" was that every team had its own set. When I became team leader in the 1980s the practices which I had been accumulating and documenting were formally adopted as the company's COBOL standards. My methods cannot be wrong simply because they work, and something that works cannot be wrong just as something which does not work cannot be right.

I did not follow these "best practices" for one simple reason - I did not know that they existed. When I became aware of them and examined them I quickly realised that by changing my code to conform to this "advice" it would have a negative effect on my productivity by using code that was more complicated than mine, more convoluted than mine and less efficient than mine, so I did the only sensible thing which was to ignore it, to consign it to the dustbin, to flush it down the toilet.

Every time I am told "you should be doing it this way" my first response is to ask "Why? What bad thing happens if I don't? What good thing happens if I do?" If there is no proof, preferably with a code sample, that following the rule has benefits then I see no reason why I should follow it.

To counter my detractors I have written a number of blog posts which identify the "best practices" which I choose to ignore as well as explain precisely why I ignore them and why I believe that my own practices are better. These are listed below:

  1. Object Relational Mappers are EVIL

    The "need" for an ORM is caused by deliberately using one methodology to design the database and a separate methodology to design the software, thereby causing a mismatch. My solution is not to create this mismatch in the first place. I design the database first, then build the software around that design using my framework.

  2. Dependency Injection is EVIL

    Some programmers insist that I use a complicated mechanism to inject every dependency even when there is not a selection of dependencies to choose from. Dependency Injection (DI) cannot be achieved without polymorphism, and the more polymorphism you have the more opportunities you have to share code using DI. Every Model class in my framework shares the same set of public methods, and as each Controller calls these methods it means that any Controller can be reused with any Model. While I do inject entities into services where there are multiple choices, I do not inject entities into entities as there is never more than a single choice.

  3. Not-so-SOLID OO principles

    I only follow those principles which I consider to be appropriate.

  4. I fail to GRASP these principles

    I only follow those principles which I consider to be appropriate.

  5. Singletons are NOT evil

    Singletons can be implemented in one of two ways - either as a separate method within each class, or as a static method within a single singleton class. The latter choice does not have the problems encountered in the former, so those coders who think that singletons are bad are obviously using the inferior implementation.

  6. Getters and Setters are EVIL

    Getters and setters (also known as Accessors and Mutators) grew out of the idea that every entity which has multiple items of data, such as a database table and its collection of columns, must have a separate class property defined for each of those columns. This then forces those columns to require separate pieces of code to put data into and get data out of the object. This is not how any of the DBMS systems which I have ever used, whether hierarchical, network or databases, have operated, nor is it how the various programming languages have dealt with their data. In languages such as COBOL table columns were never defined or addressed individually, they were always defined in aggregates (also known as records, blocks, structures or composite data types).

    When I was building the prototype application which provided the basis for my later framework I noticed very quickly that PHP also had the ability to deal with aggregated data in the form of arrays. I also noticed that data being submitted from an HTML document was presented in the form of the $_POST array and the data being retrieved from the data was also presented as a FETCH array

    In each of my table classes I do NOT have a separate property for each column with its own getter and setter, I pass all the data around, both in and out, in a single array argument which is precisely how it is handled in the HTML front end and the SQL back end. This reduces the amount of code which I have to write, and directly contributes to loose coupling which is supposed to be a good thing.

  7. Object Interfaces are EVIL

    I don't use object interfaces as they were invented to get around a restriction in statically typed languages. PHP is not statically typed so it does not have this restriction, therefore adding code to avoid a restriction which does not exist would be a violation of YAGNI. Besides, I get better results from using abstract classes.

  8. Object Associations are EVIL

    The "approved" technique is to handle associations and aggregations using custom code with each entity. I prefer to use standard solutions which I have built into the framework.

  9. Inheritance is NOT evil

    The rule "favour composition over inheritance" was formulated by someone who never learned how to use inheritance properly. Instead of inheriting from one concrete class to create a different concrete class you should only ever inherit from an abstract class. As well as giving instant access to reusable code it also enables the Template Method Pattern which lies at the heart of framework design.

  10. Composition is a procedural technique for code reuse

    As an argument against inheritance some bright spark argued that "Inheritance is a procedural technique for code reuse", but as someone who programmed in COBOL, the most widely used procedural language of all time, I know that this is not true. Instead I contend that it is object composition which deserves that description.

  11. Namespaces are for numpties

    Namespaces were added to PHP to solve a problem when importing third-party libraries into your application as there may be name clashes between that library and your application. This is because the library was developed without any knowledge of your application, so any name clashes will produce an error which won't be noticed until after the library has been installed. All those libraries which are imported using Composer would have been built using namespaces as standard. It should be obvious to even a junior programmer that if every third-party library automatically includes namespaces then there is no need to include them in your application as that would be overkill. It should also be obvious that PHP itself does not need to use namespaces as any name clashes would be reported when the code being developed was run, and would have been fixed before the code was released.

    It may come as a surprise to some of you numpties out there but RADICORE is not a library which can be installed using Composer, it is a framework which is an application, and as such forcing it to use namespaces would not offer any benefits and so would be a violation of YAGNI.

  12. Autoloaders are abominations

    Autoloaders are the solution to the self-inflicted problem of requiring multiple files to be loaded from multiple locations before a class can be instantiated. This arises because far too many junior programmers do not understand the meaning of encapsulation and high cohesion.

    I avoid the problem by only have one class file for each table which is always stored in a standard location, and that file contains all the methods which are required by that class. The idea that a class should not contain more than X number of methods was obviously dreamed up by some schoolboy who can't count above ten without taking his shoes and socks off. Provided that the principle of high cohesion is maintained I do not recognise any limit on the size of a class.

  13. Decoupling is delusional

    The idea that "decoupling" your software is a good idea shows a complete lack of understanding of the term "coupling". If there is a call from one module to another then they are coupled whether you like it or not, and the strength of that coupling is either tight or loose. To "decouple" means to remove the call. To introduce a third module to act as an intermediary between the first two does not remove the coupling, it actually doubles it by replacing one method call with two.

  14. Strict typing is for stick-in-the-muds

    PHP was designed from the outset to be dynamically and weakly typed, and it was provided with the ability to perform automatic type coercion which could convert any string value into a different type depending on the context. An error is produced only when the conversion fails, and that can only happen when the developer fails to sanitize the user's input. The idea that forcing every developer to insert additional code to perform this type casting manually, thus replacing the automatic conversion, could only have been dreamed up by a dogmatist and a pedant, someone who does not understand the meaning of pragmatism and efficiency.

    Strict typing was only supposed to be turned on for user-defined functions when the developer deliberately used the declare(strict_types=1) directive, but those stupid core developers make a cock-up and forced it to be turned on automatically for all internal functions.

    Forcing developers to insert lines of code which are not necessary is therefore a violation of YAGNI.

  15. Value objects are a worthless waste of time

    I have read quite a few articles recently extolling the virtues and advantages of value objects in PHP. I have read what they have to say, and as far as I am concerned the whole idea is complete bunkum as all they are doing is introducing a complicated mechanism to achieve something which is ridiculously simple. The notion that it is "better" to create an object for each value, especially one that encapsulates it business rules, just demonstrates that they have a perverted view of what the word "better" actually means. Does it produce less code? Certainly not. Does it reduce the possibility of errors in that code? Certainly not. In fact, the increase in the amount of code you have to write simply increases the possibility of errors creeping in. If you don't believe me then consider the following:

    1. Before you can access any value object you must first create a class which can be instantiated into an object. You must then populate this class with all the relevant methods to deal with that column's value.
    2. This also means that within any object which references that column you must also include code to load that class, instantiate it into an object, an then call its methods.

    I develop nothing but database applications, known as enterprise applications, and my current ERP application contains 500 tables and 5,000 columns. If I were to create an extra class for each of those columns that would result in 5,000 extra classes, extra code to load and instantiate those 5,000 classes, plus whatever code would be necessary to call the methods on those 5,000 objects.

    Considering that I can handle all those columns WITHOUT the need to create, maintain, instantiate and call methods on all those value objects, then I consider the whole idea of value objects to be a complete waste of time and a violation of YAGNI. Not only that, I have never seen anybody identify the problem for which value objects are the solution, and it is a policy of mine that if I don't have a problem then I don't need a solution.

  16. I don't do Domain Driven Design

    An enterprise application may have to deal with different business areas for that organisation, such as Order Processing, Invoicing, Inventory/Stock Control and Shipments. Although each of these areas handles totally different data and totally different business rules, these differences can all be handled in a similar fashion. All the data is spread across numerous database tables, and each table can be handled in exactly the same way. The business rules will be different, but they can all be handled in the same way. By building these similarities into a separate system with components that can be shared, each business area can then be regarded as a separate sub-domain or sub-system as part of a larger domain or system.

    The RADICORE framework was built to develop database applications as it provides pre-written components to deal with all the similarities. Each subsystem is developed as an extension or add-in to the framework and then run under the control of the framework.


Better practices

In the previous section I identified those so-called "best practices" which I ignore. In this section I will describe my path to a set of practices which have yielded better results.


Previous experience

Everything started with the following basic observations from my previous experience of developing database applications using COBOL and UNIFACE. Note that during this time I had worked with a mixture of hierarchical, network and relational databases.

In a large ERP application, such as the GM-X Application Suite, which is comprised on a number of subsystems, each subsystem has a unique set of attributes:

Despite the fact that these two areas are completely different for each subsystem, they each have their own patterns and so can be handled using standard reusable code provided by the framework:


Switching Languages

Switching to a new language is not just about switching to a different syntax, it sometimes means switching to a different way of writing and structuring your code. With COBOL we used an ordinary text editor, so there was no debugger. The code we put into a single source file had a data division for data definitions, and a procedure division for code. The code itself consisted of statements, sentences (one or more statements terminated with a period), paragraphs and optional sections. The PERFORM statement was for performing a paragraph or section within the current source file while the CALL statement was for passing control to a subprogram/subroutine which was compiled from a separate source file. As a junior developer I had been taught to create a small number of monolithic programs which were large and complex, but later on, while developing my first Role Based Access Control system, I learned that a larger number of smaller but simpler programs were easier to build and maintain.

UNIFACE was totally different. Instead of a text editor we had to use a special piece of proprietary software known as a Graphical Form Painter (GFP) with which we built form components which referenced entities within its Application Model, an internal database. Entities (tables) were first constructed within the Application Model to define their columns, indexes and relationships, then exported to produce CREATE TABLE scripts in the chosen DBMS. Proc code was entered into various pre-named triggers where different triggers could be fired depending on different mouse movements. Each form component was compiled in the GFP and then run via the UNIFACE Runtime Engine. The early versions were 2-tier as the form component combined the user interface and the business logic into a single unit with all database access logic provided by a separate database driver which was provided by UNIFACE, where there was a different version for each DBMS. We never had to write any SQL queries ourselves as they were all generated by the database driver. Unfortunately each entity in a component could only build SQL queries for that particular table, which meant that it was not possible to build queries with JOINs to other tables, thus producing what is known as the N+1 Problem.

UNIFACE v7.2.04 allowed a form component to be split into two by separating the user interface from the business logic which was moved to a separate separate service component, one for each entity, thus converting it from 2-tier into 3-tier. The transfer of data between the outer presentation layer and the middle business layer was via XML streams. This version also introduced the idea of Component Templates.

UNIFACE was originally developed to create desktop applications, but when they saw the growing demand for internet applications they decided to add in the capability to create web pages. I found their approach to be too clumsy and too clunky. I only worked on one project which attempted to use this new capability, and it was a complete and utter disaster. I decided that UNIFACE was the wrong choice when it came to building web applications, so I looked for a different language that was fit for this purpose. That is when I discovered PHP. I played with it, I liked what I saw, (especially the ability to Generate dynamic web pages using XSL and XML), so I decided to build another version of my development framework to see if I could boost my levels of productivity.

PHP uses scripts which can be constructed using a simple text editor, but I prefer to use a full blown Integrated Development Environment (IDE) with colour coding and debugging. PHP is a multi-paradigm language in that you can write code which is completely procedural, or you can use objects if you want to. It does not force any particular structure on the developer, so you can choose whatever structure you like whether that is monolithic or multi-tier. I found PHP very easy to learn due to the quality of the online manual and the sample code which was available in numerous online resources and books. I was particularly impressed with arrays which could be indexed, associative or multi-level and which were much better at handling collections of data items, such as database data, than the records or aggregated data types that I had used before.

I quickly accustomed myself to the fact that a PHP script could be activated either by an HTTP GET request from the browser's address bar, or an HTTP POST from an HTML form. I learned how to overcome PHP's stateless nature by using the session handling functions, and I learned how to activate another script from within the current script using the header('location ...') function.


Learning Object Oriented Programming

While I had heard about this thing called "object oriented programming" during my work with UNIFACE I hadn't a clue what it meant or what made it so special, so I did some research on the interweb thingy. I found a few descriptions which used vague and meaningless terminology, but then I came across the following description:

Object Oriented Programming entails writing programs which are oriented around objects, thus taking advantage of Encapsulation, Inheritance and Polymorphism to increase code reuse and decrease code maintenance.

After looking at the descriptions in the PHP manual I identified the distinguishing features, those which exist in OO languages but not in non-OO languages, as follows:


My implementation of OOP

At this time I was not aware of all those things called "best practices" which I was supposed to follow, so I went ahead to see how I could take advantage of encapsulation and inheritance (with polymorphism still be a mystery) in order to fulfil the aims of OOP which is to increase code reuse and decrease code maintenance. My previous experience had already taught me that the more reusable code I had at my disposal then the less code I had to write, and that being able to achieve results by writing less code was the path to higher productivity.

There were several features I found in UNIFACE which influenced my approach to rebuilding my framework in PHP:

I experimented with ways to implement these ideas in PHP by first building a small sample application as a Proof of Concept (POC) using the following steps:


Adding more reusable components

As this prototype proved that all my ideas worked I then proceeded to build a brand new version of my framework which I called RADICORE. Full details can also be found in Evolution of the RADICORE framework. While building this framework I added to my library of reusable software in the following ways:


The results of this volume of reusability

As a result of these efforts I have managed to create these Levels of reusability.

Having this amount of reusable software at my disposal means that when I am developing an application there is a huge amount of code which I don't have to write, which can be summarised as follows:

  1. I don't need to waste any time with OOD or DDD
  2. I don't need to waste time in designing class hierarchies
  3. I don't need to waste any time choosing which design patterns to use
  4. I don't need to waste time in building each table class (Model) by hand
  5. I don't need to waste time in designing methods and properties for each table class
  6. I don't need to waste time with writing code to validate user input
  7. I don't need to waste time writing code for associations and aggregations
  8. I don't need to waste time with designing Controllers for each of my Model classes
  9. I don't need to waste time designing and building HTML Views for each web page
  10. I don't need to waste time writing code for each of the components on a web page
  11. I don't need to waste time in writing code to perform standard SQL queries
  12. I don't need to waste time with writing a separate Data Access Object (DAO) for each table
  13. I don't need to waste time writing code to build new transactions
  14. I don't need to waste time designing and building any Access Control Lists (ACL)
  15. I don't need to waste time designing and building menu structures
  16. I don't need to waste time writing detailed program specifications
  17. I don't need to waste time changing method signatures after changing a table's structure

References

These are reasons why I consider some ideas on how to do OOP "properly" to be complete rubbish:


Amendment History

28 Nov 2024 Added Some Universal practices
16 Aug 2024 Added Better practices
27 Jug 2024 Added I don't do Domain Driven Design

counter