Performance Benchmarking on
Many-to-Many (M:N) Relationships
with Variations in Indexing and
Sizes of Object Databases

Performance Comparison: PostgreSQL vs. MariaDB

The tests presented here are identical to those conducted for the MariaDB database, as documented at this link. However, they are now executed for PostgreSQL to provide a comparative analysis against MariaDB.

The object model and the characteristics of the created test databases, including various indexing strategies and sizes, are described in the following section.

Object Model and Test Data for Performance Tests

The object model consists of 12 Java classes N1-N4, M1-M4 and S1-S4.

Object Model

Blue arrow relationships between classes are M:N (many-to-many). All M:N relationships have 1-3 child objects at random.

Black arrow relationships between classes are M:1 (many-to-one).

Twelve object databases were created to evaluate how performance is affected by the number of objects and the extent of indexing coverage.

The tests are categorized into three groups based on database size, with object counts of 12 x 100K, 12 x 1M, and 12 x 2M (where K=Kilo, M=Million), as shown in the table below.

Each category applies four different levels of indexing: no indexing (none), indexing only the value field in each class (value fields), indexing only the relationships of each class (all relations), and the broadest scope, which includes indexing all relationships in addition to the value field (value fields, all relations).

RootDB Object Databases in PostgreSQL
Database Name Number of Objects Indexes Creation Time (s), CT Objects per Second PostgreSQL/ MariaDB
CT Coeff.
Database Size (GB) PostgreSQL/ MariaDB
Size Coeff.
           
ex100Knx 12 x 100K none 69 17395 1,26 1,47 2,28
ex100Kvx 12 x 100K value fields 70 17169 1,33
ex100Krx 12 x 100K all relations 353 3397 4,34
ex100Kvrx 12 x 100K value fields, all relations 257 4667 6,28 1,19 1,36
           
ex1Mnx 12 x 1M none 685 17521 1,25 15,29 2,95
ex1Mvx 12 x 1M value fields 716 16764 1,42
ex1Mrx 12 x 1M all relations 3537 3393 4,59
ex1Mvrx 12 x 1M value fields, all relations 3557 3373 4,58 8,35 1,18
           
ex2Mnx 12 x 2M none 1354 17722 1,21 30,42 2,95
ex2Mvx 12 x 2M value fields 1417 16941 1,53
ex2Mrx 12 x 2M all relations 7067 3396 4,51
ex2Mvrx 12 x 2M value fields, all relations 7144 3360 4,56 16,63 1,22

Processor = AMD Ryzen 7 3700X, Disk Samsung SSD 860 EVO M.2 1TB

Creating object databases without indexing (none) or with only single-field indexing (value fields) is highly efficient, allowing 16,764 to 17,722 objects to be written per second, regardless of database size. This represents a performance improvement of 1.21× to 1.53× compared to the corresponding tests on MariaDB.

Indexing all 14 relationships (with or without value fields) significantly reduces write speed, limiting it to 3,393 – 4,667 objects per second. Despite this reduction, the performance remains highly efficient, offering extensive indexing coverage while achieving a 4.34× to 6.28× speed improvement over MariaDB. While PostgreSQL is widely recognized for its strong indexing capabilities, the magnitude of its advantage over MariaDB in this scenario is remarkably striking.

Similar to MariaDB, a particularly positive aspect is that increasing the database size does not degrade object write performance. The write speed remains consistently stable, regardless of the extent of indexing.

However, the trade-off with PostgreSQL is its higher disk space consumption, ranging from 1.18× to 2.95× compared to MariaDB. Notably, the largest differences - 2.28× and 2.95× - occur in databases without indexing, which is somewhat unexpected. This suggests that PostgreSQL's basic row storage structure inherently requires more space than MariaDB's.

The results above clearly indicate that indexing can be applied more liberally in PostgreSQL than in MariaDB, as execution time for writing data to the database is typically far more critical than disk space usage in most database applications. Moreover, indexing is essential for improving query performance, and its efficiency holds significantly greater importance than the disk space it consumes.

RQL Queries with Variations in Indexing and Object Database Sizes

12 Object Databases Tested in PostgreSQL

The search conditions used for performance testing of the object databases described above are shown in the image below.

In all tests, the same search conditions are used, which searches M1 class objects from the database. The red X marks indicate the classes N3, M4, and S4, where search conditions are applied to the value fields within the data structure of class M1. The paths from the root class M1 to the search condition classes are shown with red dashed lines.

The found M1 objects are returned as complete data structures, including all objects that are reachable from the M1 root class. This of course includes classes N3, M4, and S4, which have the value search conditions, but also objects from classes M2, M3, and N4, which are reachable from class M1. All reachable classes from class M1 are marked with blue X marks.

Object Model

Blue arrow relationships between classes are M:N (many-to-many). All M:N relationships have 1-3 child objects at random.

Black arrow relationships between classes are M:1 (many-to-one).

The search conditons for the three value fields in classes N3 (value=976), S4 (value=50) and M4 (value=770) are represented as a RQL query expression below.

RQL query:

(m2.n3.value = 976, m2.m3.s4.value = 50, m2.n3.m4.value = 770)                        

The Java source code that executes the RQL query expression for the class M1 in the object database is shown below.

//Variable rdb is the opened RootDB object database.
//RootStore object is created for the M1 class.
RootStore r2 = rdb.getRootStore("fi.rootrql.ex100Knx.M1");
Query query =  Query.createQuery()
        		.addQ1Query(r2,
                    "(m2.n3.value = 976," +
                    " m2.m3.s4.value = 50," +
                    " m2.n3.m4.value = 770"
                    + ")");
List<M1> list = query.queryQ1Objects();
rdb.close();

An example of query results is shown below, containing the complete data structures of the retrieved four M1 objects. Note that, for example, the expanded [0]=M2 object has a list of three M3 objects in its field m3 and only the expanded object [2]=M3 has the searched value 50 in its s4=S4 object. Thus, the lists may contain objects that do not match the search criteria, but are included in the results because the returned M1 objects contain their complete data structures.

Query Result Example: The four found M1 objects are returned in a list, with the first M1 object expanded to display the value fields where the search conditions were applied.

ex100KnxQuery100Kresult1

The twelve object databases with variations in indexing and object counts were all tested using the same query described above. The results categorized by database size and indexing strategy are presented in the table below.

Note the column 'MariaDB/PostgreSQL Search Coeff.,' which compares the values in the 'Search' column to those provided by the MariaDB database.

RootDB Object Databases in PostgreSQL Execution Times / Seconds  
Database Name Number of Objects Indexes Number of Found Objects Total Create Objects Search MariaDB/ PostgreSQL
Search Coeff.
               
ex100Knx 12 x 100K none 4 12,244 0,039 12,205 0,41
ex100Kvx 12 x 100K value fields 4 4,164 0,032 4,132 0,84
ex100Krx 12 x 100K all relations 4 1,096 0,010 1,086 1,52
ex100Kvrx 12 x 100K value fields, all relations 4 0,317 0,028 0,289 1,24
             
ex1Mnx 12 x 1M none 40 87,96 0,192 87,77 0,59
ex1Mvx 12 x 1M value fields 40 48,54 0,200 48,34 0,66
ex1Mrx 12 x 1M all relations 40 40,46 0,181 40,28 1,28
ex1Mvrx 12 x 1M value fields, all relations 40 2,61 0,169 2,44 1,98
             
ex2Mnx 12 x 2M none 80 199,51 0,411 199,10 0,74
ex2Mvx 12 x 2M value fields 80 108,43 2,906 105,52 0,72
ex2Mrx 12 x 2M all relations 80 51,04 0,366 50,67 0,97
ex2Mvrx 12 x 2M value fields, all relations 80 4,43 0,295 4,13 2,92

Processor = AMD Ryzen 7 3700X, Disk Samsung SSD 860 EVO M.2 1TB

The number of M1 objects found is 4 in the smallest databases with 12 x 100K objects. The number of objects found increases with the size of the database, so that a database with 12 x 2M objects has 80 matches for the query conditions.

The focus of the results is of course on execution times and their changes depending on the size of the database and the indexing used. Three execution times are shown, which are the search time of the query (Search), the creation time of found M1 objects (Create Objects), and the total time (Total) summing up these two times. The results show that the performance can be evaluated solely on the basis of the search time of the query (Search), since the object creation time (Create Objects) per result object seems to be approximately the same in all cases.

However, PostgreSQL's performance can be assessed and compared to the results of the same tests conducted on MariaDB, which are presented in a similar table at this link. To facilitate comparison, the PostgreSQL results table includes an additional column, MariaDB/PostgreSQL Search Coeff., which quantifies the relative performance of searches in PostgreSQL compared to MariaDB. A value greater than 1 indicates that PostgreSQL performs better, while a value less than 1 signifies that MariaDB is faster. For example, a coefficient of 2.0 means that PostgreSQL executes the same search twice as fast as MariaDB, whereas a coefficient of 0.5 indicates that MariaDB is twice as fast as PostgreSQL.

According to the results, MariaDB outperforms PostgreSQL when no relationships are indexed, specifically in tests using the none and value fields indexing strategies. MariaDB demonstrates greater efficiency, with a MariaDB/PostgreSQL Search Coefficient ranging from 0.41 to 0.81, corresponding to a 2.44× to 1.23× faster search time compared to PostgreSQL.

PostgreSQL outperforms MariaDB when indexing strategies include relationships, specifically in the all relations and value fields, all relations strategies. The highest MariaDB/PostgreSQL Search Coefficient indicates a 2.92× faster search time compared to MariaDB, and all but one coefficient are substantially greater than one, highlighting PostgreSQL’s significant efficiency in these scenarios.

Summary

PostgreSQL appears to be a superior database platform for RootDB compared to MariaDB when indexing strategies that include relationships are applied. This advantage remains consistent regardless of database size. PostgreSQL's superior performance is particularly evident in its ability to write objects with complex indexes at remarkable speed - approximately five times faster than MariaDB.

RootDB demonstrates a significant advantage in that, even without indexing, query execution times scale roughly linearly with database size, even when queries traverse multiple many-to-many relationships. This near-linear growth ensures that object database performance remains predictable and reliable without indexing. Although query times in large databases can become considerably longer, there is no risk of queries freezing due to excessive complexity.

Naturally, the complexity of the object model and the number of relationships within the database influence execution time. However, indexing relationships alone already yields significant performance improvements. By additionally indexing the fields referenced in queries, RootDB can meet even the most demanding requirements for fast query execution in large databases.

Links to other pages on this site.


Page content © 2025 

company name

Contact us:

mail address