Excel·VBA数组组合函数、组合求和

76 篇文章 27 订阅
8 篇文章 0 订阅

《百度百科-组合》
组合、组合数:从n个不同元素中,任取m(m≤n)个元素并成一组,叫做从n个不同元素中取出m个元素的一个组合;从n个不同元素中取出m(m≤n)个元素的所有组合的个数,叫做从n个不同元素中取出m个元素的组合数

  • 组合个数公式:C(n,m)=n!/(m!(n-m)!)

1,组合代码思路

从1-7共7个数字中选出5个数字,共21种组合,如图
在这里插入图片描述
观察每个组合的数字排列规律,可以发现每次最后一个数字排列到尾数7的时候,前面一个数字递增1,后面依次排列到尾数7;且后面的几个数字如果存在依次递增的情况时,前面的一个数字递增1,后面依次排列到尾数7,直至完成21种组合

2,VBA数组组合函数(返回二维数组)

Function combin_arr(ByVal arr, ByVal n&)
    'arr一维数组,内含m个元素,抽取n个进行组合,返回二维数组,每行为一个组合(数组从1开始计数)
    Dim i&, j&, k&, l&, m&, kk&, t&, temp
    If LBound(arr) = 0 Then  '转为从1开始计数
        arr = WorksheetFunction.Transpose(WorksheetFunction.Transpose(arr))
    End If
    If n = 1 Then combin_arr = WorksheetFunction.Transpose(arr): Exit Function
    m = UBound(arr) - LBound(arr) + 1
    kk = Application.Combin(m, n)
    ReDim brr(1 To kk, 1 To n)
    
    ReDim a&(1 To n)
    For j = 1 To n - 1
        a(j) = j
    Next
    
    i = n - 1: k = 0 ': j = n  '上面for结束后j=n;加不加j = n都一样
    Do
        For i = a(n - 1) + 1 To m  '仅修改最后一位
            a(n) = i
            k = k + 1
            For l = 1 To n
                brr(k, l) = arr(a(l))
            Next
        Next
        If a(n - 1) <> a(n) - 1 And a(n) = m Then
            a(n - 1) = a(n - 1) + 1
        ElseIf a(n - 1) = a(n) - 1 And a(n) = m Then
            For t = n - 1 To 1 Step -1  'a(j)进步,避免n=2情况报错,因而只n-1
                If a(t) <> a(t + 1) - 1 Then
                    temp = a(t) + 1: a(t) = temp: t = t + 1
                    Do Until t = n      '为真退出,先判断;最后一位不修改
                        a(t) = a(t - 1) + 1: t = t + 1
                    Loop
                    Exit For
                End If
            Next
        End If
    Loop Until k = kk
    combin_arr = brr
End Function

以上代码及思路,部分参考excelhome大神“香川群子”,原版代码如下

Function CombinArr(arr, n&)
    'arr一维数组,内含m个元素,抽取n个进行组合,返回二维数组,每行为一个组合(数组从1开始计数)
    '源代码by kagawa,https://club.excelhome.net/?218774
    Dim i&, j&, k&, l&, m&
    m = UBound(arr) - LBound(arr) + 1
    k = Application.Combin(m, n)
    ReDim brr(1 To k, 1 To n)
    
    ReDim a&(1 To n)
    For j = 1 To n - 1
        a(j) = j
    Next
    
    i = n - 1: k = 0 ': j = n
    Do
        For i = i + 1 To m
            a(j) = i
            k = k + 1
            For l = 1 To n
                brr(k, l) = arr(a(l))
            Next
        Next
        
        For j = j - 1 To 1 Step -1
            i = a(j) + 1: a(j) = i
            If i = m - n + j Then
                k = k + 1
                For l = 1 To n
                    brr(k, l) = arr(a(l))
                Next
            Else
                j = j + 1
                Do Until j = n
                    i = i + 1: a(j) = i: j = j + 1
                Loop
                If i = m Then Exit Do Else Exit For
            End If
        Next
    Loop Until j = 0
    CombinArr = brr
