在Excel中另存为时保存时间缓慢增加

dpiehjr4  于 2023-01-27  发布在  其他
关注(0)|答案(1)|浏览(136)

我创建了一个启用宏的Excel文件,该文件执行以下操作(高级):
1.用户通过文件对话框选择模板文件(本身启用宏)
1.用户也通过文件对话框选择数据文件(未启用宏)
1.宏将遍历数据文件,并逐个打开它们,设置数据格式,将数据迁移到中间工作簿中的新工作表,然后关闭数据文件而不保存它
1.循环通过所有文件后,中间工作簿也将保存,但保持打开状态
1.循环通过所有数据文件后,将循环通过中间工作簿的每个工作表,将当前工作表中的数据传输到模板文件,并将模板文件另存为具有唯一标签的新文件。此现在包含数据的文件中的一行数据将复制到摘要工作表中
选择的数据文件数量达数千个(到目前为止,我们尝试的最大运行是4000个文件)。随着宏的运行,保存这些文件所需的时间缓慢但稳定地变长。开始时大约为5秒,但到最后,一些文件需要大约5分钟才能保存。
我添加了一个迭代特性,一旦所有数据文件都被循环通过,它就完全关闭模板文件,并使用不同的设置打开它的一个新示例,然后重新开始这个过程。
这将使保存时间恢复正常。
在此步骤中,还将保存并关闭摘要文件,并为新的小版本打开一个新文件。
我曾考虑过每隔一百个左右的数据文件就关闭并重新打开模板文件,但我更希望得到一个适当的解决方案。
如果我每次都打开和关闭模板文件,我就避免了时间问题,但是宏会变得非常不稳定,有时会在随机点崩溃。
这是在一台与互联网或任何类型的网络隔离的计算机上,并保存到固态驱动器(我们试图控制相当多的变量)。

  1. Option Explicit
  2. Public Sub Example()
  3. Dim Trial As Integer, Trials As Integer, DataSet As Integer
  4. Dim TrialChecker As Boolean
  5. Dim StartTime As Double, WaitTime As Double
  6. Dim StartDate As Date
  7. Dim FileSaveName As String
  8. Dim CopiedDataRange As Range
  9. Dim SummaryRunTimes As Worksheet, Calcs As Worksheet, CutoffsShifts As Worksheet
  10. Dim SheetObjects() As Worksheet
  11. Dim IntermediaryWorkbook As Workbook, Summary As Workbook, Template As Workbook
  12. Application.ScreenUpdating = False
  13. Application.Calculation = xlCalculationManual
  14. 'The 1 and Trials are actually set by Lbound and Ubound funcitons, but the premise is the same
  15. For Trial = 1 To Trials
  16. Workbooks.Add
  17. Set Summary = ActiveWorkbook
  18. 'I use this one sheet to keep track of how long different parts of the code take to run
  19. Set SummaryRunTimes = Summary.Worksheets(1)
  20. SummaryRunTimes.Name = "Run Times"
  21. SummaryRunTimes.Cells(1, 1).Value = "ID"
  22. SummaryRunTimes.Cells(1, 2).Value = "Data Copy Time (s)"
  23. SummaryRunTimes.Cells(1, 3).Value = "Formula Copy and Calc Time (s)"
  24. SummaryRunTimes.Cells(1, 4).Value = "Summary Copy Time (s)"
  25. SummaryRunTimes.Cells(1, 5).Value = "Save and Cleanup Time (s)"
  26. 'sheetnames is defined elsewhere in the code (it's a global variable right now. I intend to change that later).
  27. 'It's simply an array of strings with six elements.
  28. For Counter = LBound(sheetnames) To UBound(sheetnames)
  29. Summary.Worksheets.Add
  30. Summary.ActiveSheet.Name = sheetnames(Counter)
  31. Next Counter
  32. 'Again, TemplateLocation is defined elsewhere. It's just a string grabbed from a filedialog
  33. Workbooks.Open (TemplateLocation)
  34. Set Template = ActiveWorkbook
  35. Set Calcs = Template.Sheets("Calcs")
  36. Set CutoffsShifts = Template.Sheets("Log Cutoffs & Shifts")
  37. 'SheetObjects is simply used as a convenient reference for various sheets in the template file. I found
  38. 'it cleaned up the code a bit. Some might say it's unnecessary.
  39. For Counter = LBound(sheetnames) To UBound(sheetnames)
  40. Set SheetObjects(Counter) = Template.Sheets(sheetnames(Counter))
  41. Next Counter
  42. 'This is where the parameters for the given trial are set in the template file. Trialchecker is set elsewhere
  43. '(it checks a yes/no dropdown in the original spreadsheet). ParameterAddresses is a range that's grabbed from a
  44. 'table object in the original spreadsheet and contains where these parameters go in the template file. These
  45. 'will not change depending on the trial, thus column = 1. TrialParameters is in the same table, and are the
  46. 'parameters themselves. These DO depend on the trial, and thus the column is equal to the trial number
  47. If TrialChecker = True Then
  48. For Counter = LBound(ParameterAddresses) To UBound(ParameterAddresses)
  49. CutoffsShifts.Range(ParameterAddresses(Counter, 1)).Value = TrialParameters(Counter, Trial)
  50. Next Counter
  51. End If
  52. For DataSet = 1 To IntermediaryWorkbook.Worksheets.Count - 1
  53. 'This is where I start my timers
  54. StartTime = Timer
  55. StartDate = Date
  56. 'This is where the data is actually copied from the intermediary file into the template. It's always five
  57. 'columns wide, but can be any number of rows. the SummaryRunTimes statement is merely grabbing the unique
  58. 'identifier of that given worksheet
  59. With IntermediaryWorkbook
  60. Set CopiedDataRange = Calcs.Range("$A$3:$E$" & .Worksheets(Counter).UsedRange.Rows.Count + 1)
  61. CopiedDataRange.Value = IntermediaryWorkbook.Worksheets(Counter).Range("$A$2:$E$" & .Worksheets(Counter).UsedRange.Rows.Count).Value
  62. SummaryRunTimes.Cells(Counter + 1, 1) = Calcs.Cells(3, 1).Value
  63. End With
  64. 'First timestamp
  65. SummaryRunTimes.Cells(Counter + 1, 2) = CStr(Round(86400 * (Date - StartDate) + Timer - StartTime, 1))
  66. StartTime = Timer
  67. StartDate = Date
  68. 'This statement copies down the formulas that go with the data (which is aobut 100 columsn worth of formuals).
  69. 'Throughout this process, calculation is set to manual, so calculation is manually triggered here (Don't ask
  70. 'me why I do it twice. If I recall, it's because pivot tables are weird)
  71. Set CopiedFormulaRange = Calcs.Range("$F$3:$KL$" & Calcs.UsedRange.Rows.Count)
  72. CopiedFormulaRange.FillDown
  73. Application.Calculate
  74. Template.RefreshAll
  75. Application.Calculate
  76. 'Second timestamp
  77. SummaryRunTimes.Cells(Counter + 1, 3) = CStr(Round(86400 * (Date - StartDate) + Timer - StartTime, 1))
  78. StartTime = Timer
  79. StartDate = Date
  80. 'This is a separate function that copies data from the template file into the summary sheet.
  81. 'I know you can't see the code, but just know that it only copies six sets of seven cells, so
  82. 'as far as I can tell, it's not what is causing the problem. The timestamp supports this idea, as
  83. 'it's consistent and short
  84. Call SummaryPopulate(Summary, sheetnames, SheetObjects, r)
  85. r = r + 1
  86. 'Third timestamp
  87. SummaryRunTimes.Cells(Counter + 1, 4) = CStr(Round(86400 * (Date - StartDate) + Timer - StartTime, 1))
  88. StartTime = Timer
  89. StartDate = Date
  90. 'These following few lines are meant to save the template file as a new file. As I mentioned, this is where
  91. 'things get bogged down. FileNameSuffix is a string set via a InputBox. TrialNames is set via the table object
  92. 'mentioned above, and is an array of strings.
  93. Application.DisplayAlerts = False
  94. If TrialChecker = True Then
  95. FileSaveName = FolderLocation & "\" & Replace(Calcs.Cells(3, 1).Value, "/", " ") & " OOIP " & FileNameSuffix & " - " & TrialNames(1, Trial) & ".xlsm"
  96. Else
  97. FileSaveName = FolderLocation & "\" & Replace(Calcs.Cells(3, 1).Value, "/", " ") & " OOIP " & FileNameSuffix & ".xlsm"
  98. End If
  99. Template.SaveAs Filename:=FileSaveName, ConflictResolution:=xlLocalSessionChanges
  100. Application.DisplayAlerts = True
  101. 'This part clears the copied data and formulas. I added the two Set Nothing lines in the hopes that it would
  102. 'solve my problem, but it doesn't seem to do anything
  103. CopiedDataRange.ClearContents
  104. CopiedDataRange.Offset(1, 0).Rows.Delete
  105. Set CopiedDataRange = Nothing
  106. Set CopiedFormulaRange = Nothing
  107. 'Fourth and final timestamp
  108. SummaryRunTimes.Cells(Counter + 1, 5) = CStr(Round(86400 * (Date - StartDate) + Timer - StartTime, 1))
  109. 'It seems to run a bit better if there's this Wait line here, but I'm not sure why. The WaitTime
  110. 'is grabbed from the original worksheet, and is a Double
  111. Application.Wait (TimeSerial(Hour(Now()), Minute(Now()), Second(Now()) + WaitTime))
  112. Next DataSet
  113. 'This but simply saves the summary file and then closes that and the template file. Then the process starts anew.
  114. 'This seems to be the key for resetting something that reduces the run times.
  115. If TrialChecker = True Then
  116. Summary.SaveAs Filename:=FolderLocation & "\" & "OOIP Summary " & FileNameSuffix & " - " & TrialNames(1, Trial) & ".xlsx"
  117. Else
  118. Summary.SaveAs Filename:=FolderLocation & "\" & "OOIP Summary " & FileNameSuffix & ".xlsx"
  119. End If
  120. Template.Close False
  121. Summary.Close False
  122. Next Trial
  123. Application.ScreenUpdating = True
  124. Application.Calculation = xlCalculationAutomatic
  125. IntermediaryWorkbook.Close False
  126. End Sub
