VBA(7)字典及常用应用

1、字典直接创建

Dim dic as object

Set dic = Createobject("scripting.dictionary")

2、引用法

工具-引用-浏览-scrrun.dll-确定

microsoft scripting runtime   打勾

注:两者在使用上经常用创建多一点;并无太大区别。用创建的字典装入数据后并不能直接用dic.keys(N)/dic.items(N) 的格式来引用字典元素.字典元素从 dic.keys(0)开始

3、字典常用的属性与方法

方法:

  .add         '创建新的元素

  .keys                 '字典的元素

  .items               '元素对应的值

  .exists               '是否存在

  .remove           '清除提定元素

  .removeall       '清除所有元素

属性:  

  .key

  .item

  .count                '统计元素个数

  .comparemode'值为1/0/2 文本/英文/数据库格式      '文本格式下不区分大小写

4、基础用法概念

Sub dic1()
    'Dim d As New dictionary                '需工具-引用-microsoft scripting runtime选取后
    Dim d As Object
    Dim arr()
    Dim x%
    Dim m
    Set d = CreateObject("scripting.dictionary")
        d.CompareMode = 1                   '设置为不区分大小写
    [A1:A3] = Application.WorksheetFunction.Transpose(Array("A", "B", "C"))
    [B1:B3] = Application.WorksheetFunction.Transpose(Array(1, 2, 3))
    arr = Range("a1:b3")
    For x = 1 To UBound(arr, 1)
        d(arr(x, 1)) = arr(x, 2)
    Next
    'Stop
    'MsgBox d.keys(2)                       '在本机测试需DIM new dictionary方式下方可用
    [C1:E1] = d.Keys                        '字典keys元素
    [C2:E2] = d.Items                       '字典一个key对应一个Item
    [C3] = d.Count                          '字典D统计元素组个数
    If d.Exists("D") Then                   'Exists查看元素是否存在
        MsgBox "1"
    Else
        'd("D") = 4                          '不存在则增加
        d.Add "D", 4                         'd.add =    d(key)=item
    End If
    Stop                                    '暂停查看字典变化
        d.Key("C") = "5"                     'd.key属性指定元素改变key值
        m = Application.Index(d.Keys, 3)     '用createobject创建的字典用工作表index函数取出赋值
        MsgBox m

    If d.Exists("c") Then                   '前期装入的是"C"。如果不设置区分大小写则不存在
        MsgBox "C存在"
    Else
        d.Remove "b"                        '字典按装入数据的顺序排序
    End If
        d.RemoveAll                        'd.remove 指定keys删除或者Removeall全部
    Stop                                    '暂停查看字典变化
End Sub

5、常见应用:读取数据/去重复/计算/匹配

Sub dic2()  '提取不重复
    Dim d As Object
    Dim arr()
    Dim x As Integer
    Set d = CreateObject("scripting.dictionary")
    d.CompareMode = 1                                                    '设置不分大小写
    [A1:A10] = Application.Transpose(Array("A", "B", "C", "a", "B", "C", "A", "B", "C", "D"))
    [B1:B10] = Application.Transpose(Array(1, 2, 3, 4, 5, 6, 7, 8, 9, 10))
    For x = 1 To 10
        d(Cells(x, 1).Value) = Cells(x, 2).Value + d(Cells(x, 1).Value)        '合并相对应累加,相当于SUMIF
        'd.Add Cells(x, 1).Value, Cells(x, 2).Value                      '用add如遇有重复情况会报错
    Next
    Range("d1").Resize(d.Count) = Application.Transpose(d.Keys)         '字典Keys关键字如装入重复则覆盖上一个,利用此特性可去重复,转置输出单元格
    Range("e1").Resize(d.Count) = Application.Transpose(d.Items)        '对应值汇总值输出
    arr = d.Items
    Stop
End Sub
Sub dic3()  '查询匹配
    Dim d As Object
    Dim x%, y%
    Dim arr()
    [A1:A6] = Application.Transpose(Array("A", "B", "C", "a", "d", "D"))
    [B1:B6] = Application.Transpose(Array(1, 2, 3, 4, 5, 6))
        Set d = CreateObject("scripting.dictionary")
        arr = Range("a1:b6")
        For y = 1 To UBound(arr)
            d(arr(y, 1)) = arr(y, 2)            '把双列的数据分别装入1与2列,对应的值相互查询
            d(arr(y, 2)) = arr(y, 1)
        Next
        Stop
    MsgBox d("a") & d(1) & d("A")
