this post was submitted on 08 Nov 2025
150 points (97.5% liked)
Programmer Humor
27631 readers
1210 users here now
Welcome to Programmer Humor!
This is a place where you can post jokes, memes, humor, etc. related to programming!
For sharing awful code theres also Programming Horror.
Rules
- Keep content in english
- No advertisements
- Posts must be related to programming or programmer topics
founded 2 years ago
MODERATORS
you are viewing a single comment's thread
view the rest of the comments
view the rest of the comments
Someone explain the joke to me please!
There is a subtle scenario with read-modify-write transactions in MVCC where SQLite lacks some grace (in my opinion).
In MVCC, transactions work with a point-in-time (read "between atomic transactions") consistent "read snapshot" of the database.
Consider this example:
foo.foo.There is no conflict here because these transactions are isolated from each other via the snapshot mechanism. Transaction A's read snapshot is immutable and will not see any writes from transaction B, even if they are happening concurrently.
Now what happens in this example (from the OP):
foo.foo.foo.This is a true conflict because both transactions are trying to write to
foo, and transaction A's writes might be based on what it just read. There is no consistent way for A to proceed, because B already wrote tofoo, invalidating A's read snapshot.So SQLite handles this by returning an error to A, effectively requiring A to restart the transaction.
There are other ways this could be handled though. The DB could optimistically retry the transaction for you. There is even a special
BEGIN IMMEDIATE;statement that it could use to proactively take a write lock onfooso that the transaction doesn't get starved by other writers. But SQLite puts all of the responsibility on users to handle this.I'm not an expert, so there could be a very good reason that SQLite works this way, but it feels a bit annoying as a user.
I don't actually know off the top of my head how PostgresQL handles this particular scenario.
Thanks for the explanation.
I feel like the one who decided this behaviour, expected the users of the database, to make their own system of prioritising the transactions.
More likely: SQLite is built to be small, simple and lightweight, not to be super highly concurrent.
If this situation happens rarely, just make sure you have a retry on the query. If it happens often, switch to postgres.