Solved Option Explicit (lack of use thereof) ?

Gasman

Enthusiastic Amateur
Local time
Today, 02:56
Joined
Sep 21, 2011
Messages
15,994
Hi all,
Is there any reason anyone can think of as to why you would NOT use Option Explicit?

The lack of doing so came back to bite this user big time. I did not spot although inadvertently I was on the right track. :)

Solution is on page 2. Unfortunately that site does not have post numbers. :(

I have mentioned that thread in another post whereby that user was using a method of setting recordset fields unknown to me.
 
Is there any reason anyone can think of as to why you would NOT use Option Explicit?
Ignorance, arrogance, a death wish all come to mind.

This is such poor practice, MS should have long ago changed the default to set this option on. It would have upset a bunch of sloppy coders but they could have easily fixed their errors and moved on.
 
I cannot think of a reason that this is even allowed because it can cause problems that are nearly impossible to see. MS somehow thinks this is actually helping people and making it easier to code. However, I disagree that is the cause of the OP's problem.

VBA does support polymorphism in a way.
I can ensure you the following code surprisingly works and sets all my agent names to New Name.

Code:
Option Compare Database

Public Sub NoDeclaration()
  Dim rs As DAO.Recordset
  Set rs = CurrentDb.OpenRecordset("tblAgents")
  Set fld12 = rs!AgentName
  Debug.Print TypeName(fld12)
  Do While Not rs.EOF
    rs.Edit
      fld12.Value = "New Name"
    rs.Update
    rs.MoveNext
  Loop
  fld12 = "Hello World"
  Debug.Print fld12 & "  " & TypeName(fld12)
  fld12 = Date
  Debug.Print fld12 & "  " & TypeName(fld12)
  Set fld12 = rs
  Debug.Print fld12.Name & "  " & TypeName(fld12)
End Sub

fld12 is a undeclared variable, so it is of type variant. When you set it to an dao.field it subclasses to a dao field. In fact whatever you set it to it subclasses to that. Here is the debug.print. You can see it is a Field, String, Date, and Recordset depending on the assignment.

Field3
Hello World String
6/23/2023 Date
tblAgents Recordset2
 
Probably it stems from the original Beginner's All-purpose Symbolic Instruction Code (BASIC), where as early as the late 1960s, declarations were NEVER necessary for any data type up to and including arrays with subscripts 0 to 10. The version I was using then was DEC BASIC for PDP-10. (Which was a butt-kicking 36-bit machine.)
 
I would suggest that anyone not using Option Explicit is:
a) new to Access/vba, or
b) lazy.

I do not understand why M$oft's NOT setting this ON by default is in anyone's best interest.
 
I cannot think of a reason that this is even allowed because it can cause problems that are nearly impossible to see. MS somehow thinks this is actually helping people and making it easier to code. However, I disagree that is the cause of the OP's problem.

VBA does support polymorphism in a way.
I can ensure you the following code surprisingly works and sets all my agent names to New Name.

Code:
Option Compare Database

Public Sub NoDeclaration()
  Dim rs As DAO.Recordset
  Set rs = CurrentDb.OpenRecordset("tblAgents")
  Set fld12 = rs!AgentName
  Debug.Print TypeName(fld12)
  Do While Not rs.EOF
    rs.Edit
      fld12.Value = "New Name"
    rs.Update
    rs.MoveNext
  Loop
  fld12 = "Hello World"
  Debug.Print fld12 & "  " & TypeName(fld12)
  fld12 = Date
  Debug.Print fld12 & "  " & TypeName(fld12)
  Set fld12 = rs
  Debug.Print fld12.Name & "  " & TypeName(fld12)
End Sub

