How fast is SQLite compared to Microsoft Access MDB?

后端 未结 4 1910
青春惊慌失措
青春惊慌失措 2020-12-05 11:31

Currently I\'m thinking about replacing the usage of Microsoft Jet MDB databases on a single-user .NET C# Windows Forms application by a SQlite database.

My goal is

相关标签:
4条回答
  • 2020-12-05 11:50

    In case you decide to do your own benchmark testing, I offer this procedure to export your Jet tables to CSV files. Then you can import them into your SQLite database.

    Public Sub DumpTablesAsCsv()
        Dim db As DAO.Database
        Dim tdf As DAO.TableDef
        Dim strCsvFile As String
        Dim strFolder As String
        Dim strTable As String
    
        strFolder = CurrentProject.Path & Chr(92)
        Set db = CurrentDb
        For Each tdf In db.TableDefs
            strTable = tdf.Name
            If Not (strTable Like "MSys*" Or strTable Like "~*") Then
                strCsvFile = strFolder & strTable & ".csv"
                DoCmd.TransferText acExportDelim, , strTable, _
                    strCsvFile, HasFieldNames:=True
            End If
        Next tdf
        Set tdf = Nothing
        Set db = Nothing
    End Sub
    
    0 讨论(0)
  • 2020-12-05 11:50

    Actually in a fact, I'm not sure you're really asking the right question here.

    It sounds to me like you're looking for solution by changing your tools, and not changing your design and your approaches. In a fact, the access jet engine is substantially faster than something like a oracle, or mySQL, or SQL server for most operations. The reason is those other systems are huge mass of server based systems that have socket connections to the server. They have layers of transaction processing. There is probably 500 extra layers of software and systems between you and the actual data that resides on the hard drive.

    Contrast that to access which is essentially an in process program (not as running service). You do not connect to Access data files through some TCP/IP connection like you do with server based systems (in fact most of those server based systems force you to connect through and networking layer, even on your local machine and less you use a local memory connection, assuming that option is available).

    JET (Access database engine) is not a service, and is simply scrape the file off the hard drive and displays the results. That scraping of the data off the disk drive occurs at the same speed as oracle or SQL server and all of those other systems (we're assuming the same machine and hardware here ). Yet those other systems still have another 500 perhaps even 1000 extra layers of code and Software and Network connections and massive amounts of thngs like user security etc. All of these things substantially slow down that getting to the data on the disk drive by large amounts.

    Now course if you talking about a connection over some type of network, then those server based systems are better, because you want all the processing and all that majic to occur BEFORE any data starts to flow down the network pipe.

    However in your scenario, the server and the machine are one and the same. Therefore it makes complete sense to eliminate the mass of huge context of thousands of extra layers of software. As I pointed out, and these types of scenarios, jet can be 50% or even double the speed of server based systems like MySql or Oracle.

    Access can join, categorized, and total up inventoried for 150,000 records in well under a second, and that with a several table join.

    Now on the other hand, in any of these systems, usually the large overhead is, is to open up a connection to a particular table. In fact the time it takes to open a table is about the cost of 30,000 records to transfer. So, this means you want to ensure that your code and use of these tables does not unnecessary open up a new table (especially in some type of code loop. In other words, even in places of repeatedly executing an insert command a SQL, you're far better off to open up a record set, and then do inserts that way, as then you're not using SQL commands anymore, and for each row insert you're not executing a separate parsing of the text in that sql (this can give you about 100 times increase in performance when using access this way – in other words the often quoted advice here is that using SQL commands is faster than opening a record set, is completely incorrect).

    What this means is, if your are experiencing some kind of slow down here, I would look at your code and designs, and ensure that record sets and datasets are not being repeatedly opened and closed. You should not be experiencing any kind of noticeable delay in your data operations given the tiny size of the files you mention here.

    To be fair, sqlLITE is also (i believe) a in-process non server based edition of MySql, and most of the advantages pointed out above would also apply. But then again, your bottle neck would not be much differnt in each case, and thus we back to desing issues here.

    In other words, you're barking up the wrong tree, and a developer who looks for changes in their tools to fix performance is simply looking for a fix by blaming the tools when in most cases the problem lies in the designs adopted.

    0 讨论(0)
  • 2020-12-05 11:57

    Jet 4.0, DAO, MDAC and ADO have been included as part of the Windows OS since Windows 2000. Thus there is no need to distribute any Jet "drivers" with your application.

    0 讨论(0)
  • 2020-12-05 12:10

    More than 4 years later, I actually did a small (probably somewhat naive) performance comparison test between MDB and SQLite.

    I've also added more databases.

    Datebases I've tested

    • SQL Server Express 2014 on the same local PC and local SSD drive as the test application.
    • SQL Server Express 2014 on a server in a gigabit LAN.
    • SQL Server Compact (CE) on the same local PC and local SSD drive as the test application.
    • Microsoft Access MDB/Jet on the same local PC and local SSD drive as the test application.
    • Microsoft SQLite on the same local PC and local SSD drive as the test application.
    • Microsoft VistaDB 5 on the same local PC and local SSD drive as the test application.

    Since some databases do not support connection pooling, I've done two tests:

    1. One test with closing the connection as soon as possible through a using block.
    2. Another test with an always open connection to each database for the entire application lifetime

    Test results when closing the connections immediately

    • SQL Express running locally was the fastest.
    • SQL Express on our local network was at second position.
    • SQL Compact Edition (CE) was much faster than SQLite and Jet/MDB.
    • Jet/MDB was a little bit faster than SQLite and much slower than SQL CE.
    • SQLite was a little bit slower than Jet/MDB.
    • VistaDB 5 was the slowest database in my test.

    Test results when keeping the connections open

    The results are rather similar to the results when closing a connection immediately.

    Relatively to each other, the order from the fastest to the slowest did not change. Some databases with no actual connection pooling improved their absolute performance quite a bit.

    • SQL Express running locally was the fastest.
    • SQL Express on our local network was at second position.
    • SQL Compact Edition (CE) was much faster than SQLite and Jet/MDB.
    • Jet/MDB was a little bit faster than SQLite and much slower than SQL CE.
    • SQLite was a little bit slower than Jet/MDB.
    • VistaDB 5 was the slowest database in my test.

    Detailed output of my test application when closing the connections immediately

    1.: 1 x DELETE FROM Tabelle1 (Closing connections):
    - SQL Express local : 00:00:00.1723705
    - SQL Express remote: 00:00:00.2093229
    - SQL CE            : 00:00:00.3141897
    - MS Access         : 00:00:00.3854029
    - SQLite            : 00:00:00.4639365
    - VistaDB           : 00:00:00.9699047
    
    2.: 1 x INSERT INTO Tabelle1 (Name1, Wert1) VALUES ({LOOPCTR}, '{LOOPCTR}') (Closing connections):
    - SQL Express local : 00:00:00.0039836
    - SQL Express remote: 00:00:00.0062002
    - SQL CE            : 00:00:00.0432679
    - MS Access         : 00:00:00.0817834
    - SQLite            : 00:00:00.0933030
    - VistaDB           : 00:00:00.1200426
    
    3.: 10 x INSERT INTO Tabelle1 (Name1, Wert1) VALUES ({LOOPCTR}, '{LOOPCTR}') (Closing connections):
    - SQL Express local : 00:00:00.0031593
    - SQL Express remote: 00:00:00.0142514
    - SQL CE            : 00:00:00.3724224
    - MS Access         : 00:00:00.7474003
    - SQLite            : 00:00:00.8818905
    - VistaDB           : 00:00:00.9342783
    
    4.: 100 x INSERT INTO Tabelle1 (Name1, Wert1) VALUES ({LOOPCTR}, '{LOOPCTR}') (Closing connections):
    - SQL Express local : 00:00:00.0242817
    - SQL Express remote: 00:00:00.1124771
    - SQL CE            : 00:00:03.6239390
    - MS Access         : 00:00:07.3752378
    - SQLite            : 00:00:08.6489843
    - VistaDB           : 00:00:09.0933903
    
    5.: 1000 x INSERT INTO Tabelle1 (Name1, Wert1) VALUES ({LOOPCTR}, '{LOOPCTR}') (Closing connections):
    - SQL Express local : 00:00:00.2735537
    - SQL Express remote: 00:00:01.2657006
    - SQL CE            : 00:00:36.2335727
    - MS Access         : 00:01:13.8782439
    - SQLite            : 00:01:27.1783328
    - VistaDB           : 00:01:32.0760340
    
    6.: 1 x SELECT * FROM Tabelle1 (Closing connections):
    - SQL Express local : 00:00:00.0520670
    - SQL Express remote: 00:00:00.0570562
    - SQL CE            : 00:00:00.1026963
    - MS Access         : 00:00:00.1646635
    - SQLite            : 00:00:00.1785981
    - VistaDB           : 00:00:00.2311263
    
    7.: 10 x SELECT * FROM Tabelle1 (Closing connections):
    - SQL Express local : 00:00:00.0183055
    - SQL Express remote: 00:00:00.0501115
    - SQL CE            : 00:00:00.3235680
    - MS Access         : 00:00:00.7119203
    - SQLite            : 00:00:00.7533361
    - VistaDB           : 00:00:00.9804508
    
    8.: 100 x SELECT * FROM Tabelle1 (Closing connections):
    - SQL Express local : 00:00:00.1787837
    - SQL Express remote: 00:00:00.4321814
    - SQL CE            : 00:00:03.0401779
    - MS Access         : 00:00:06.8338598
    - SQLite            : 00:00:07.2000139
    - VistaDB           : 00:00:09.1889217
    
    9.: 1000 x SELECT * FROM Tabelle1 (Closing connections):
    - SQL Express local : 00:00:01.6112566
    - SQL Express remote: 00:00:03.9542611
    - SQL CE            : 00:00:29.1209991
    - MS Access         : 00:01:07.2309769
    - SQLite            : 00:01:10.3167922
    - VistaDB           : 00:01:31.4312770
    
    10.: 1 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID ORDER BY a.ID (Closing connections):
    - SQL Express local : 00:00:00.0029406
    - SQL Express remote: 00:00:00.0088138
    - SQL CE            : 00:00:00.0498847
    - MS Access         : 00:00:00.0893892
    - SQLite            : 00:00:00.0929506
    - VistaDB           : 00:00:00.2575795
    
    11.: 10 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID ORDER BY a.ID (Closing connections):
    - SQL Express local : 00:00:00.0174026
    - SQL Express remote: 00:00:00.0400797
    - SQL CE            : 00:00:00.3408818
    - MS Access         : 00:00:00.7314978
    - SQLite            : 00:00:00.7653330
    - VistaDB           : 00:00:01.9565675
    
    12.: 100 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID ORDER BY a.ID (Closing connections):
    - SQL Express local : 00:00:00.1565402
    - SQL Express remote: 00:00:00.3787208
    - SQL CE            : 00:00:03.3516629
    - MS Access         : 00:00:07.2521126
    - SQLite            : 00:00:07.5618047
    - VistaDB           : 00:00:19.5181391
    
    13.: 1000 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID ORDER BY a.ID (Closing connections):
    - SQL Express local : 00:00:01.5686470
    - SQL Express remote: 00:00:03.7414669
    - SQL CE            : 00:00:35.3944204
    - MS Access         : 00:01:14.6872377
    - SQLite            : 00:01:17.9964955
    - VistaDB           : 00:03:18.1902279
    
    14.: 1 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID WHERE a.ID < 100 OR a.ID > 300 ORDER BY a.ID (Closing connections):
    - SQL Express local : 00:00:00.0053295
    - SQL Express remote: 00:00:00.0089722
    - SQL CE            : 00:00:00.0395485
    - MS Access         : 00:00:00.0797776
    - SQLite            : 00:00:00.0833477
    - VistaDB           : 00:00:00.2554930
    
    15.: 10 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID WHERE a.ID < 100 OR a.ID > 300 ORDER BY a.ID (Closing connections):
    - SQL Express local : 00:00:00.0168467
    - SQL Express remote: 00:00:00.0552233
    - SQL CE            : 00:00:00.3929877
    - MS Access         : 00:00:00.7886399
    - SQLite            : 00:00:00.8209904
    - VistaDB           : 00:00:02.1248734
    
    16.: 100 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID WHERE a.ID < 100 OR a.ID > 300 ORDER BY a.ID (Closing connections):
    - SQL Express local : 00:00:00.1705345
    - SQL Express remote: 00:00:00.3969228
    - SQL CE            : 00:00:03.4886826
    - MS Access         : 00:00:07.4564258
    - SQLite            : 00:00:07.7828646
    - VistaDB           : 00:00:20.4092926
    
    17.: 1000 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID WHERE a.ID < 100 OR a.ID > 300 ORDER BY a.ID (Closing connections):
    - SQL Express local : 00:00:01.6237424
    - SQL Express remote: 00:00:03.9816212
    - SQL CE            : 00:00:35.1441759
    - MS Access         : 00:01:14.7739758
    - SQLite            : 00:01:17.9477049
    - VistaDB           : 00:03:24.0049633
    

    Detailed output of my test application when keeping the connections open

    1.: 1 x DELETE FROM Tabelle1 (keeping connection open):
    - SQL Express local : 00:00:00.0426930
    - SQL Express remote: 00:00:00.0546357
    - SQL CE            : 00:00:00.0786765
    - MS Access         : 00:00:00.0909099
    - SQLite            : 00:00:00.1101572
    - VistaDB           : 00:00:00.4637726
    
    2.: 1 x INSERT INTO Tabelle1 (Name1, Wert1) VALUES ({LOOPCTR}, '{LOOPCTR}') (keeping connection open):
    - SQL Express local : 00:00:00.0030936
    - SQL Express remote: 00:00:00.0051136
    - SQL CE            : 00:00:00.0054226
    - MS Access         : 00:00:00.0074847
    - SQLite            : 00:00:00.0154474
    - VistaDB           : 00:00:00.0373701
    
    3.: 10 x INSERT INTO Tabelle1 (Name1, Wert1) VALUES ({LOOPCTR}, '{LOOPCTR}') (keeping connection open):
    - SQL Express local : 00:00:00.0023271
    - SQL Express remote: 00:00:00.0109913
    - SQL CE            : 00:00:00.0119872
    - MS Access         : 00:00:00.0152531
    - SQLite            : 00:00:00.1131698
    - VistaDB           : 00:00:00.1261859
    
    4.: 100 x INSERT INTO Tabelle1 (Name1, Wert1) VALUES ({LOOPCTR}, '{LOOPCTR}') (keeping connection open):
    - SQL Express local : 00:00:00.0201695
    - SQL Express remote: 00:00:00.0888872
    - SQL CE            : 00:00:00.0966017
    - MS Access         : 00:00:00.1256167
    - SQLite            : 00:00:01.3632978
    - VistaDB           : 00:00:01.9422151
    
    5.: 1000 x INSERT INTO Tabelle1 (Name1, Wert1) VALUES ({LOOPCTR}, '{LOOPCTR}') (keeping connection open):
    - SQL Express local : 00:00:00.1693362
    - SQL Express remote: 00:00:00.9181297
    - SQL CE            : 00:00:01.0366334
    - MS Access         : 00:00:01.2794199
    - SQLite            : 00:00:13.9398816
    - VistaDB           : 00:00:19.8319476
    
    6.: 1 x SELECT * FROM Tabelle1 (keeping connection open):
    - SQL Express local : 00:00:00.0481500
    - SQL Express remote: 00:00:00.0507066
    - SQL CE            : 00:00:00.0738698
    - MS Access         : 00:00:00.0911707
    - SQLite            : 00:00:00.1012425
    - VistaDB           : 00:00:00.1515495
    
    7.: 10 x SELECT * FROM Tabelle1 (keeping connection open):
    - SQL Express local : 00:00:00.0157947
    - SQL Express remote: 00:00:00.0692206
    - SQL CE            : 00:00:00.0898558
    - MS Access         : 00:00:00.1196514
    - SQLite            : 00:00:00.1400944
    - VistaDB           : 00:00:00.3227485
    
    8.: 100 x SELECT * FROM Tabelle1 (keeping connection open):
    - SQL Express local : 00:00:00.1517498
    - SQL Express remote: 00:00:00.3399897
    - SQL CE            : 00:00:00.5497382
    - MS Access         : 00:00:00.8619646
    - SQLite            : 00:00:01.0463369
    - VistaDB           : 00:00:02.8607334
    
    9.: 1000 x SELECT * FROM Tabelle1 (keeping connection open):
    - SQL Express local : 00:00:01.5042900
    - SQL Express remote: 00:00:03.8431985
    - SQL CE            : 00:00:05.9075477
    - MS Access         : 00:00:09.2642402
    - SQLite            : 00:00:11.4427914
    - VistaDB           : 00:00:30.8470936
    
    10.: 1 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID ORDER BY a.ID (keeping connection open):
    - SQL Express local : 00:00:00.0033803
    - SQL Express remote: 00:00:00.0062499
    - SQL CE            : 00:00:00.0141105
    - MS Access         : 00:00:00.0188573
    - SQLite            : 00:00:00.0208236
    - VistaDB           : 00:00:00.1796513
    
    11.: 10 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID ORDER BY a.ID (keeping connection open):
    - SQL Express local : 00:00:00.0168644
    - SQL Express remote: 00:00:00.0377185
    - SQL CE            : 00:00:00.1121558
    - MS Access         : 00:00:00.1599104
    - SQLite            : 00:00:00.1799435
    - VistaDB           : 00:00:01.4042534
    
    12.: 100 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID ORDER BY a.ID (keeping connection open):
    - SQL Express local : 00:00:00.1547275
    - SQL Express remote: 00:00:00.3692526
    - SQL CE            : 00:00:01.1215470
    - MS Access         : 00:00:01.5577172
    - SQLite            : 00:00:01.7519790
    - VistaDB           : 00:00:14.5687575
    
    13.: 1000 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID ORDER BY a.ID (keeping connection open):
    - SQL Express local : 00:00:01.4992800
    - SQL Express remote: 00:00:03.7601806
    - SQL CE            : 00:00:11.1738426
    - MS Access         : 00:00:15.8112902
    - SQLite            : 00:00:17.8045042
    - VistaDB           : 00:02:21.4492368
    
    14.: 1 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID WHERE a.ID < 100 OR a.ID > 300 ORDER BY a.ID (keeping connection open):
    - SQL Express local : 00:00:00.0048145
    - SQL Express remote: 00:00:00.0076790
    - SQL CE            : 00:00:00.0152074
    - MS Access         : 00:00:00.0204568
    - SQLite            : 00:00:00.0229056
    - VistaDB           : 00:00:00.2091614
    
    15.: 10 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID WHERE a.ID < 100 OR a.ID > 300 ORDER BY a.ID (keeping connection open):
    - SQL Express local : 00:00:00.0156564
    - SQL Express remote: 00:00:00.0377571
    - SQL CE            : 00:00:00.1138433
    - MS Access         : 00:00:00.1598932
    - SQLite            : 00:00:00.1793267
    - VistaDB           : 00:00:01.4667061
    
    16.: 100 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID WHERE a.ID < 100 OR a.ID > 300 ORDER BY a.ID (keeping connection open):
    - SQL Express local : 00:00:00.1512625
    - SQL Express remote: 00:00:00.4658652
    - SQL CE            : 00:00:01.2441809
    - MS Access         : 00:00:01.7224126
    - SQLite            : 00:00:01.9297231
    - VistaDB           : 00:00:14.9351318
    
    17.: 1000 x SELECT a.* FROM Tabelle1 a LEFT JOIN Tabelle1 b ON a.ID=b.ID WHERE a.ID < 100 OR a.ID > 300 ORDER BY a.ID (keeping connection open):
    - SQL Express local : 00:00:01.5223833
    - SQL Express remote: 00:00:03.9885174
    - SQL CE            : 00:00:11.8356048
    - MS Access         : 00:00:16.5977939
    - SQLite            : 00:00:18.6504260
    - VistaDB           : 00:02:26.0513056
    

    Update 1, April 2019

    I did some tests comparing Microsoft Access MDB to LiteDB, an embedded database for .NET.

    So again, some rather naive comparison, but I still wanted to get a feeling.

    This code does 1000 reads and writes and resulted in these values:

    Access             of 1000 WRITE iterations took 00:00:39.6488351.
    LiteDB             of 1000 WRITE iterations took 00:00:01.6596646.
    LiteDB (in-memory) of 1000 WRITE iterations took 00:00:00.1617220.
    Access             of 1000 READ  iterations took 00:00:48.8517302.
    LiteDB             of 1000 READ  iterations took 00:00:00.0082401.
    LiteDB (in-memory) of 1000 READ  iterations took 00:00:00.0097933.
    

    So in my scenario, LiteDB was much faster than Access.

    Update 2, April 2019

    I've also ran my original code against VistaDB 6 Beta 1 in comparison to VistaDB 5.

    I've got very similar speed results. The Beta of VistaDB 6 was slightly slower compared to VistaDB 5, most likely because it was a debug build.

    As a conclusion, I see no significant performance improvements in my scenario between VistaDB 5 and VistaDB 6 Beta 1. I will have to try again with the final version of VistaDB 6.

    0 讨论(0)
提交回复
热议问题