当 With 在同一个工作簿中指定不同的 sheet 时,我可以将 With 嵌套在 With 中吗? VBA
Can I nest a With inside a With when both are designating a different sheet in the same workbook? VBA
基本上,我有一个工作簿可以从一个 sheet 导入值,然后允许您 select 一些,添加额外数据,然后传输 selected 项目,您的附加值,以及包含在 sheet 中的附加信息,其值最初从另一个工作簿中提取。我需要能够将初始工作 sheet 中的值与另一个 sheet (在同一工作簿中)中的相应信息相匹配,以便我可以从两者中获取数据并将它们循环。我我担心当他们都在同一个工作簿中指定 worksheets 时,我不允许嵌套我的 Withs,因为我不断收到错误 91(未设置对象变量或 With 块变量)。这是真的吗?
Dim RMs As Worksheet
Dim FLF As Worksheet
Set RMs = Workbooks("FLF Coding Trials.xlsm").Worksheets("Program")
Set FLF = Workbooks("FLF Template.xlsx").Worksheets("FLF Sheet")
Dim lng As Long
Dim cnt As Long
Dim check As Long
Dim length As Long
Dim namecheck As Range
Dim vencheck As Range
With ThisWorkbook.Sheets("Program")
lng = Me.Cells(.Rows.Count, "N").End(xlUp).Row
For cnt = 1 To lng - 1 Step 1
'The two below lines work reliably
FLF.Range("B" & cnt + 23).Value = Me.Range("N" & cnt + 1).Value
FLF.Range("N" & cnt + 23).Value = Me.Range("P" & cnt + 1).Value
'Adding the following variable designations and the With gives me the error. If I change the below two variables to strings I get a compile error
namecheck.Value = Range("N" & cnt + 1).Value
vencheck.Value = Range("P" & cnt + 1).Value
With ThisWorkbook.Sheets("Names and Vendors")
length = Me.Cells(.Rows.Count, "B").End(xlUp).Row
check = ThisWorkbook.Sheets("Names and Vendors").Evaluate("MATCH(1,(B1:B" & length & "=""" & namecheck.Value & """)*(C1:C" & length & "=""" & vencheck.Value & """),0)")
'more commands will go here
End With
Next cnt
End With
如果您在 With
块中并且您的代码进入第二个 With
(没有先点击 End With
),那么第二个 [=11] 定义的范围=] 成为活动范围,直到您点击 End With
(或另一个 With
)
With A
'scope here is A
With B
'scope here is B
End with
'scope here is A
End With
您的代码中存在很多问题,none 其中与 With
块有关。
根据评论我假设我在 With 范围内指定了对象 Me.
的使用应该只替换为 .
请参阅下面标有 '~~~
的代码中的注释
Dim RMs As Worksheet
Dim FLF As Worksheet
Set RMs = Workbooks("FLF Coding Trials.xlsm").Worksheets("Program")
Set FLF = Workbooks("FLF Template.xlsx").Worksheets("FLF Sheet")
Dim lng As Long
Dim cnt As Long
Dim check As Long
Dim length As Long
Dim namecheck As Range
Dim vencheck As Range
With ThisWorkbook.Sheets("Program")
'~~~ .Rows refers to ThisWorkbook.Sheets("Program").Rows
lng = .Cells(.Rows.Count, "N").End(xlUp).Row '~~~ drop the Me
For cnt = 1 To lng - 1 Step 1
'The two below lines work reliably
'~~~ Me implies this code is in a Worksheet code behind module, and refers to that sheet. Is this what you intend?
FLF.Range("B" & cnt + 23).Value = .Range("N" & cnt + 1).Value '~~~ drop the Me
FLF.Range("N" & cnt + 23).Value = .Range("P" & cnt + 1).Value '~~~ drop the Me
'Adding the following variable designations and the With gives me the error. If I change the below two variables to strings I get a compile error
'~~~ there is no reference to the With block in these 2 lines of code
'~~~ And you havn't Set namecheck or vencheck. Add
Set namecheck = FLF.Range("B" & cnt + 23) '~~~ or whatever range you actually meant
Set vencheck = FLF.Range("N" & cnt + 23)
namecheck.Value = Range("N" & cnt + 1).Value '~~~ the unqualified Range(...) refers to the ActiveSheet. Maybe should be .Range?
vencheck.Value = Range("P" & cnt + 1).Value
With ThisWorkbook.Sheets("Names and Vendors")
'~~~ .Rows now refers to ThisWorkbook.Sheets("Names and Vendors").Rows
length = .Cells(.Rows.Count, "B").End(xlUp).Row '~~~ drop the Me
'~~~ you can leave out ThisWorkbook.Sheets("Names and Vendors") as this is in the With block for that sheet
check = .Evaluate("MATCH(1,(B1:B" & length & "=""" & namecheck.Value & """)*(C1:C" & length & "=""" & vencheck.Value & """),0)")
'more commands will go here
End With
'~~~ .Anything now refers to ThisWorkbook.Sheets("Program").Anything again
Next cnt
End With
这个问题更正确的答案是 With 构造应该限制为一个对象的一系列简单的 getter 和 setter。
Word Find 对象就是一个很好的例子,下面的代码很常见
With ActiveDocument.Content.Find
.Text = "FindText"
.Replacement.Text = "ReplaceText"
.Forward = True
.Wrap = wdFindStop
Do While .Execute() = True
.TypeParagraph
.MoveLeft Unit:=wdWord, Count:=2, Extend:=wdExtend
.Find.Replacement.Font.Italic = True
.Font.Bold = True
.Collapse Direction:=wdCollapseEnd
Loop
End With
解决您提出的问题的更正确方法是使用适当的局部范围对象来提供您需要的快捷方式,例如
而不是
With ThisWorkbook.Sheets("Program")
'Do xyz
End With
使用
Dim mySourceWb as excel.Worksheet
Set mySOurceWb = ThisWorkbook.Sheets("Program")
Dim myDestWb as Excel.Worksheet
Set myDestWB = ADifferentWb.Sheets("YetAnotherSpreadsheet")
'Do xyz
set mySourceWb=nothing
set myDestWb=nothing
基本上,我有一个工作簿可以从一个 sheet 导入值,然后允许您 select 一些,添加额外数据,然后传输 selected 项目,您的附加值,以及包含在 sheet 中的附加信息,其值最初从另一个工作簿中提取。我需要能够将初始工作 sheet 中的值与另一个 sheet (在同一工作簿中)中的相应信息相匹配,以便我可以从两者中获取数据并将它们循环。我我担心当他们都在同一个工作簿中指定 worksheets 时,我不允许嵌套我的 Withs,因为我不断收到错误 91(未设置对象变量或 With 块变量)。这是真的吗?
Dim RMs As Worksheet
Dim FLF As Worksheet
Set RMs = Workbooks("FLF Coding Trials.xlsm").Worksheets("Program")
Set FLF = Workbooks("FLF Template.xlsx").Worksheets("FLF Sheet")
Dim lng As Long
Dim cnt As Long
Dim check As Long
Dim length As Long
Dim namecheck As Range
Dim vencheck As Range
With ThisWorkbook.Sheets("Program")
lng = Me.Cells(.Rows.Count, "N").End(xlUp).Row
For cnt = 1 To lng - 1 Step 1
'The two below lines work reliably
FLF.Range("B" & cnt + 23).Value = Me.Range("N" & cnt + 1).Value
FLF.Range("N" & cnt + 23).Value = Me.Range("P" & cnt + 1).Value
'Adding the following variable designations and the With gives me the error. If I change the below two variables to strings I get a compile error
namecheck.Value = Range("N" & cnt + 1).Value
vencheck.Value = Range("P" & cnt + 1).Value
With ThisWorkbook.Sheets("Names and Vendors")
length = Me.Cells(.Rows.Count, "B").End(xlUp).Row
check = ThisWorkbook.Sheets("Names and Vendors").Evaluate("MATCH(1,(B1:B" & length & "=""" & namecheck.Value & """)*(C1:C" & length & "=""" & vencheck.Value & """),0)")
'more commands will go here
End With
Next cnt
End With
如果您在 With
块中并且您的代码进入第二个 With
(没有先点击 End With
),那么第二个 [=11] 定义的范围=] 成为活动范围,直到您点击 End With
(或另一个 With
)
With A
'scope here is A
With B
'scope here is B
End with
'scope here is A
End With
您的代码中存在很多问题,none 其中与 With
块有关。
根据评论我假设我在 With 范围内指定了对象 Me.
的使用应该只替换为 .
请参阅下面标有 '~~~
Dim RMs As Worksheet
Dim FLF As Worksheet
Set RMs = Workbooks("FLF Coding Trials.xlsm").Worksheets("Program")
Set FLF = Workbooks("FLF Template.xlsx").Worksheets("FLF Sheet")
Dim lng As Long
Dim cnt As Long
Dim check As Long
Dim length As Long
Dim namecheck As Range
Dim vencheck As Range
With ThisWorkbook.Sheets("Program")
'~~~ .Rows refers to ThisWorkbook.Sheets("Program").Rows
lng = .Cells(.Rows.Count, "N").End(xlUp).Row '~~~ drop the Me
For cnt = 1 To lng - 1 Step 1
'The two below lines work reliably
'~~~ Me implies this code is in a Worksheet code behind module, and refers to that sheet. Is this what you intend?
FLF.Range("B" & cnt + 23).Value = .Range("N" & cnt + 1).Value '~~~ drop the Me
FLF.Range("N" & cnt + 23).Value = .Range("P" & cnt + 1).Value '~~~ drop the Me
'Adding the following variable designations and the With gives me the error. If I change the below two variables to strings I get a compile error
'~~~ there is no reference to the With block in these 2 lines of code
'~~~ And you havn't Set namecheck or vencheck. Add
Set namecheck = FLF.Range("B" & cnt + 23) '~~~ or whatever range you actually meant
Set vencheck = FLF.Range("N" & cnt + 23)
namecheck.Value = Range("N" & cnt + 1).Value '~~~ the unqualified Range(...) refers to the ActiveSheet. Maybe should be .Range?
vencheck.Value = Range("P" & cnt + 1).Value
With ThisWorkbook.Sheets("Names and Vendors")
'~~~ .Rows now refers to ThisWorkbook.Sheets("Names and Vendors").Rows
length = .Cells(.Rows.Count, "B").End(xlUp).Row '~~~ drop the Me
'~~~ you can leave out ThisWorkbook.Sheets("Names and Vendors") as this is in the With block for that sheet
check = .Evaluate("MATCH(1,(B1:B" & length & "=""" & namecheck.Value & """)*(C1:C" & length & "=""" & vencheck.Value & """),0)")
'more commands will go here
End With
'~~~ .Anything now refers to ThisWorkbook.Sheets("Program").Anything again
Next cnt
End With
这个问题更正确的答案是 With 构造应该限制为一个对象的一系列简单的 getter 和 setter。
Word Find 对象就是一个很好的例子,下面的代码很常见
With ActiveDocument.Content.Find
.Text = "FindText"
.Replacement.Text = "ReplaceText"
.Forward = True
.Wrap = wdFindStop
Do While .Execute() = True
.TypeParagraph
.MoveLeft Unit:=wdWord, Count:=2, Extend:=wdExtend
.Find.Replacement.Font.Italic = True
.Font.Bold = True
.Collapse Direction:=wdCollapseEnd
Loop
End With
解决您提出的问题的更正确方法是使用适当的局部范围对象来提供您需要的快捷方式,例如
而不是
With ThisWorkbook.Sheets("Program")
'Do xyz
End With
使用
Dim mySourceWb as excel.Worksheet
Set mySOurceWb = ThisWorkbook.Sheets("Program")
Dim myDestWb as Excel.Worksheet
Set myDestWB = ADifferentWb.Sheets("YetAnotherSpreadsheet")
'Do xyz
set mySourceWb=nothing
set myDestWb=nothing