End Function

以上2种代码写法输出效果一致,耗时基本一致
测试在excel表格中输出1-16共16个数字的全组合形式,共65535种组合,用时都为1秒

Private Sub combin_arr测试()
    Dim arr, brr, i&, r&
    tm = Timer
    arr = Array(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16)
    arr = WorksheetFunction.Transpose(WorksheetFunction.Transpose(arr))  '从1开始计数一维数组
    [a1] = "全组合"
    For i = 1 To UBound(arr)
        brr = combin_arr(arr, i)
        r = Cells(1, "a").CurrentRegion.Rows.count + 1
        Cells(r, "a").Resize(UBound(brr), UBound(brr, 2)) = brr
    Next
    Debug.Print "组合用时" & Format(Timer - tm, "0.00")  '耗时
End Sub

3,组合求和(利用二维数组)

组合最常见的应用就是从一堆数字中凑金额,即组合求和
举例:从A列19个数字中选取6个数字,使其和为39085,在右侧输出结果
在这里插入图片描述

Function TransposeArr(ByVal data_arr, Optional ByVal res& = 1)
    '二维数组与一维嵌套数组互相转换函数,data_arr和返回数组从1开始计数
    Dim i&, j&, result
    If res = 1 Then  '转为一维嵌套数组
        ReDim result(1 To UBound(data_arr) - LBound(data_arr) + 1)
        For i = LBound(data_arr) To UBound(data_arr)
            temp = Application.index(data_arr, i)
            j = j + 1: result(j) = temp
        Next
        TransposeArr = result
    ElseIf res = 2 Then  '转为二维数组
        Dim rr&, cc&, r&, c&, tmp&
        rr = UBound(data_arr) - LBound(data_arr) + 1
        For Each a In data_arr
            tmp = UBound(a) - LBound(a) + 1
            If tmp > cc Then cc = tmp
        Next
        ReDim result(1 To rr, 1 To cc)
        For Each a In data_arr
            r = r + 1: c = 0
            For i = LBound(a) To UBound(a)
                c = c + 1: result(r, c) = a(i)
            Next
        Next
        TransposeArr = result
    End If
End Function
Sub 组合求和()
    Dim m&, n&, h, j&, arr, brr, crr
    tm = Timer
    m = [a1].End(xlDown).row - 1  '待组合元素个数
    n = [b4]  '组合个数
    h = [b2]  '目标和值
    arr = [a2].Resize(m): arr = WorksheetFunction.Transpose(arr)  '单列转一维数组
