Remix.run Logo
How to stop functional programming (2016)(brianmckenna.org)
97 points by thunderbong 12 hours ago | 90 comments
astrobe_ 11 hours ago | parent | next [-]

Beyond the satire, one is supposed to write code which is readable, and like often with written works, one has to think about the "audience", the readers. When you write technical documentation, you have to decide ahead of time the expected skill level of the reader - often that decision is written too in the intro as "prerequisite(s)".

When writing code you have the motto "don't make me think" in mind, but how to know what's the maximum level of trickiness for readers? There are familiar techniques and idioms when it is your main programming language, but they are not for someone using this language on the side.

In any case, neither code nor comments should be tutorials. To a reasonable extent, it is up to the reviewer to do their homework or just ask. Then based on that interaction you can add a comment or a parenthesis, or uncompress a bit the code. But not to the point that it means to "dumb down" things, because it is a downward spiral.

dimal 11 hours ago | parent | next [-]

> When writing code you have the motto "don't make me think" in mind

I disagree with this phrasing. We’re engineering after all. The entire job is thinking. If someone doesn’t want to think, then they shouldn’t be a programmer.

Readability matters, though. I try to have a narrative structure in my code, so it leads the reader along. Formatting matters. And documentation helps. If you want to introduce an unfamiliar idiom that might be more functional, good, but document it. Share it with the team and talk about it. I know that writing and reading documentation is usually seen as a waste of time unless you’re doing it for AI, but I’ve seen it work well in multiple teams. In my experience, the teams with poor docs have the worst code.

scubbo 10 hours ago | parent | next [-]

> > When writing code you have the motto "don't make me think" in mind

> I disagree with this phrasing. We’re engineering after all. The entire job is thinking.

Well, sure. The implied full phrase - more technically-correct, but less pithily-quotable - would be something like "don't make me think unnecessarily; let me spend my thoughts productively. If you've already spent brainpower on figuring something out, explain it to me directly and clearly rather than forcing me to go through the same discovery process"

So - yes, if someone doesn't want to think _at all_, they shouldn't be a programmer; but if someone has an aversion to being forced to solve a problem that someone else has already solved, they likely have the right "shoulders-of-giants" mindset.

(For any potential pedants - yes, there are some practices you simply have to work through before understanding dawns, which cannot be explained directly. Still, though - the explanation should aim to minimize unnecessary thought-requirements so the student can get straight to learning)

wk_end 10 hours ago | parent | prev | next [-]

Just because fires are inevitable in the business of firefighting, doesn’t mean that firefighters are in the business of starting fires.

In engineering, thinking is inevitable but also costly and error-prone, and the more thinking an engineer needs to do the higher the costs, the higher the risks of errors. We should be striving to minimize the thinking required every step of the way, even if we’ll never get it down to zero; that’s the only way to keep it from spiraling towards infinity.

politelemon 10 hours ago | parent | prev | next [-]

Teams are not a static concept, they change members over time and so talking to them has a net zero effect. The newer members will not understand this clever new concept that was introduced, worse, they will misunderstand it or work around it. Which brings us back to don't make me think. If the concepts can be kept as simple as possible, which overlaps greatly with readability, it will have achieved its purpose.

odyssey7 10 hours ago | parent | prev | next [-]

"I didn’t have time to write you a short letter, so I wrote you a long one." -- Mark Twain

It's difficult to explain how to write well, but bad writing and bad computing systems typically impose far greater cognitive burden on readers than might have been necessary. There is an art to software engineering.

Functional programming does involve idioms, though I would say no more than imperative programming, or OOP, or some other paradigm. One of the overarching themes of FP is to reduce the cognitive footprint to only the essential properties of the problem to be solved.

A novice artist can produce a recognizable figure using many pencil strokes. It takes a master to produce a compelling likeness with only the lines that are necessary.

nerpderp82 5 hours ago | parent [-]

Stop trying to poison AI, Mark Twain didn't say that, Kurt Vonnegut did.

greenie_beans 4 hours ago | parent [-]