End Sub
Sub dic4()  '多列汇总,字典key1列对应item多列
    Dim hrr(1 To 100, 1 To 3)       '定义一个足够大的放汇总数组
    Dim row As Integer
    Dim arr(), x#, k#
    Dim d As Object
    Set d = CreateObject("scripting.dictionary")
    [A1:C1] = Array("品名", "汇总1", "汇总2")   '生成测试用数值
    [A2:C2] = Array("A", "1", "5")
    [A3:C3] = Array("A", "2", "6")
    [A4:C4] = Array("C", "3", "7")
    [A5:C5] = Array("C", "4", "8")
    arr = Range("a2:c" & Range("a65536").End(xlUp).row)
    For x = 1 To UBound(arr)
        If d.Exists(arr(x, 1)) Then         'Exists某个元素在字典中是否存在
            row = d(arr(x, 1))      '如果存在。行数等于字典中的顺序序号
            hrr(row, 2) = hrr(row, 2) + arr(x, 2)       '对应累加
            hrr(row, 3) = hrr(row, 3) + arr(x, 3)       '如果计数将累加数值改1
        Else
            k = k + 1               '如果不存在。记录序号
            d(arr(x, 1)) = k            '装入字典
            hrr(k, 1) = arr(x, 1)           '数组直接装入对应数值
            hrr(k, 2) = arr(x, 2)
            hrr(k, 3) = arr(x, 3)
        End If
        Next
        Range("G2").Resize(k, 3) = hrr      '汇总后结果输出
End Sub
Sub dic5()      '交叉表样式汇总
    Dim hrr(1 To 100, 1 To 4)
    Dim d As Object
    Dim row1&, column1&
    Dim arr(), x#, k#
    Set d = CreateObject("scripting.dictionary")
    [A1:C1] = Array("品名", "月份", "值")
    [A2:C2] = Array("A", "1月", "5")
    [A3:C3] = Array("A", "2月", "6")
    [A4:C4] = Array("C", "3月", "7")
    [A5:C5] = Array("C", "2月", "8")
    arr = Range("a2:c" & Range("a65536").End(xlUp).row)
    For x = 1 To UBound(arr)
        column1 = (InStr("1月2月3月", arr(x, 2)) + 1) / 2 + 1 'InStr(查找的字符串,找什么字符)返回字符所在位置排列的数字
        If d.Exists(arr(x, 1)) Then
            row1 = d(arr(x, 1))
            hrr(row1, column1) = hrr(row1, column1) + arr(x, 3)
        Else
            k = k + 1                       '多行多列汇总值在于确定行数装入,及怎么样区别列值
            d(arr(x, 1)) = k                '需注意列值是单条件还是多条件取值
            hrr(k, 1) = arr(x, 1)
            hrr(k, column1) = arr(x, 3)
        End If
    Next
    Range("f1:h1") = Array("品名/月份", "1月", "2月")
    Range("f2").Resize(k, 3) = hrr
End Sub
Sub dic6()  '指定条件求整余
    Dim d As Object
    Dim x%
    Dim arr()
    Set d = CreateObject("Scripting.Dictionary")
    For x = 1 To 10
        d.Add x & IIf(Abs(x Mod 3) = 0, "@", ""), ""    '循环1至10如果符合则加标识号@
    Next
    arr = WorksheetFunction.Transpose(Filter(d.Keys, "@"))  '筛选只有标识号的值
    [A1].Resize(UBound(arr), 1) = arr                       '筛选完之后输出单元格"3@,6@,9@"
    [A:A].Replace "@", ""                                   '替代掉标识符
    Set d = Nothing                                         '关闭字典
End Sub
Sub dic7()      '指定类型分类并提取不重复
    Dim str1$, str2$, str3$
    Dim nRow%, d As Object
    Dim Brr(), arr
    Dim s(1 To 4) As Integer, i%
    Set d = CreateObject("scripting.dictionary")
    str1 = Join(Array("类型1A", "类型1B", "类型1C"), ",")      '定义类型1,以数组形式储存
    str2 = Join(Array("种类2A", "种类2B", "种类2C"), ",")      '定义类型2,以数组形式储存
    str3 = Join(Array("型3A", "型3B", "型3C"), ",")      '定义类型3,以数组形式储存
    nRow = Range("a1").End(xlDown).row
    arr = Range("a1:a" & nRow)                          '数据源装入
    ReDim Brr(1 To nRow, 1 To 4)                        '筛选后放置数组
    For i = 2 To nRow                                       '遍历数据源
        If Not d.Exists(arr(i, 1)) Then                         '如果数据不存在执行
            d(arr(i, 1)) = ""                                  '放入字典
            If str1 Like "*" & Left(arr(i, 1), 2) & "*" Then'类型文本1中有跟遍历值相像的话
                s(1) = s(1) + 1                                 '记录类型1列的行累加
                Brr(s(1), 1) = arr(i, 1)                            '赋值给1列
            ElseIf str2 Like "*" & Left(arr(i, 1), 2) & "*" Then
                s(2) = s(2) + 1
                Brr(s(2), 2) = arr(i, 1)
            ElseIf str3 Like "*" & Left(arr(i, 1), 2) & "*" Then
                s(3) = s(3) + 1
                Brr(s(3), 3) = arr(i, 1)
            Else
                s(4) = s(4) + 1                               '都与123类型不类似的放入第4列
                Brr(s(4), 4) = arr(i, 1)                        '记录4列的行累加
            End If
        End If
    Next
    [K1:N1] = Array("类型1", "种类2", "型3", "其他")
    Range("k2:n" & nRow) = Brr                              '输出结果到单元格
End Sub

你可能感兴趣的:(VBA,microsoft)