SQL Server vs MySQL vs PostgreSQL Delete Performance Comparison

By:   |   Comments (11)   |   Related: > PostgreSQL


Problem

Not long ago, I was working on a SQL Server database that was growing quickly and was using a lot of disk space. After an initial check, I noticed that the amount of records in the largest table was not that many (only about 150,000) and the size of the database was a little over 100GB. The large table contained only 2 columns, an ID and a VARBINARY field that was quite wide (in terms of the amount of data stored in each row), hence the size. Eventually, I had to come up with a strategy to purge old information in chunks, because a single delete would have caused blocking and other potential issues.

I was thinking to myself, what if I run into a similar situation with other RDBMS (MySQL or PostgreSQL)? Would I see similar results if I had to delete lots of rows from tables with very wide rows? I got curious enough to give it a try and will share my results. I hope you find this as interesting as I did while doing the tests.

Solution

In this tip I will be presenting the results I compiled after performing records deletions within each respective RDBMS. The tests will be done against a set of 10,000 records in a table that only has an id column and a wide string (1,000,000 characters to be precise).

Initial considerations

As you probably already have guessed by the title of this article, I'm performing the tests against a SQL Server, MySQL and PostgreSQL database instance. To keep things as fair as possible, I'm performing each test in a Linux virtual machine that has the exact same specs; for the data type of the wide column, I tried to choose something as homologous as possible between them. One last detail for the RDBMS setup is that I tested them right out of the box, without any best practices, tuning, or specific parameter settings to possibly help improve performance.

VM Specs

  • OS: Ubuntu Server 20.04 LTS
  • CPU: Intel Core i7 2.7GHz (3820QM)
  • RAM: 8GB DDR3 1600MHz
  • Volume Size: 25GB

Information related to the RDBMSs and their respective runs

I used the latest available version of each product to keep things as fair as possible.

The scenario for the test goes like this:

1 - A simple test database is created in each DB instance and a table containing the wide column, with a structure similar to this:

CREATE TABLE test(
    **id INT NOT NULL PRIMARY KEY AUTOINCREMENT,
    ***wide VARCHAR(MAX) NOT NULL
)

The syntax for the auto increment id varies within each RDBMS, but it is just to give you an idea.

The data type of the wide column is not exactly the same across all RDBMSs, but I tried to choose one for each that allows me to store the exact same data in each of them, to keep things as similar as possible.

2 - A round of 10,000 records is inserted and the following rounds of deletes are performed:

  • Time to delete 1,000 rows, as well as the size of the respective transaction log file after its completion.
  • Time to delete 2,000 rows, as well as the size of the respective transaction log file after its completion.
  • Time to delete 3,000 rows, as well as the size of the respective transaction log file after its completion.
  • Time to delete 4,000 rows, as well as the size of the respective transaction log file after its completion.
  • Time to delete 5,000 rows, as well as the size of the respective transaction log file after its completion.
  • Time to delete 10,000 rows, as well as the size of the respective transaction log file after its completion.

3 - I'm inserting a string 1,000,000 characters long, in the wide column of each row (the exact same across all of the rows), which I'm attaching as a sample in case you want to try it in your system.

4 - For the tests in each RDBMs, I made sure that only that RDBMS was running in the VM while the other two were completely shut down.

Insertion Round

These metrics were measured on a completely empty instance for each case.

RDBMS Version Wide Column DataType Time to insert 10,000 records Size of data file after the insertion
SQL Server 2019 CU6 VARCHAR(MAX) 18min 25sec 9,864MB
MySQL 8.0.21 LONGTEXT 13min 15sec 9251.52MB
PostgreSQL 12.2 VARCHAR(1000000) 4min 20sec 147MB

Deletion Rounds

After each deletion round, the log file is cleaned/wiped before the next round takes place.  For each example, there's a chart for comparison to get a better sense of the results. Note, I multiplied the time results by 10 just for visualization purposes.

1,000 Records Delete Test

RDBMS Time to delete records Size of log file after the deletion
SQL Server 44sec 72MB
MySQL 13.8sec 954MB
PostgreSQL 0.095sec 16MB
comparison chart

2,000 Records Delete Test

RDBMS Time to delete records Size of log file after the deletion
SQL Server 1min 2sec 136MB
MySQL 29.24sec 1.9GB
PostgreSQL 0.179sec 16MB
comparison chart

3,000 Records Delete Test

RDBMS Time to delete records Size of log file after the deletion
SQL Server 50sec 200MB
MySQL 2min 32sec 2.8GB
PostgreSQL 0.264sec 16MB
comparison chart

4,000 Records Delete Test

RDBMS Time to delete records Size of log file after the deletion
SQL Server 1min 36sec 264MB
MySQL 3min 43sec 3.7GB
PostgreSQL 0.134sec 16MB
comparison chart

5,000 Records Delete Test

RDBMS Time to delete records Size of log file after the deletion
SQL Server 2min 25sec 355MB
MySQL 5min 44sec 4.6GB
PostgreSQL 0.314sec 16MB
comparison chart

10,000 Records Delete Test

RDBMS Time to delete records Size of log file after the deletion
SQL Server 3min 53sec 648MB
MySQL 10min 51sec 12GB
PostgreSQL 0.639sec 16MB
comparison chart