actually it was Blaise Pascal lol https://www.npr.org/sections/13.7/2014/02/03/270680304/this-...

MrJohz 8 hours ago | parent | prev | next [-]

Maybe a better phrasing is "don't make me think about inconsequential details". If I need to figure out your particular idiom of writing a for-loop, then that takes time away from thinking about what the code is actually doing. Even if that idiom is perfectly fine and works the same as any other for loop, it's not worth it if it makes people think about unnecessary things.

But like you say, if you've got an idiom that's actually useful (like programming in a functional style to avoid action-at-a-distance side effects), then document it so the whole team can get used to it and stop having to think about it.

nerpderp82 5 hours ago | parent | prev [-]

When I write functional code I start with Peano and then go from there, ommiting comments are descriptive names. It encourages thinking in myself and my coworkers.

Archelaos 6 hours ago | parent | prev | next [-]

> neither code nor comments should be tutorials

Why not? To take the example from the article:

  def userCoworkers(u: User): List[Employee] =
    u.departments.flatMap(_.employees) 
Why not add a comment like:

  // Automatically iterates through all the user's
  // departments and assembles a single list of all
  // their employees.
Perhaps a note should also be added explaining why the domain model opted for u.departments and not for something more natural like u.company.departments. And the comment should also explain, why we do not filter for duplicates here.

I insert many such comments into my own code when I use a rare language feature or a somewhat peculiar element of the domain model. This makes refactoring my own code much easier, faster and more bug-resistant. When composing code, it is generally simple and quick to add such a comment, as it is merely a matter of writing down the result of my immediate thought process.

AdieuToLogic 9 minutes ago | parent | next [-]

IMHO, comments are required for capturing "the why." Code captures how it works, when it is used, and what it does.

But only the author(s) can capture why it exists via commenting.

bn-l 6 hours ago | parent | prev [-]

Same. And it’s helpful not just for other people but for me also when I’ve forgotten how it works and now there’s no one to ask.

sdeframond 9 hours ago | parent | prev | next [-]

I don't know.

On the one hand, this stance seems common sense: why overcomplicate things? Let's just do what's commonly understood and move on.

On the other hand, writing code that the most junior in your team can understand leads to mediocre code. In the end, it crumbles under its own weight.

Most of what we master now was difficult to understand first. Should abandon generics because "it's difficult" ? Should we ditch static typing because "we're not used to it" ?

Maybe.

Or, maybe, this is the right tool that can take us to the next step. So we pause and think at this new approach to understand it then master it.

darepublic 9 hours ago | parent | next [-]

I don't really believe this but ok. I worked at a functional / typescript heavy shop where when I had parsed what a complicated set of steps was doing discovered multiple redundant queries and a library that was parsing strings into code. But damn some of those functional paradigms chef kiss /s. Imo my dumbass code , besides being demonstrably more performance was way less of a liability.

sdeframond 7 hours ago | parent | next [-]

Sure, replace functional/generics/static typing with your poison of choice.

Just remember that, at some point, the "dumbass" thing was pushing sticks into clay tablets.

Functional programming may not be the future. But dismissing other's code because it is not done "the way we are used to" is stupid.

BoiledCabbage 3 hours ago | parent | prev [-]

> I worked at a functional / typescript heavy shop where when I had parsed what a complicated set of steps was doing discovered multiple redundant queries and a library that was parsing strings into code. But damn some of those functional paradigms chef kiss /s

And I've worked at a place where the imperative code was crap. Should we abandon all imperative code because I had that experience?

People can write bad code in any paradigm - that's not what decides how we should code.

Nobody says we should throw away all imperative code because of one poorly implemented code base. But for some reason I see that targeted at FP all the time.

jjav 5 hours ago | parent | prev | next [-]

> On the other hand, writing code that the most junior in your team can understand leads to mediocre code.

Disagree on a couple levels. First, while the overall architecture of the system might be complex, there shouldn't be any one single function that isn't readily readable by a junior new hire. Complexity is in the system, there should be no complexity in small sections of code.