0tdrvxhp

0tdrvxhp1#

很抱歉把这个作为答案发布,其实不是,但我需要一点空间。我看了你的代码,我发现没有定义IntermediateWorkbook,并决定定义它不会有什么不同。我确信您已经完成了我可能想到要做的所有事情,我对您的代码的研究不会发现您尚未发现的任何事情。因此,我寻找一种解决方案,首先将这些过程分开,然后用另一种方式将它们重新连接起来--或者不这样做,这就是我的"解决方案"的关键:如果部分不能连接,让他们单独运行。因此,我设置的任务是创建单独的部分。
第1部分这在你的第2点到第4点中有描述,即创建中间工作簿。你没有说明为什么用户必须在创建工作簿之前选择模板,但如果这与模板可以打开和关闭有关。我建议的重要部分是在保存中间工作簿时结束该过程。关闭它。关闭模板。项目完成-第1部分。
第2部分打开中间文件并循环遍历其数据,创建新文件。每个文件都基于一个模板。如果有多个模板可供选择,并且中间工作簿中的数据不支持自动选择,则可能需要提供代码以启用正确模板的选择。在此过程中,您一次只能打开中间工作簿和一个新文件。每个文件在创建新文件之前都会被关闭。在这个过程结束时,中间文件也会被关闭。(顺便说一句,我突然想到,您对模板的处理可能是问题的原因。在我的过程描述中,模板从未打开过。相反,新的工作簿是基于它创建的,这是发明者的设计。)
第3部分创建或打开摘要文件。打开每个新创建的工作簿,并将一行复制到摘要文件中。然后关闭每个工作簿,并打开下一个工作簿。在此过程结束时,关闭摘要工作簿。
连接部件:坦率地说,我会尝试从一开始就把第三部分整合到第二部分中。我不认为多打开一本练习册会有什么不同。但如果有,那就把任务分开。
您的两个或三个单独的过程应该在一个加载项或一个工作簿中,该工作簿除了保存代码外什么也不做(向两个或三个其他工作簿添加一个打开的工作簿-Excel可以轻松处理这一点)。在此工作簿中的代码中添加一个sub,该sub依次调用两个或三个procs。
在这种程序结构中,当保存每个新工作簿所需的时间逐渐增加时,您的问题可能会在第2部分中再次出现,如果发生这种情况,问题的性质将发生变化,应该更容易理解,并且希望更容易解决。

相关问题