Interesting Facts

  • Without a doubt, PostgreSQL is the clear winner of this experiment. Out of the 3 RDBMSs, it was the one that made both the inserts and deletes in the least amount of time, while keeping the transaction log much smaller. It's very likely that the transaction log management of the pg_wal files and checkpoint mechanism are very well implemented, or there's something else behind the curtain that I'm just not aware of (since I'm not a PostgreSQL expert as of right now). If you have the explanation, feel free to share.
  • Compared to SQL Server, MySQL has an advantage within the first 2,000 records, in terms of the time to perform the deletes. However, at every stage of the process it takes a big hit on the transaction log size which goes to the roof at the end.
  • It would be interesting to perform the exact same experiment after applying best practices and properly tuning each RDBMS to see if better numbers are achieved. Although, I guess it would be a bit pointless for PostgreSQL as I don't think I would get better numbers than the ones I already got.
  • In the world of RDBMSs, I have always heard that PostgreSQL is way better than MySQL. I have never used PostgreSQL in a highly transactional production environment, but after seeing the results from this experiment, I guess I can find some truth to that.
  • I know for a fact that managing delete operations like these in SQL Server, by chunks, plays a whole lot better in terms of the transaction log usage. Also, locks on the table are way more relaxed and it allows you to interact with the table even if the deletes are taking place, something that doesn't happen on larger deletes.
  • If you have to perform something similar in a real production system and you are working with MySQL, I guess a well thought out strategy has to be applied. Of course, after conducting several test runs to see the results first.
  • It would have been interesting to throw more RDBMSs in the mix like Oracle, DB2, or jump into the NoSQL space and toss MongoDB, Cassandra, Redis, etc.
Next Steps


sql server categories

sql server webinars

subscribe to mssqltips

sql server tutorials

sql server white papers

next tip



About the author
MSSQLTips author Alejandro Cobar Alejandro Cobar is an MCP, SQL Server DBA and Developer with 10+ years of experience working in a variety of environments.

This author pledges the content of this article is based on professional experience and not AI generated.

View all my tips



Comments For This Article




Monday, January 31, 2022 - 1:56:59 PM - Jeff Moden Back To Top (89715)
Thanks for the testing. I only know about SQL Server and so I have to ask...
What recovery model was SQL Server in and do the other two database engines have a similar setting and what were those settings?

Thursday, September 9, 2021 - 10:35:18 AM - Alejandro Cobar Back To Top (89218)
Thanks for sharing Jean!

Tuesday, September 7, 2021 - 6:44:22 PM - Jean Baro Back To Top (89213)
Postgresql is my favourite RDBMS but you need to count the VACCUM time PostgreSQL will take to actually remove the tuples from disk. Thanks for sharing.

Friday, October 23, 2020 - 3:30:07 PM - Shiwangini Shishulkar Back To Top (86690)
Good Test!.. I agree with everyone's comment. Yeah, in postgres the row doesn't get deleted actually - instead of that it's just marked for deletion - that's the reason why deletion is so fast here. Same happen with update statement here. Real overhead is taken care by autovacuum process later because of which space doesn't get released at the same time. However, maintenance of that is very affordable because there are various open source plugin available for postgres - which can immediately do your full vacuum after large deletion without operation's impact and release all fragmentation and extra space.

Thursday, October 22, 2020 - 3:40:32 PM - A Golovnya Back To Top (86688)
There is no magic here. For sure we need to see both initial configuration of all of three servers and sample of data was inserted in database.

Thursday, October 22, 2020 - 11:24:59 AM - Hayri Back To Top (86685)
Hi,
don't get me wrong but something is definitely not correct with this test. Could you please share your script? I tried similar test SQL server 2019 CU6 docker images insertion took just 1 minute 32 seconds, deletion of 1000 rows just took less than second. I haven't tried for Other RDMS but definitely I don't think so the number you provide correct.

Thursday, October 22, 2020 - 7:29:34 AM - Andrea Gnemmi Back To Top (86682)
Hi Alejandro and Marian,
correct Marian, the explicit vacuum is the instruction that in PostgreSQL clears the dead tuples and releases the space to the table, until that moment the space occupied by the deleted tuples cannot be reused, so this is a big difference with SQL Server. Note that vacuum can be automated using autovacuum and all its tuning techniques, otherwise it can be invoked manually even for a single table.
Regards
Andrea

Thursday, October 22, 2020 - 6:51:05 AM - Andrey Back To Top (86680)
Postgres just used pglz to compress long string.

Wednesday, October 21, 2020 - 10:07:29 PM - Alejandro Cobar Back To Top (86677)
Interesting...

Thanks for sharing Marian!

Wednesday, October 21, 2020 - 7:07:13 PM - nyx Back To Top (86676)
In PostgreSQL, as pointed by earlier comment, records marked for deletion and then collected by autovacuum process. Space is reclaimed after autovacuum, but it doesn’t shrink table size, unless you perform full blocking vacuum.

Wednesday, October 21, 2020 - 9:08:27 AM - Marian Back To Top (86670)
Hi Alejandro,

As far as I know, records are not directly deleted in PostgreSQL, but only marked as deleted. The used space in the database is also not directly released, but by a vacuum process.

Speaking for performance, the behavior is certainly quite good. But the real "hard work" is done by another process for which the times were not considered in the article.

Kind regards
Marian














get free sql tips
agree to terms