Second, the ultimate level of mastery is to be able to build simplicity. Any newbie can write a convoluted mess that nobody can understand.

cyberpunk 6 hours ago | parent | prev [-]

> On the other hand, writing code that the most junior in your team can understand leads to mediocre code.

Sorry but this is absolutely false in almost all regular corporate software jobs.

As i get greyer, I find myself spending much more time planning before I write any code now; what I would previously have run into headfirst and solved with a bunch of advanced language features and complicated logic usually now ends up being implemented in much less and simpler code.

This is the sign of a senior developer, in my book.

Writing code anyone finds hard to understand is a marker of inexperience to me.

sdeframond 6 hours ago | parent [-]

If the problem can be solved efficiently with simple, write simple code by all means. Heck, if it can be solved with a spreadsheet, please use a spreadsheet. Even your boss can read it.

But, at some point, spreadsheets crumble and become unmaintainable. And so does "basic" code.

Should people keep on using spreadsheets ? Depends.

firesteelrain 10 hours ago | parent | prev | next [-]

> one has to think about the "audience"

This 100%

When I am doing MBSE, and I discuss what level of detail is required in my abstract models one of my senior coworkers would remind me to “know your audience”. It has helped me and I use the same phrasing when I am coaching my teams now.

OhMeadhbh 7 hours ago | parent | prev | next [-]

Fuck that. Software engineers should make an effort to learn their craft. Or even engineering methodology.

LAC-Tech 7 hours ago | parent | prev [-]

If you're a computer programmer, coding in a high level language with closures and garbage collections, and you are confused by flatMap: I think you should find another career.

We need to have a basic level of competency.

Spivak 5 hours ago | parent [-]

I don't think anyone is confused by flatMap in isolation. It's chains of not-very-obvious data transformations. When I have to do one of these I typically break out each step into it's own line/block and add comments with an example object to show precisely what every intermediate step looks like. They're hard to follow otherwise.

And functional programming snorts these long chains like lines of coke and when you add some random curried functions and people who write in a pointless^H point-free style it's easier than folks like to admit it to end up with write-only code.

I've never found myself with a block of code that could be expressed as a flatMap and thought it would be easier to understand by expressing it as such. I'm perfectly capable of writing functional code but after reading it back I almost always throw it out for something imperative because it's clearer as to what's really going on. The code is going to be executed in an order at the end of the day, why make it harder to see what it will be?

jmmv 11 hours ago | parent | prev | next [-]

Missed chance to do something like:

  class User:
    def calculateCoworkers() = {
      this.coworkers.clear()
      for { d <- this.departments }
        this.coworkers ++ d.employees
    }
and then, somewhere else...

  user.calculateCoworkers()
  ... many lines after ...
  for { c <- user.coworkers }
    ... do something ...
