何谓自定义排序,就是按指定的顺序对数据源进行排序呗。
 
共分享了三种方法:
第1种方法是系统自带的OrderCustom,优点是代码简洁,缺点是自定义序列有字符长度限制(255个)。
第2种方法是字典+数组设置序列号,再使用了辅助列进行排序。优点是不会破坏单元格的形式和结构,比如单元格中存在的公式、背景等。
第3种方法是只使用字典+数组,借助简单桶排序的技巧,直接对数据在数组中进行排序。优点是效率较高,缺点是会破坏单元格的结构,比如消除公式等。
(第1种建议掌握,第2种建议了解,第3种……能懂就懂,不懂先放着吧~)
 
举个例子。
如下图所示,A:C列是数据源。
现需要根据E列所指定的部门先后顺序,对数据源进行重新排序,如果部门不在指定序列内,则排放在数据源末尾。
Sub 自定义排序1()
    'eh技术论坛 VBA编程学习与实践 看见星光
    Dim n&, rng As Range
    Set rng = Range("e2:e" & Cells(Rows.Count, "e").End(xlUp).Row)
   Application.AddCustomList (rng)
   '增加一个自定义序列,该参数除了支持单元格对象,也支持数组。
    n = Application.CustomListCount
    '自定义序列的数目
    Range("a:c").Sort key1:=[a1], order1:=xlAscending, HEADER:=xlYes, ordercustom:=n + 1
   '使用自定义排序,ordercustom指定使用哪个自定义序列排序。
   '当使用自定义排序时,需要将OrderCustom参数设置为指定的序列在自定义列表中的顺序加1
   Application.DeleteCustomList n
    '删除新增的自定义序列
End Sub
 
Sub 自定义排序2()
    Dim d As Object, r, i&, arr, brr
    Set d = CreateObject("ing.dictionary")
    r = Range("e2:e" & Cells(Rows.Count, "e").End(xlUp).Row).Value
    For i = 1 To UBound(r)
       d(r(i, 1)) = i '目标序列循环装入字典,序号作为item
    Next
    arr = Range("a2:c" & Cells(Rows.Count, 1).End(xlUp).Row)
    '数据源装入数组arr
    ReDim brr(1 To UBound(arr), 1 To 1)
   '声明数组brr装原部门在指定序列中的序号
    For i = 1 To UBound(arr)
       If d.exists(arr(i, 1)) Then
           brr(i, 1) = d(arr(i, 1)) '将原部门在指定序列中的序列号装入brr
       Else
           brr(i, 1) = "指定序列不存在"
       End If
    Next
    [d:d].Insert
    '在D列插入一列
    [d2].Resize(UBound(brr), 1) = brr
    '新的序列号放入D列
    Range("a:d").Sort key1:=[d1], order1:=xlAscending, Header:=xlYes 'D列升序排序
    [d:d].Delete '删除D列
    Set d = Nothing
End Sub
 
Sub 自定义排序3()
    'eh技术论坛公众号 VBA编程学习与实践 看见星光
    Dim d As Object, i&, n&, x&, k&, j&
    Dim r, arr, brr, crr
    Set d = CreateObject("ing.dictionary")
    '后期绑定字典
    r = Range("e2:e" & Cells(Rows.Count, "e").End(xlUp).Row).Value
    For i = 1 To UBound(r)
       d(r(i, 1)) = i '目标序列循环装入字典,序号作为item
    Next
    arr = Range("a2:c" & Cells(Rows.Count, 1).End(xlUp).Row)
    '数据源装入数组
    ReDim brr(1 To d.Count + 1, 1 To 1)
   'brr数组用于按序号装数组arr的行号,类似于桶排序的桶
    For i = 1 To UBound(arr)
       If d.exists(arr(i, 1)) Then
          '如果字典中存在相关部门……
           n = d(arr(i, 1))
          '该部门在指定序列中的序号
           brr(n, 1) = brr(n, 1) & "," & i
          '将该部门在arr中的行号装入数组brr对应的序号行
       Else
          brr(UBound(brr), 1) = brr(UBound(brr), 1) & "," & i
          '如果字典中不存在,放入数组brr最后一行
       End If
    Next
    ReDim crr(1 To UBound(arr), 1 To UBound(arr, 2))
    '数组crr放排序后的结果
    For i = 1 To UBound(brr)
       If brr(i, 1) <> "" Then
          '如果不为空,则有符合指定排序条件的关键词
           r = Split(brr(i, 1), ",")
          '将brr该位置储存的行号取出
           For x = 1 To UBound(r)
              k = k + 1 '累加行
              For j = 1 To UBound(arr, 2)
                 crr(k, j) = arr(r(x), j)
                 '遍历指定行位置数组arr的值移到crr
              Next
           Next
       End If
    Next
    Range("a2:c" & Cells(Rows.Count, 1).End(xlUp).Row) = crr
   '将数组crr排序后的结果放回单元格区域
    Set d = Nothing '释放字典
    Erase arr: Erase brr: Erase crr
    '释放数组
End Sub