fld12 is a undeclared variable, so it is of type variant. When you set it to an dao.field it subclasses to a dao field. In fact whatever you set it to it subclasses to that. Here is the debug.print. You can see it is a Field, String, Date, and Recordset depending on the assignment.
No idea as to why not for the O/P then. :( Once he corrected that issue and Dimmed the fields, he said it then worked.

I just tried with this code and record is not updated?
Option Explicit commented out as well.
Code:
Sub UpdateRst()
Dim db As DAO.Database
'Dim fld As DAO.Field
Dim rst As DAO.Recordset

Set db = CurrentDb
Set rst = db.OpenRecordset("Select Description from TestTransactionsDelete")

Set fld = rst!Description

Do Until rst.EOF
    rst.MoveLast
    rst.Edit
    fld = "Test Description 4"
    rst.Update
    rst.MoveNext
Loop
Set fld = Nothing
Set rst = Nothing
Set db = Nothing
End Sub
 
Last edited:
MoveNext after MoveLast?
Functioning is a broad concept. Maybe you dare to take the step towards working GOOD (code makes sense, correct, useful, ...)
 
Try:
Code:
...
Set fld = rst!Description

Do Until rst.EOF
    rst.MoveLast
    rst.Edit
    Debug.Print TypeOf fld Is DAO.Field  ' <-- True
    fld = "Test Description 4"
    Debug.Print TypeOf fld Is DAO.Field  ' <-- False !!!
    rst.Update
    rst.MoveNext
Loop
...
vs.
Code:
...
Set fld = rst!Description

Do Until rst.EOF
    rst.MoveLast
    rst.Edit
    Debug.Print TypeOf fld Is DAO.Field  ' <-- True
    fld.Value = "Test Description 4"
    Debug.Print TypeOf fld Is DAO.Field  ' <-- True
    rst.Update
    rst.MoveNext
Loop
...
 
MoveNext after MoveLast?
Functioning is a broad concept. Maybe you dare to take the step towards working GOOD (code makes sense, correct, useful, ...)
How else am I going to get to EOF?
 
Try:
Code:
...
Set fld = rst!Description

Do Until rst.EOF
    rst.MoveLast
    rst.Edit
    Debug.Print TypeOf fld Is DAO.Field  ' <-- True
    fld = "Test Description 4"
    Debug.Print TypeOf fld Is DAO.Field ' <-- False !!!
    rst.Update
    rst.MoveNext
Loop
...
vs.
Code:
...
Set fld = rst!Description

Do Until rst.EOF
    rst.MoveLast
    rst.Edit
    Debug.Print TypeOf fld Is DAO.Field  ' <-- True
    fld.Value = "Test Description 4"
    Debug.Print TypeOf fld Is DAO.Field ' <-- True
    rst.Update
    rst.MoveNext
Loop
...
I did spot the .Value in MajP's code, but have got so used to not using it except with TempVars.
 
MoveLast + MoveNext results in EOF on the first pass. The loop is meaningless, only EOF can come after MoveLast.
 
MoveLast + MoveNext results in EOF on the first pass. The loop is meaningless, only EOF can come after MoveLast.
Which is exactly what I wanted, just amend the last record.?

What is your problem? :(
 
How else am I going to get to EOF?
What is the loop for anyway?
According to the code, you want to edit only the last record in the recordset.

without Do Until rst.EOF:
Code:
...
    Set fld = rst!Description

    rst.MoveLast
    rst.Edit
    fld.Value = "Test Description 4"
    rst.Update
...
 
Because I started with a loop as I amended all the records, like the O/P did, then I just wanted to update the last record with different text.

I could have used FindFirst on the ID field for a single record, but as I already had a loop, I just amended that.
 
Then everything is clear. It's just a leftover from testing. ;)
Not related to the actual problem anyway.
 
It is nothing but test code.
I keep stuff like that in a module named ModCrap. :)
 
Which is exactly what I wanted, just amend the last record.?



Just to be clear, .MoveNext following a .MoveLast should put you at the last record for DAO or AFTER the last record for ADO. DAO won't move beyond EOF but might give an error depending on how you moved it. In essence, the .MoveNext issued after a .MoveLast is superfluous for DAO recordsets.
 
1687548810891.png

or
1687548908801.png
 
@Gasman, I understand the point now. I see that I missed something, because my example was not what the OP did. This really echos your point of how you can get into big trouble without declaring variables. This is pretty subtle, and few people would probably see what is really going on.

Without declaring a fld12 variable, if the OP did the following this would work
Fld12.value = Abs(Fld5) * CountTicks * 7.8125 - Abs(Fld5 * 1.5) - Abs(Fld5 * 0.67) - Abs(dblPrevCom) - Abs(dblPrevFee)
This is what they did, and it does not work. Why?
Fld12 = Abs(Fld5) * CountTicks * 7.8125 - Abs(Fld5 * 1.5) - Abs(Fld5 * 0.67) - Abs(dblPrevCom) - Abs(dblPrevFee)

To demo why this happens, this code would work and would update the recordset if I assign x to fld12.value. It will not work if I omit the default .value property.

Code:
Public Sub NoDeclaration()
  Dim rs As DAO.Recordset
  
  Set rs = CurrentDb.OpenRecordset("tblAgents")
  Set fld12 = rs!AgentName
  Debug.Print TypeName(fld12)
  Do While Not rs.EOF
    rs.Edit
        'fld12.value = "x"
        fld12 = "x"
      Debug.Print TypeName(fld12)
    rs.Update
    rs.MoveNext
  Loop
End Sub

The reason is that in this line
Set fld12 = rs!AgentName
resolves to a Field
In this line if I do this
fld12.value = "x"
It remains a Field variable
Without the .value property VBA coerces the fld12 back to a string.
Confusing enough? The below print out showing the casting back to a string from a Field3.
Field3
String
String
String
String
String
 
Yes, Josef P. pointed that out as well.
As I rarely use the .Value as I mentioned, that would catch me out, but then again, I never knew of that method. :)
You see it mentioned many times by people, that there is no need as default property.

Plus I would always use Option Explicit and Dim my variables.
 

Users who are viewing this thread

Back
Top Bottom