'--------------------原生函数返回二维数组
'    brr = combin_arr(arr, n)  '调用函数返回组合,二维数组
'    For j = 1 To UBound(brr)  '遍历数组
'        temp = Application.index(brr, j): temp_sum = WorksheetFunction.Sum(temp)
'        If temp_sum = h Then
'            r = Cells(65535, "i").End(xlUp).row + 1
'            Cells(r, "i").Resize(1, 3) = Array(n, temp_sum, Join(temp, "+"))
'        End If
'    Next
'--------------------一维嵌套数组,遍历方法1
    crr = TransposeArr(brr)  '''调用函数,将二维数组转为一维嵌套数组
'    For j = 1 To UBound(crr)
'        temp_sum = WorksheetFunction.Sum(crr(j))
'        If temp_sum = h Then
'            r = Cells(65535, "i").End(xlUp).row + 1
'            Cells(r, "i").Resize(1, 3) = Array(n, temp_sum, Join(crr(j), "+"))
'        End If
'    Next
'--------------------一维嵌套数组,遍历方法2,速度无影响
    For Each c In crr
        temp_sum = WorksheetFunction.Sum(c)
        If temp_sum = h Then
            r = Cells(65535, "i").End(xlUp).row + 1
            Cells(r, "i").Resize(1, 3) = Array(n, temp_sum, Join(c, "+"))
        End If
    Next
    Debug.Print "组合求和完成,累计用时:" & Format(Timer - tm, "0.00")  '耗时
End Sub

测试结果:共27132个组合,耗时秒数

二维数组一维嵌套数组1一维嵌套数组2
241243239

显然速度非常慢,那么数组组合函数直接生成一维嵌套数组时,组合求和速度能否提高呢?

4,VBA数组组合函数(返回一维嵌套数组)

Function combin_arr1(ByVal arr, ByVal n&)
    'arr一维数组,内含m个元素,抽取n个进行组合,返回一维嵌套数组,每行为一个组合(数组从1开始计数)
    Dim i&, j&, k&, l&, m&, kk&, t&, temp
    If LBound(arr) = 0 Then  '转为从1开始计数
        arr = WorksheetFunction.Transpose(WorksheetFunction.Transpose(arr))
    End If
    m = UBound(arr) - LBound(arr) + 1
    kk = Application.Combin(m, n): ReDim brr(1 To kk)
    If n = 1 Then
        For i = 1 To m
            brr(i) = WorksheetFunction.Transpose(WorksheetFunction.Transpose(Array(arr(i))))
        Next
        combin_arr1 = brr: Exit Function
    End If
    
    ReDim a&(1 To n), b(1 To n)
    For j = 1 To n - 1
        a(j) = j
    Next
    
    i = n - 1: k = 0 ': j = n  '上面for结束后j=n,加不加j = n都一样
    Do
        For i = a(n - 1) + 1 To m  '仅修改最后一位
            a(n) = i
            For l = 1 To n
                b(l) = arr(a(l))
            Next
            k = k + 1: brr(k) = b
        Next
        If a(n - 1) <> a(n) - 1 And a(n) = m Then
            a(n - 1) = a(n - 1) + 1
        ElseIf a(n - 1) = a(n) - 1 And a(n) = m Then
            For t = n - 1 To 1 Step -1      'a(j)进步,避免n=2情况报错,因而只n-1
                If a(t) <> a(t + 1) - 1 Then
                    temp = a(t) + 1: a(t) = temp: t = t + 1
                    Do Until t = n          '为真退出,先判断;最后一位不修改
                        a(t) = a(t - 1) + 1: t = t + 1
                    Loop
                    Exit For
                End If
            Next
        End If
    Loop Until k = kk
    combin_arr1 = brr
End Function
Sub 组合求和1()
    Dim m&, n&, h, j&, arr, brr, crr
    tm = Timer
    m = [a1].End(xlDown).row - 1  '待组合元素个数
    n = [b4]  '组合个数
    h = [b2]  '目标和值
    arr = [a2].Resize(m): arr = WorksheetFunction.Transpose(arr)  '单列转一维数组
    brr = combin_arr1(arr, n)  '调用函数返回组合,一维嵌套数组
    For Each b In brr
        temp_sum = WorksheetFunction.Sum(b)
        If temp_sum = h Then
            r = Cells(65535, "i").End(xlUp).row + 1
            Cells(r, "i").Resize(1, 3) = Array(n, temp_sum, Join(b, "+"))
        End If
    Next
    Debug.Print "组合求和完成,累计用时:" & Format(Timer - tm, "0.00")  '耗时
End Sub

测试结果:共27132个组合,耗时仅0.12秒,比二维数组快了2000倍

5,VBA组合求和(利用一维嵌套数组)

Sub 组合求和()
    Dim m&, n&, n2&, h, h2, i&, arr, brr
    tm = Timer
    '参数检查、获取
    If Len([b2]) = 0 Then Debug.Print "B2不得为空": Exit Sub
    m = [a1].End(xlDown).row - 1  '【待组合元素个数m】
    n = [b4]: n2 = [b5]: If n2 = 0 Then If n = 0 Then n = 1: n2 = m Else n2 = n  '【组合个数范围】[n,n2]
    If n = 0 Then n = 1  '情况:n=0, n2=自然数
    h = [b2]: h2 = [b3]: If Len(h2) = 0 Then h2 = h  '【目标和值范围】[h,h2]
    arr = [a2].Resize(m): arr = WorksheetFunction.Transpose(arr)  '单列转一维数组
    For i = n To n2
        brr = combin_arr1(arr, i)  '调用函数返回组合,一维嵌套数组
        For Each b In brr
            temp_sum = WorksheetFunction.Sum(b)
            'If temp_sum >= h And temp_sum <= h2 Then  'h为浮点数时,有时相等却为False,改为下行
            If Abs(Round(temp_sum - h, 6)) < (0.1 ^ 6) Or Abs(Round(temp_sum - h2, 6)) < (0.1 ^ 6) _
            Or (temp_sum >= h And temp_sum <= h2) Then
                r = Cells(65535, "i").End(xlUp).row + 1
                Cells(r, "i").Resize(1, 3) = Array(i, temp_sum, Join(b, "+"))
            End If
        Next
    Next
    Debug.Print "组合求和完成,累计用时:" & Format(Timer - tm, "0.00")  '耗时
End Sub

在这里插入图片描述
测试结果:全组合2^19-1共5242887个组合,耗时仅2.67秒

目录 '1.函数作用:返回 Column 英文字........................9 '2.函数作用:查询某一值第num次出现的值................9 '3.函数作用:返回当个人工资薪金所得为2000元(起征点为850元)时的应纳个人所得税税额.............................10 '4.函数作用:从形如"123545ABCDE"的字符串中取出数字....11 '5.函数作用:从形如"ABCD12455EDF"的字符串中取出数字...11 '6.函数作用:按SplitType取得RangeName串值中的起始位置12 '7.函数作用:将金额数字转成中文大写....................13 '8.函数作用:计算某种税金..............................18 '9.函数作用:人民币大、小写转换........................19 '10.函数作用:查汉字区位码.............................20 '11.函数作用:把公元年转为农历.........................21 '12.函数作用:返回指定列数的列标.......................42 '13.函数作用:用指定字符替换某字符.....................43 '14.函数作用:从右边开始查找指定字符在字符串中的位置...43 '15.函数作用:从右边开始查找指定字符在字符串中的位置...44 '16.函数作用:计算工龄.................................44 '17.函数作用:计算日期差,除去星期六、星期日...........45 '18.函数作用:将英文字反转的自定函数...................46 '19.函数作用:计算个人所得税...........................46 '20.函数作用:一个能计算是否有重复单元的函数...........47 '21.数字金额转中文大写................................48 '22.函数作用:将数字转成英文...........................49 '23.函数作用:人民币大小写转换.........................52 '24.函数作用:获取区域颜色值...........................53 '25.函数作用:获取活动工作表名.........................53 '26.函数作用:获取最后一行行数.........................54 '27.函数作用:判断是否连接在线.........................54 '28.函数作用:币种转换.................................54 '29.函数作用:检验工作表是否有可打印内容...............55 '30.函数作用:查找一字符串(withinstr)在另一字符串中(findstr1)中某一次(startnum)出现时的位置,返回零表示没找到。..................................................57 '31.函数作用:增加文件路径最后的“\”符号..............58 '32.函数作用:计算所得税...............................58 '33.函数作用:从工作表第一行的标题文字以数字形式返回所在列号..................................................58 '34.函数作用:在多个工作表中查找一个范围内符合某个指定条件的项目对应指定范围加总求和..........................59 '35.函数作用:返回 Column 英文字.......................60 '36.函数作用:查找指定列名的列数.......................60 '37.函数作用:文字格式的时间(分:秒)转化为数字格式(秒)..61 '38.函数作用:将"hh:mm:ss"格式的时分秒数转换成秒数.....62 '39.函数作用:金额中文大写转数字.......................62 '40.函数作用:把角度转为度秒分、弧度等显示.............63 '41.函数作用:身份证号码侦测...........................64 '42.函数作用:显示公式.................................65 '43.函数作用:方便财务人员理帐查找.....................66 '44.函数作用:数值转换为字符地址.......................68 '45.函数作用:字符地址转换为数值.......................69 '46.函数作用:等待时间(以秒计算).....................69 '47.函数作用:得到字符串实际的长度(以单字节记).......70 '48.函数作用:18位身份证最后一位有效性验证............70 '49.函数作用:计算符合maturity condition的拆解金额....72 '50.函数作用:对多个用同一分隔符分隔的待查找元素,逐一在表区域首列内搜索,将返回选定单元格的值相加,............72 '51.函数作用:根据个人所得税(工资)反算工资数.........73 '52.函数作用:判断表是否存在...........................74 '53.函数作用:角度转弧.................................74 '54.函数作用:比较相同的字符串.........................75 '55.函数作用:对选定的数组进行排序.....................76 '56.函数作用:取得指定月份天数.........................77 '57.函数作用:排序工作表活页薄.........................77 '58.函数作用:统计数组中非重复数据个数.................78 '59.函数作用:摘取子字符串.............................79 '60.函数作用:计算20000余个汉字的笔画.................79 '61.函数作用:删除当前工作表中的全部超连接.............80 '62.函数作用:取得相近数据.............................81 '63.函数作用:提取定串中汉字...........................81 '64.函数作用:搜索重复数据(选定范围)...................81 '65.函数作用:字符型转数字型...........................82 '66.函数作用:小写人民币转大写人民币...................83 '67.函数作用:取得指定月份人星期天个数.................84 '68.函数作用:侦测档案是否包含宏.......................84 '69.函数作用:获取循环参照单元格.......................85 '70.函数作用:创建桌面快捷方式.........................86 '71.函数作用:自动建立多级目录.........................86 '72.函数作用:统计经筛选后符合条件的记录条数...........87 '73.函数作用:复制单元格列高与栏宽.....................87 '74.函数作用:取消隐藏工作表(包括vba Project工程保护的)88 '75.函数作用:删除单元格自定义名称.....................88 '76.函数作用:从文件路径中取得文件名...................89 '77.函数作用:取得一个文件的扩展名.....................89 '78.函数作用:取得一个文件的路径.......................90 '79.函数作用:十进制转二进制...........................90 '80.函数作用:检查一个数组是否为空.....................90 '81.函数作用:字母栏名转数字栏名.......................91 '82.函数作用:数字栏名转文字栏名.......................91 '83.函数作用:判断一件活页夹中是否还有子目录...........92 '84.函数作用:判断一个文件是否在使用中.................92 '85.函数作用:列出档案详细摘要信息.....................93 '86.函数作用:获取菜单ID编号及名称列表................93 '87.函数作用:状态列动态显示文字.......................94 '88.函数作用:取得一个文件的路径2.....................94 '89.函数作用:取得一个文件的路径3.....................95 '90.函数作用:取得Activecell的栏名....................95 '91.函数作用:取得单元格中指定字符前的字符.............95 '92.函数作用:前单元格指定字符前的字符颜色改成红色.....95 '93.函数作用:根据数字返回对应的字母列号...............96 '94.函数作用:取工作表名字.............................96 '95.函数作用:取消所有隐藏的宏表.......................97 '96.函数作用:导出VBA Project代码.....................97 '97.函数作用:导入VBA Project代码.....................97 '98.函数作用:取得汉字拼音的第一个字母.................98 '99.函数作用:获取两栏中相同的数据....................100 '100.函数作用:选取当前工作表中公式出错的单元格﹐关返回出错个数...............................................101 '101.函数作用:将工作表中最后一列作为页脚打印在每一面页尾101 '102.函数作用:获取vbproject引用项目.................102 '103.函数作用:移除Excel工作表中的外部数据连接.......103 '104.函数作用:将选择定单元格作成镜像图片.............103 '105.函数作用:反选择单元格中的数.....................105 '106.函数作用:在Excel中加入一个量度尺(以厘米为单位).106 '107.函数作用:在Excel中加入一个量度尺(以寸为单位)...109 '108.函数作用:取得一个短文件名的长文件名.............111 '109.函数作用:取得临时文件名.........................112 '110.函数作用:等用Shell调用的程序执行完成后再执行其它程序...................................................112 '111.函数作用:将Mouse显示成动画.....................113 '112.函数作用:限制Mouse移动范围.....................114 '113.函数作用:取得当前激活窗品句柄及标题.............114 '114.函数作用:取得屏幕分辨率.........................115 '115.函数作用:自动建立多级目录.......................115 '116.函数作用:将文件长度置零.........................116 '117.函数作用:读取WIN9X / Me共享文件夹密码..........116 '118.函数作用:取得预设的打印机及设置预设的打印机.....119 '119.函数作用:获得当前操作系统的打印机个数及检测打印是否存在.................................................120 '120.函数作用:枚举打印机名称清单.....................120 '121.函数作用:读取网络服务器当前时间.................122 '122.函数作用:下载文件到指定目录.....................123 '123.函数作用:自动映射网络驱动器.....................124 '124.函数作用:自动断开网络驱动器.....................125 '125.函数作用:连接选定单元格中的内容.................125 '126.函数作用:获取一个单元格中有指定字体颜色部份数据.126 '127.函数作用:对指定文件加XLS加密...................126 '128.函数作用:选择指定范围内使用了填充颜色的单元格...127 '129.函数作用:在特定的区域内查找文本,返回值是包含查找文本的单元格...........................................127 '130.函数作用:返回特定区域中最大值的地址.............128 '131.函数作用:删除表格中使用范围内的所有空白单元格...129 '132.函数作用:返回数组中有多少个指定的字符串.........129 '133.函数作用:返回当前工作表中引用了指定的单元的地址.130 '134.函数作用:获取Excel中字型列表...................131 '135.函数作用:获取一个字符串中有多少个数字字符.......131 '136.函数作用:在Excel中对多列进行填充...............131 '137.函数作用:对选定的范围进行数据填充(忽略单元格格式)132 '138.函数作用:VBA Project加密及解密.................132 '139.函数作用:列出收藏夹中的网址.....................133 '140.函数作用:计算两个日期之间相隔的年份,比如年龄,工龄等.可计算从1000年01月01日起的日期....................134 '141.函数作用:从字符串提取纯数字.....................135 '142.函数作用:将一个数组按升序排列...................136 '143.函数作用:将一个数组按降序排列...................137 '144.函数作用:删除空白列.............................137 '145.函数作用:判断工作表是否为空白...................138 '146.函数作用:将数据按类分到不同工作薄...............138 '147.函数作用:单元格内数据排序.......................139 '148.函数作用:对多栏排序.............................140 '149.函数作用:返回计算公式的值 [,值的计算公式].......140 '150.函数作用:把第一列=某个值对应的第二列的内容连在一起,并用、隔开...........................................141 '151.函数作用:取得系统使用模式.......................142 '152.函数作用:计算机注销/关机/重启...................142 '153.函数作用:更改计算机名称.........................143 '154.函数作用:从n位开始取出字符串中的汉字、英文字母、数字...................................................143 '155.函数作用:在指定列中寻找含有指定字符串的单元格,并将符合条件的单元格标为红色,并将对应的下一列单元格赋值为1。.....................................................144 '156.函数作用:清除字符串中的空格.....................145 '157.函数作用:查找合并单元格位置.....................145 '158.函数作用:阴阳历转换和阴阳历生日.................145 '159.函数作用:利用数组和Substitute来替换某字符......149 '160.函数作用:一键创建斜线表头.......................150 '161.函数作用:自动获取指定月的工作日.................151
评论 14
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

薛定谔_51

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值