【转载】EXCEL VBA 自定义排序的三种方法的更多相关文章

  1. 转载:WinForm中播放声音的三种方法

    转载:WinForm中播放声音的三种方法 金刚 winForm 播放声音 本文是转载的文章.原文出处:http://blog.csdn.net/jijunwu/article/details/4753 ...

  2. Qt 自定义事件(三种方法:继承QEvent,然后Send Post就都可以了,也可以覆盖customEvent函数,也可覆盖event()函数)

    Qt 自定义事件很简单,同其它类库的使用很相似,都是要继承一个类进行扩展.在 Qt 中,你需要继承的类是 QEvent. 继承QEvent类,你需要提供一个QEvent::Type类型的参数,作为自定 ...

  3. linux之在当前目录下按照文件大小进行排序的三种方法

    当前目录下按照文件大小排序 [root@test23 script]# ls -lSh 总用量 44K -rw-r--r-- 1 root root 2.4K 12月 8 17:24 test.con ...

  4. Qt下存储读写应用程序设置的三种方法

    一.简介 用户对应用程序经常有这样的要求:要求它能记住它的settings,比如窗口大小.位置和密码等等.有三种方法可以实现: 使用注册表: 使用配置文件(.ini): 使用自定义文件(例如.txt) ...

  5. 【转】asp.net导出数据到Excel的三种方法

    来源:http://www.cnblogs.com/lishengpeng1982/archive/2008/04/03/1135490.html 原文出处:http://blog.csdn.net/ ...

  6. 三种方法实现Hadoop(MapReduce)全局排序(1)

    我们可能会有些需求要求MapReduce的输出全局有序,这里说的有序是指Key全局有序.但是我们知道,MapReduce默认只是保证同一个分区内的Key是有序的,但是不保证全局有序.基于此,本文提供三 ...

  7. 服务器文档下载zip格式 SQL Server SQL分页查询 C#过滤html标签 EF 延时加载与死锁 在JS方法中返回多个值的三种方法(转载) IEnumerable,ICollection,IList接口问题 不吹不擂,你想要的Python面试都在这里了【315+道题】 基于mvc三层架构和ajax技术实现最简单的文件上传 事件管理

    服务器文档下载zip格式   刚好这次项目中遇到了这个东西,就来弄一下,挺简单的,但是前台调用的时候弄错了,浪费了大半天的时间,本人也是菜鸟一枚.开始吧.(MVC的) @using Rattan.Co ...

  8. vue自定义指令,比onerror更优雅的方式实现当图片加载失败时使用默认图,提供三种方法

    首先,来看下效果图(演示一下图片正常加载与加载失败时的效果) 在线体验地址:https://hxkj.vip/demo/vueImgOnerror/ 一.常规方法解决 我们都知道,img标签支持one ...

  9. 【朝花夕拾】Android自定义View篇之(四)自定义View的三种实现方式及自定义属性使用介绍

    前言 转载请声明,转自[https://www.cnblogs.com/andy-songwei/p/10979161.html],谢谢! 尽管Android系统提供了不少控件,但是有很多酷炫效果仍然 ...

随机推荐

  1. python+request+pymysql+pytest数据驱动

    一.pymysql简单使用 1.安装mysql 下载地址:https://www.mysql.com/,安装教程这里不做介绍了,网上一大推. 2.安装pymysql库 在Terminal终端输入:pi ...

  2. this硬绑定

    一.this显示绑定 this显示绑定,顾名思义,它有别于this的隐式绑定,而隐式绑定必须要求一个对象内部包含一个指向某个函数的属性(或者某个对象或者上下文包含一个函数调用位置),并通过这个属性间接 ...

  3. How to Create DLL(Dynamic link library)

    该文章属于在YouTube视频上看到的,链接如下: https://www.youtube.com/watch?v=EmDJsl7C9-k&t=3s 1.创建一个工程并建立一个控制台程序 2. ...

  4. python的基本运用

    python基础 Python语言是一种解释型.面向对象.动态数据类型的高级程序设计语言 开发者:Guido van Rossum(人称龟叔) 基本概念 1.变量 变量名必须是大小写英文字母.数字或下 ...

  5. uoj220【NOI2016】网格

    刚了几个小时啊,这tm要是noi我怕不是直接滚粗了.我判答案为1的情况试了几种做法,最后终于想到了一个靠谱的做法,然后细节巨多,调了好久,刚拿到97分时代码有6.2KB了,后来发现有些东西好像没啥用就 ...

  6. Vue学习之---浏览器本地存储(8/17)

    博客园(纯干货):https://www.cnblogs.com/zheng-yuzhu/ 文章目录 1.基础知识 2.代码实例(localStorage.html) 3.测试效果 4.代码实例(se ...

  7. java中获取当前执行线程的名称

    Thread.currentThread().getName()

  8. 设计模式常用的UML图------类图

    关系 UML将事物之间的联系归纳为6种,对应响应的图形 关联 定义:表示拥有的关系,具有方向性,一个类单向访问一个类,为单向关联.两个类可以相互访问,为双向关联. 聚合 定义:整体与部分的关系. 组合 ...

  9. 都卷Java,你看看你得学多少技术栈才能工作!

    作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言:授业解惑 我知道,你不知道的还有很多! 你了解计算机要从哪里开始学习吗?你清楚为了能 ...

  10. 43.Permission源码解析和自定义权限类

    drf的权限类位于permission模块   如何确定权限 认证.限流,权限决定是否应该接收请求或拒绝访问 权限检查在视图的最开始处执行,在继续执行其他代码前 权限检查通常会使用request.us ...