一、数组方案
Sub CustomFilter()
Dim Rng As Range, Arr As Variant
Dim EndRow As Long, EndCol As Long
Dim i As Long, j As Long
Dim n As Long
Dim StartDate, EndDate
Dim BeginTime, EndTime
Dim Brr() As String
Dim StartTime As Variant
Dim UsedTime As Variant
StartTime = VBA.Timer
'获取原始数据
With Sheets("原始数据")
'获取A列最后一行(非空行)的行号
EndRow = .Cells(.Cells.Rows.Count, 1).End(xlUp).Row
'获取第一行最后一列(非空列)的列号
EndCol = .Cells(1, .Cells.Columns.Count).End(xlToLeft).Column
'保存数据
Set Rng = .Range(.Cells(2, 1), .Cells(EndRow, EndCol))
'Debug.Print Rng.Address
'存入数组
Arr = Rng.Value
End With
'获取时间设定
With Sheets("筛选设定")
StartDate = .Range("A2").Text
EndDate = .Range("B2").Text
BeginTime = .Range("A4").Text
EndTime = .Range("B4").Text
End With
'循环筛选符合条件的数据
'重新声明数组,用于保存筛选出来的数据
ReDim Brr(1 To EndCol, 1 To 1)
'初始化筛选结果的数量
n = 0
For i = LBound(Arr) To UBound(Arr)
If DateDiff("d", CDate(StartDate), CDate(Arr(i, 1))) >= 0 And _
DateDiff("d", CDate(Arr(i, 1)), CDate(EndDate)) >= 0 And _
Arr(i, 2) >= TimeValue(BeginTime) And _
Arr(i, 2) <= TimeValue(EndTime) Then
'时间在 Arr=Rng.Value的时候已经自动转为TimeValue
n = n + 1
ReDim Preserve Brr(1 To EndCol, 1 To n)
For j = 1 To EndCol
Brr(j, n) = Arr(i, j)
Next j
End If
Next i
'输出结果
With Sheets("筛选数据")
'清除首行标题以外的内容
.UsedRange.Offset(1).ClearContents
'设置筛选数据的输出区域
Set Rng = .Range("A2")
Set Rng = Rng.Resize(UBound(Brr, 2), UBound(Brr))
'输出筛选结果
Rng.Value = Application.WorksheetFunction.Transpose(Brr)
End With
Set Rng = Nothing
UsedTime = VBA.Timer - StartTime
MsgBox "本次运行耗时:" & Format(UsedTime, "#0.0000秒")
End Sub
二、SQL方案
Sub ADO_SQL_QUERY_LOOP()
Dim StartTime As Variant
Dim UsedTime As Variant
StartTime = VBA.Timer
'变量声明
Dim Wb As Workbook
Dim ResultSht As Worksheet
Dim DataSht As Worksheet
Dim Rng As Range
Dim DataPath As String
Dim SQL As String
Dim StartDate, EndDate
Dim BeginTime, EndTime
Dim CNN As Object
Dim RS As Object
Dim DATA_ENGINE As String
'实例化对象
Set Wb = Application.ThisWorkbook
DataPath = Wb.FullName
Set DataSht = Wb.Worksheets("原始数据")
Set ResultSht = Wb.Worksheets("筛选数据")
'获取时间设定
With Wb.Worksheets("筛选设定")
StartDate = .Range("A2").Text
EndDate = .Range("B2").Text
BeginTime = .Range("A4").Text
EndTime = .Range("B4").Text
End With
'根据版本设置连接字符串
Select Case Application.Version * 1
Case Is <= 11
DATA_ENGINE = "Provider=Microsoft.Jet.OLEDB.4.0;Extended Properties='Excel 8.0;HDR=YES;IMEX=2';Data Source="
Case Is >= 12
DATA_ENGINE = "Provider=Microsoft.ACE.OLEDB.12.0;Extended Properties='Excel 12.0;HDR=YES;IMEX=2'; Data Source= "
End Select
'创建ADO Connection 连接器 实例
Set CNN = CreateObject("ADODB.Connection")
'创建 ADO RecordSet 记录集 实例
Set RS = CreateObject("ADODB.RecordSet")
'连接数据源
CNN.Open DATA_ENGINE & DataPath
With ResultSht
'清除首行标题以外的内容
.UsedRange.Offset(1).ClearContents
EndRow = .Cells(.Cells.Rows.Count, 2).End(xlUp).Row
'设置输出结果区域
Set Rng = .Range("A2")
'设置查询语句
SQL = "SELECT * FROM [" & DataSht.Name & "$A1:Z] WHERE 日期 BETWEEN #" & StartDate & "# AND #" & EndDate & "# AND " & _
" 时间 BETWEEN #" & BeginTime & "# AND #" & EndTime & "#"
Debug.Print SQL
'执行查询 返回记录集
Set RS = CNN.Execute(SQL)
'复制记录集到指定Range
Rng.CopyFromRecordset RS
End With
'关闭记录集
RS.Close
'关闭连接器
CNN.Close
Set RS = Nothing
Set CNN = Nothing
Set Wb = Nothing
Set DataSht = Nothing
Set ResultSht = Nothing
Set Rng = Nothing
UsedTime = VBA.Timer - StartTime
MsgBox "本次运行耗时:" & Format(UsedTime, "#0.0000秒")
End Sub