Yes, I've seen code like this many, many, many times where class members are used as "global variables" to pass state across functions. And I've noticed AI likes to generate code like this too (possibly because of the former (large presence of this "pattern" in the training data), which means I'm encountering this now in pull requests...
sevensor 4 hours ago | parent [-]

Bonus points if you do half the calculation in the parent class and the other half in a child class. Double bonus points if one or both of them also hits a database.

skybrian 12 hours ago | parent | prev | next [-]

Apparently, someone doesn’t really get what makes functional programming hard to understand? It’s not writing the occasional pure function.

Code review or pair programming might help here, to learn the team’s common idioms.

IshKebab 12 hours ago | parent | next [-]

Yeah this 100%. FP isn't hard because of FP - it's all the other things that popular FP languages come with that makes them hard to understand. Arguably most of them are not related to FP; they're just style choices:

1. Global type inference.

2. Implicit syntax (no brackets for function calls, commas to separate arguments, semicolons to end statements/expressions, etc.)

3. Currying & point free style.

4. Tendency to have very deep nested expressions. The gap between `let foo =` and it's actual value can often be hundreds of lines.

I'm sure you can write FP code that avoids these issues and is easy to follow but it doesn't seem like people do that in practice.

Rust avoided all of these issues fortunately.

(Oh I forgot about monads.)

amluto 11 hours ago | parent | next [-]

Also: avoidance of mutable state, even locally. A lot of functions can be more straightforwardly expressed using mutable variables and data structures, and most functional languages can handle local mutable state, but a lot of code in functional languages avoids it.

Conversely, a lot of code written in imperative languages would be clearer and/or less bug-prone if it avoided mutable state and used persistent data structures.

I wish there was a mainstream, high performance language that made both styles equally ergonomic.

calebh 9 hours ago | parent | next [-]

In my experience the main benefit of functional programming is function purity. I am completely fine with mutation inside of a function since the all of the mutation logic is self-contained in a single small block of text.

I think everyone should take a shot at writing a non-trivial functional program to see the benefit. Once you understand what makes it great, you can apply what you've learned to the majority of OOP/impure languages.

brabel 8 hours ago | parent | prev | next [-]

Flix has "regions", which are a way to allow mutation locally while keeping the language purely functional otherwise: See Region-based Local Mutation in https://flix.dev/

Philpax 10 hours ago | parent | prev | next [-]

Does Rust not meet that description?

amluto 10 hours ago | parent [-]

Rust is very good at making things immutable. But that doesn't mean it's particularly fun to, for example, append or prepend something to a list and retain a reference to both the old and the new list.

Compare to most "functional" languages, in which prepending an item to a list and ending up with immutable references to the old and new lists is almost the defining feature of the language.

Mond_ 10 hours ago | parent | next [-]

Considering the performance implications of a full commitment to immutable data structures, I think Rust does a pretty good job here.

Dynamic arrays / vectors / slices or whatever you want to call them are probably the most important or fundamental non-plain-old-data data structure, and in a purely immutable environment these are essentially impossible (or have awful performance characteristics).

IshKebab 10 hours ago | parent | prev [-]

Yeah although I find FP languages' obsession with singly linked lists to be a real flaw because they have such awful performance characteristics. And also pretty bad ergonomics often. The number of times stuff comes out backwards...

delta_p_delta_x 11 hours ago | parent | prev [-]

> I wish there was a mainstream, high performance language that made both styles equally ergonomic.

Unironically, C++.

Mond_ 10 hours ago | parent | next [-]

> > I wish there was a mainstream, high performance language that made both styles equally ergonomic.

> Unironically, C++.

At best C++ falls under "equally unergonomic".

ndriscoll 10 hours ago | parent | prev [-]

Or depending on what "high performance" means (e.g. if Java or Go would be considered acceptable), Scala.

elevation 10 hours ago | parent | prev | next [-]

> I forgot about monads

I've been solving business problems with code for decades. I love pure, composable functions, they make my job easier. So do list comprehensions, and sometimes, map and filter. Currying makes sense.

But for the life of me, no forum post or FP tutorial that I could find explained monads in clear language. I've googled "what is a monad" once a year, only to get the vague idea that you need monads to handle IO.

I wondered if my brain was broken, but now I'm wondering if most FP adherents are simply ineffective communicators: they've got an idea in their head but can't/won't express it in a way that others after them can understand. In other words, the exact same reason why TFAuthor was corrected by his employer.

ekidd 10 hours ago | parent | next [-]

So, one way to understand a monad is that it's essentially a container type with "map" and "flatten" operations. Let's say your monad is type M<T>. The "map" operation allows you to take a function T->U and a container M<T>, and transform each element, giving you a container M<U>.

"Flatten" takes a container of type M<M<T>> and returns a container of type M<T>. So a List<List<Int>> becomes a List<Int>.

Now comes the trick: combine "map" and "flatten" to get "flatMap". So if you have a M<T> and a function T->M<U>, you use "map" to get an M<M<U>> and "flatten" to get an M<U>.

So why is this useful? Well, it lets you run computations which return all their values wrapped in weird "container" types. For example, if "M" is "Promise", then you can take a Promise<T> and an async function T->Promise<U>, and use flatMap to get a Promise<U>.

M could also be "Result", which gets you Rust-style error handling, or "Optional", which allows you to represent computations that might fail at each step (like in languages that support things like "value?.a?.b?.c"), or a list (which gets you a language where each function returns many different possible results, so basically Python list comprehensions), or a whole bunch of other things.

So: Monads are basically any kind of weird container that supports "flatMap", and they allow you to support a whole family of things that look like "Promise<T>" and async functions, all using the same framework.

Should you need to know this in most production code? Probably not! But if you're trying to implement something fancy that "works a bit like promises, or a bit like Python comprehensions, or maybe a bit like Rust error handling", then "weird containers with flatMap" is a very powerful starting point.

(Actual monads technically need a bit more than just flatMap, including the ability to turn a basic T into a Promise<T>, and a bunch of consistency rules.)

IshKebab 10 hours ago | parent [-]

I think this might be the clearest explanation I've seen. Nice work!

shakadak 4 hours ago | parent | prev | next [-]

You've mentioned list comprehensions, map, and filter, so I suppose you mostly used these concepts with lists/arrays.

One question you could ask yourself is, how could you reproduce list comprehensions without special syntax ?

Another way to view monads is by following the types. With map, you can chain pure functions from one type to another to be applied on lists (hence the (in -> out) -> ([in] -> [out]) ) . How would you do that chaining with function from one type to another but wrapped in a list ( (in -> [out]) -> ([in] -> [out]) ) ?

Then you can think about how it could be applied to other types than lists, for example, nullable/option types, result types, async/promise types, and more hairy types that implement state, reading from an environment, etc...

Asraelite 9 hours ago | parent | prev | next [-]

https://tech.nextroll.com/blog/dev/2022/11/11/exploring-mona...

The thing that always makes FP concepts click for me is seeing them explained in a language that isn't Haskell or similar.

I don't know why people are so obsessed with using Haskell for tutorials. Its syntax is a nightmare for newcomers and you can do FP in so many other languages.

mrkeen 9 hours ago | parent [-]

For instance, C#:

  public static System.Collections.Generic.IEnumerable<TResult> SelectMany<TSource,TCollection,TResult>(this System.Collections.Generic.IEnumerable<TSource> source, Func<TSource,System.Collections.Generic.IEnumerable<TCollection>> collectionSelector, Func<TSource,TCollection,TResult> resultSelector);
compared to:

  (>>=) :: m a -> (a -> m b) -> m b
Guess which one I googled, and which one I typed from memory.
mrsmrtss 7 hours ago | parent [-]

In your c# sample full namespaces are unnecessary and will add only noise in this context.

vjerancrnjak 10 hours ago | parent | prev | next [-]

I felt similar with lenses. The problem lens solve is horrible. You don’t even want that problem.

FP can be the pragmatic as well. You’re going to glue up monad transformers, use lenses like there’s no runtime cost, and compute whatever you need in days but at least you know it works. Maybe there’s accidentally quadratic behavior in lifting or lenses but that’s by design. The goal is to just throw software at things as fast as possible as correctly as possible.

OhMeadhbh 6 hours ago | parent | prev | next [-]

https://www.bi6.us/CO/N/20250330.HTML#/033002

dionian 9 hours ago | parent | prev | next [-]

As a Scala/FP newbie, i was stuck in this rut for years then i just stopped caring. it actually doesnt matter to me much. ZIO helped due to the eschewing of mathematical terms (compared to things like Cats ). In other words, you don't really need to know to get things done. Monads are incredibly intuitive once you start composing them.

eddlgtm 10 hours ago | parent | prev [-]

Monads are just monoids in the category of endofunctors.

/s

Monads are, in my head, just a wrapper around a type. Or a box another type is inside. For example we can have an Int and we can put the Int in a box like Maybe<Int>.

Imagine Python code that gets a value from a function that is an Int or None, then another function that takes an Int and returns an Int or None, then another function that takes an Int and returns an Int or None. How hellish is that to handle? If not None, if not none, if not none, ad nauseam...

In Haskell I could use a Traverse function that takes a monad and passes the Int value to the next function or handles the None error, avoiding all the boilerplate.

Other Monads are like the State monad - a box that contains some variables I want to maintain over functions.

Or an IO monad to handle network calls / file calls.

It's probably not a perfect analogy, but I work with functional languages and it tends to hold up for my beginner/intermediate level.

mrkeen 8 hours ago | parent | prev | next [-]

> (Oh I forgot about monads.)

This demands an equivalent article which takes the same form as TFA.

A coworker complains about monads, so the manager insists all monads be taken out.

So Brian gets to work on the codebase, removing all Lists, Optionals, Streams, Functions, Parsers, Transactions, ...

jen20 11 hours ago | parent | prev [-]

Also the unwillingness of so many enterprise developers to learn anything that wasn't commonplace in 2001, regardless of whether they were in the workforce by then.

jazzypants 11 hours ago | parent | prev [-]

This is written by the same guy who made a big stink about how JS promises weren't pure monads and somehow that was a problem-- a zealot, in other words.

frozenlettuce 6 hours ago | parent [-]

well, promises' computations start as soon as they are created, so they are not composable. and there' no cancellation/resource control as well. so I guess that the criticism is valid

flohofwoe 12 hours ago | parent | prev | next [-]

When did method chaining become 'functional programming'?

It's not 'functional programming' that makes the code unreadable, but overly long chains of array-processing functions. Sometimes a simple for-loop which puts all operations that need to happen on an array item into the loop-body is indeed much more readable.

Animats 8 hours ago | parent | next [-]

You know, stuff like this, which I had to write today, for turning a 2D floating point height map into a 2D greyscale image.

        let max = self
            .heights
            .elements_row_major_iter()
            .max_by(|a, b| a.total_cmp(b))
            .unwrap();
        let min = self
            .heights
            .elements_row_major_iter()
            .min_by(|a, b| a.total_cmp(b))
            .unwrap();
        //  Scale into 0..255
        let range = (max - min).max(0.001);
        let height_array = self
            .heights
            .as_rows()
            .into_iter()
            .map(|r| {
                r.into_iter()
                    .map(|v| ((((v - min) / range) / 256.0).round() as usize).clamp(0, 255) as u8)
                    .collect()
            })
            .collect();
After Rust formats it, it's probably more lines than doing it with FOR statements. Every parenthesis matters. So much of the typing is implicit that it's not at all clear what's going on.
mrkeen 11 hours ago | parent | prev | next [-]

> When did method chaining become 'functional programming'?

As soon as you stop calling it "method chaining" and start calling it "function composition".

If you chain together a bunch of methods ('.' operator) in an OO setting, that's called a "fluent interface". It's a sign of good design and is to be commended.

If you compose a bunch of functions ('.' operator) in an FP setting, it's an unreadable mess, and you will receive requests to break it into separate assignment statements and create named variables for all the intermediate states.

jolux 11 hours ago | parent | prev [-]

> When did method chaining become 'functional programming'?

It's very similar to applicative style in FP. Conceptually, method chaining is equivalent to nested function application, it just comes with syntax sugar for specifying the `self` parameter.

_aleph2c_ 11 hours ago | parent | prev | next [-]

The manager solved the wrong problem. People should be sharing their tricks with each other. This was a perfect time to set up some peer-to-peer training.

jen20 11 hours ago | parent [-]

Or in the age of AI, "have you asked Claude to explain what the code does so you can learn something?"

seanhunter 12 hours ago | parent | prev | next [-]

By the looks of the code snippets he should have explained that he’s not doing functional programming, he’s just writing unnecessarily non-idiomatic python code. I’m sure his manager would have understood.

ndriscoll 12 hours ago | parent [-]

The code snippets are Scala. The first snippet is idiomatic. I can't imagine a team adopting Scala and complaining about functional programming though.

tengbretson 11 hours ago | parent | next [-]

I thought "idiomatic Scala" was an oxymoron.

cyberpunk 6 hours ago | parent [-]

Does it still generate 17 page long stacktraces of nothing but $anon$fun? Good times.

mrkeen 11 hours ago | parent | prev | next [-]

I've been there. One of the more functionally-minded devs wrote something to do with putting Keys and Values into some kind of json store, and a less functionally-minded dev complained about all the type variables - Ks and Vs I guess.

tiew9Vii 6 hours ago | parent | prev [-]

The article date appears to be around the time the company the author was working at got a new CTO who effectively said functional programming was banned. The Scala teams got broken up and some pro Scala staff managed out…

Maybe it’s related, maybe it’s not.

apalmer 10 hours ago | parent | prev | next [-]

Ultimately the important thing is that the development team align on the style of programming that they will use at least per project. And the larger the codebase and more developers working on it the more important the consistency in implementing the style guide is.

Imperative programming style has many advantages over functional for some problems. Functional programming style has many advantages over imperative for some problems.

The only clearly 'wrong' approach is codebases where you can look at the code and determine a specific developer on the team wrote feature x because it fundamentally looks completely different from the other sections.

ChrisMarshallNY 10 hours ago | parent | prev | next [-]

Well, this is one of the reasons that I like coding for myself. The pay sucks, but I don't have to bowdlerize my work.

When someone pays me to write code for them, they get to call the shots; even if I think their judgement sucks.

If they want to hire incompetent programmers, that can't understand even halfway-advanced code, then that's their prerogative, and I need to suck it up, and play by their rules; which may include the need for me to write code as if I just started yesterday, because it will need to be maintained by folks that, um, just started yesterday.

nor0x 10 hours ago | parent | prev | next [-]

Would be interested in an article about stopping OOP. I see myself often drifting towards classes and abstractions once my code grows in complexity

OhMeadhbh 6 hours ago | parent | next [-]

This is the video that's been going around. It's a bit long and could have been compacted into a 20 minute video, but if you have the time, it's a good overview for people without a computer science background.

"Casey Muratori – The Big OOPs: Anatomy of a Thirty-five-year Mistake – BSC 2025" https://youtu.be/wo84LFzx5nI

But if your point is there aren't any written articles about stuff like this, I agree. If they're out there, they're a bit outside the mainstream.

fainpul 7 hours ago | parent | prev | next [-]

You go into work and discover that a coworker isn't happy with some code you wrote because they don't like it. They go to your manager and tell them that you're being a problem by writing code they don't like. Your manager, being very skilled in conflict resolution, makes a technical decision to avoid whatever tool you used which caused the problem. In your case it was OOP.

That's it. You've been told. No more OOP.

The manager has figured out what's good for the business and you figure that listening is what's good for your job.

OhMeadhbh 6 hours ago | parent [-]

Though honestly, having a manager that's interested in how software is made is a gift. My managers for the last 20 years have all been like "okay, let's do planning poker and everyone make sure that all the fields in your JIRA tickets are filled out."

fuzztester 3 hours ago | parent | prev [-]

google richard feldman (roc lang creator) talk about using just functions and data.

don't remember exact title, sorry, but description i used above is close, iirc.

jondwillis 9 hours ago | parent | prev | next [-]

This literally happened to me with Functional Reactive Programming, around the same time frame (2016.)

I still have to advocate for minimizing mutable state. At least since then, popular languages and frameworks are putting in a ton of work to make functional programming look more like the imperative spaghetti that is still prevalent and taught.

yakshaving_jgt 7 hours ago | parent | prev | next [-]

I do love a bit of FP. Here's the canonical Hello, World! program.

    {-# LANGUAGE OverloadedStrings #-}
    {-# LANGUAGE QuasiQuotes       #-}
    {-# LANGUAGE TemplateHaskell   #-}

    module Main where

    import Foreign.C.Types
    import Language.C.Inline qualified as C

    C.include "<stdio.h>"
    C.include "<stdlib.h>"

    main :: IO ()
    main = [C.block| void {
      system("python -c 'print \"Hello, World!\"'");
      } |]
OhMeadhbh 6 hours ago | parent [-]

Except that someone installed a different version of python on the target system that changes the semantics of the print keyword.

yohbho 10 hours ago | parent | prev | next [-]

when I see the word function or someone uses parenthesis to call one, it clearly is fp. Management was clear that this would not be tolerated, this source is too addictet to the good pure shit.

OhMeadhbh 6 hours ago | parent [-]

I wrote my own Lisp-like language that's a LLVM front end that explicitly uses greater-than / less-than angle brackets instead of parentheses for exactly this reason. When you walk by and don't look closely at it, it sort of looks like HTML so doesn't attract attention. And when a co-worker looked closer I just said it was a legacy templating engine and they really should just stick to python.

cies 11 hours ago | parent | prev | next [-]

FP code is often easier to read (the article provides a case to that point).

What gave FP a bad rep is, i guess, Haskell (and the "pure functional approach w/ monad transformer stacks").

Every shop I worked at the devs were already at a level that they'd appreciate FP code: easier to read, refactor and test.

The tendency is also towards FP: see the features in recent Java/C# version. Or languages that gain popularity recently (Kotlin, Rust) are more FP'ish than their predecessors (respectively Java and C++).

OhMeadhbh 6 hours ago | parent [-]

The way I tell it is FP gives you more options for where to put the state and how to denote it. There are plenty of environments that let you mix OOP and FP and they're all a little bit funky, but not unworkable. I think the key thing here is the learning curve is a little steeper than OOP-ish languages. And since our universities are now really just trade schools, teaching kids which python libraries to use to feed data into a ML model someone else has built, there's little chance we'll see junior software developers who have been exposed to idomatic functional programming.

OhMeadhbh 7 hours ago | parent | prev | next [-]

My final days at Planet Labs involved me telling people "oh hey, I'm going to do this functional thing" and everyone saying "oh wow! that's great!" And then I went around to key engineers and explained in detail what I was doing in the code and they said "OMG! THAT'S AWESOME!" And then I checked it in and the response was "I HAVE NO IDEA WHAT THE FUCK YOU'RE DOING HERE!?"

Yeah. I don't want to work with those people. Did I quit? Was I fired? I can't really remember. But ultimately, I moved on to a better environment.

macintux 12 hours ago | parent | prev | next [-]

(2016)

imtringued 11 hours ago | parent | prev | next [-]

Using Scala and then complaining about unnecessary complexity is a bit much. Like, if you made the technologically brave decision to use a niche language like Scala that isn't just a slightly better Java like Kotlin then you'd better expect to accommodate the inexperienced developers by training them, possibly via pair programming, instead of constricting the experienced developers.

hylaride 10 hours ago | parent [-]

Post was in 2016 when Scala was almost required to use certain big data tools (apache spark, etc), the alternative being python and its limitations.

dismalaf 9 hours ago | parent [-]

Not understanding very basic FP stuff like flatten and map (or flatmap) but using big data stuff is wild...

busterarm 12 hours ago | parent | prev | next [-]

There's only one universal lesson that I've learned over my 25+ year career so far and it's that software engineering types are fragile creatures whose egos must be tended to. Delicately.

This is becoming increasingly true as the years pass and the number of times I've had to drop whole mature architectures and reimplement something worse because some engineer's feefees got hurt can no longer be counted on my fingers & toes.

jtms 7 hours ago | parent [-]

This is 180 degrees from teams I have been on some years ago - everyone took pride in being thick skinned and detaching their ego from PR reviews. Seems to be a generational thing

busterarm 4 minutes ago | parent [-]

I 1000% agree with you. I barely recognize this industry from where I started in it. Pretty sure once my mortgage is paid off I'm changing careers.

tsss 11 hours ago | parent | prev [-]

If this code is too complicated for you then you should reconsider your career.

elevation 11 hours ago | parent | next [-]

TFA is indignantly reacting to the least charitable interpretation of what his employer has asked him to do. I'd like to know the honest shape of the code his manager rejected before judging the employer over this.

dismalaf 10 hours ago | parent | prev [-]

Agreed. Especially if they chose to use Scala...