Thursday, April 12, 2007

Recipe: Reading a text file into an ADO Recordset.

Recently frustrated with trying to handle multiple imports into Microsoft Access® I decided to research if there was an easier way of handling external text files similar to .NET's simplicity of reading a text file into a stream or data set only with VBA instead. What I came across was the following article:

Much ADO about Text Files: http://msdn2.microsoft.com/en-us/library/ms974559.aspx

The gist of this article was that with a few lines of code, I would no longer need to import my data to work with it, I could use an in memory (ADO Recordset) copy of it instead. To accomplish this is rather simple, so simple in fact I can not believe I have not been using it all along.

So in a nutshell, here is a really simple example (I highly recommend visiting the aforementioned link for a much greater in-depth look, especially using/creating schema.ini files or using similar functionality with late binding techniques):

I assume that you have...
  • Microsoft Access® with a new Module open.
  • A sample delimited file: C:\TestData.csv with a header row (optional schema.ini). For my example I have a header row FirstName,LastName and a few first and last name records delimited by commas.
  • Familiar with some basic VBA syntax
Public Sub TextToADO()

'Create a variable that holds the location of your text file.
Dim myTextFile As String
myTextFile = "C:\TestData.csv"

'Create the connection string that will connect to the file. For more info
'on the Extended Properties of the string refer to the above link, however
'as you have probably guessed, HDR stands for Header and you should
'adjust accordingly.
Dim myCnnString As String
myCnnString = "Provider=Microsoft.Jet.OLEDB.4.0;" & _
"Data Source=" & myTextFile & ";" & _
"Extended Properties=text;HDR=Yes;FMT=Delimited"

'Create an ADO recordset variable and a sql string for the select statement.
Dim rs As ADODB.Recordset, sql As String
sql = "SELECT * FROM TestData.csv"
Set rs = New ADODB.Recordset
rs.Open sql, myCnnString, adOpenStatic, adLockOptimistic, adCmdText

While Not rs.EOF

Debug.Print rs("LastName") & ", " & rs("FirstName")

rs.MoveNext
Wend

rs.Close
Set rs = Nothing

End Sub

As anyone who is familiar with Access knows, it can balloon in size rather rapidly and if you don't regularly compact and occasionally decompile it, it can adversely affect performance. What this solution provides is a way of not having to import data that I want to work with (reducing Access' tendency to grow) but rather reading it into memory so it can be manipulated as if it were an actual table in my database. This was a fantastic find and it really got me thinking about ways to further use what I had learned, like linking to external text files. I have always linked to an external Excel spreadsheet but I had never given linking to a text file any thought before.

What this taught me was that I'm (we) are always learning and what one person takes for granted as assumed knowledge may be a hidden gem for someone else, hence this blog.

I hope you found this useful and as beneficial as I did.

No comments: