热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

解决MSSQL数据库中的断号问题

名词解释断号:比如,连续生成的编号,由于某种操作(通常为删除)后,产生不连续的编号,我们将这种不连续的编号称为断号。例如,数据库中有一个字段叫合同编号,正常格式为201106_011(表示2011年6月的第11个合同),那么它前面的一个合同编号应该为201106_10
名词解释
断号:比如,连续生成的编号,由于某种操作(通常为删除)后,产生不连续的编号,我们将这种不连续的编号称为断号。

例如,数据库中有一个字段叫合同编号,正常格式为201106_011(表示2011年6月的第11个合同),那么它前面的一个合同编号应该为201106_10,后面的一个应该为201106_12,当我们删除了合同201106_011,就会出现201106_010后面直接是201106_012,这种情况下叫做断号。
传统系统中,像这种断号的情况很常见,比如数据库中的列为递增类型,当删除某行后,就会出现断号,而经常有客户提出需求,不希望出现断号的情况。解决方案通常就是,如果删除了某行数据,那么下次新增时,应该将断号补齐。
问题很简单,解决方法也很简单:
写一个C#方法,用来获取下一条记录的编号:
public static int GetNextNumber(int[] iNumList)  
{  
    int iTempStr = iNumList[0];   //用一个临时变量保存上一条记录的编号  
    for (var i = 0; i < iNumList.Length - 1; i++)  
    {  
        if (i == 0)  
        {  
            iTempStr = iNumList[i];  
        }  
        //如果出现断号,则补齐断号  
        if ((iNumList[i] - iTempStr) > 1)  
        {  
            return iTempStr + 1;  
        }  
        else 
        {  
            iTempStr = iNumList[i];  
        }  
        continue;  
    }  
    return iNumList[iNumList.Length - 1] + 1;  

当然,这段代码也可以简写为以下形式:
public static int GetNextNumber3(int[] iNumList)  
{  
    for (int i = 0, j = 1; j < iNumList.Length - 1; i++, j++)  
    {  
         //如果出现断号,则补齐断号  
         if ((iNumList[j] - iNumList[i]) > 1)  
         {   
              return iNumList[i] + 1;    
         }     
    }   
    return iNumList[iNumList.Length - 1] + 1;   

测试
static void Main(string[] args)  
{  
    int[] iNums = { 1, 2, 4, 5, 6, 9, 10 };        //删除了数组中的3,7,8,即3,7,8为断号,下次新增时,希望产生的断号为3  
    System.Console.WriteLine(BreakNumber.GetNextNumber3(iNums));  
    System.Console.WriteLine(BreakNumber.GetNextNumber(iNums));  

运行结果如下:
 
前几天再次接触到这个问题,由于特殊的场景,再用C#反而会增加开发难度,如果想法通过SQL来解决问题:
建表及制造数据SQL:
CREATE TABLE testTable  
(  
    Code int primary key 
)  
INSERT INTO testTable(Code) VALUES (1)  
INSERT INTO testTable(Code) VALUES (2)  
INSERT INTO testTable(Code) VALUES (3)  
INSERT INTO testTable(Code) VALUES (4)  
INSERT INTO testTable(Code) VALUES (5)  
INSERT INTO testTable(Code) VALUES (6)  
INSERT INTO testTable(Code) VALUES (7)  
INSERT INTO testTable(Code) VALUES (8)  
INSERT INTO testTable(Code) VALUES (9)  
INSERT INTO testTable(Code) VALUES (10) 
然后再同样删除第3、7、8行的数据,使这三行产生断号:
DELETE FROM testTable WHERE Code in (3,7,8) 
分析:要产生连号,即是要让Code这一列上连续的,也就是说每每两行之间的Code相差为1
由于Code是从1开始的(从其他数字开始的也是同理计算),即按Code从小到大排序号,Code为1的行应该为第一行,Code为10的行应该在第10行,即Code=行号,
既然这样,预览数据如下:
删除数据前的排号:

删除数据后的排号:
 
很明显发现,删除数据前,Code=行号,删除后Code不等于等号,而删除数据后的第一行Code不等于行号的数据,即是第一个出现断号的数据,即为我们想要查询的结果。
如是,如果数据库中有断号,则可以用以下语句直接查出断号:
 
结果立现。
这段代码还存在一个缺陷,即此方法专用来处理有断号的情况,如果不存在断号时,应该返回Max(RowNumber)+1。正确代码应该如下:
 
至此,我今天要讲的基本结束,此处借用了SQL2005的方法row_number ,其他数据库中也有类似的方法,大家可以自己摸索。
问题完全解决了吗?大家可以发现,以上出现了断号的情况,都是从小开始补号,比如3,7,8同时为断号,则补3。假如有客户要求从大号开始补号(即3,7,8断号时,补8呢),怎么处理?
前面两种通过C#方法操作的就很容易了,这里主要说一下通过SQL处理的方法:

那么再扩展一下,如何查出所有的断号呢?
要实现这个功能,一般想法是将当前Code与上一行的Code进行对比,但由于可能出现连续断号的情况(例如删除了 Code=7、8、9三行)。此时该如何处理呢?
我的解决方法是,假如max(code)等于100,那么我先构造出100行(怎样构造?数据库中随便找个行数大于100的表,select top 100就行了,如果没有行数大于100的表,就联合查询构造出100行吧),再用这一100行的行号分别和code进行对比,如果存在Code<>行号的,即该处为断裂号,示例如下:
假设系统中已经存在另一张表A,它的总行数>max(Code),【注:当然,如果不存在这样的表,也可以通过select 的方式构造出来】,
查询所有断号的SQL如下:
 
至此,问题结束,以上代码的优点在于只用一个SQL语句,而不需要用存储过程、用户自定义函数或C#中的循环,就可以解决各种断号问题,当然为了性能方面还可以再做优化,在此不列出。

推荐阅读
  • 本文探讨了在SQL Server中处理几何类型列时遇到的INTERSECT操作限制,并提供了解决方案,包括通过转换数据类型和使用额外表结构的方法。 ... [详细]
  • 本文详细介绍了如何在C#中生成随机整数,并提供了实用的代码示例和相关资源链接。 ... [详细]
  • 本文详细介绍了在 Ubuntu 16.04 系统上安装和配置 PostgreSQL 数据库的方法,包括如何设置监听地址、启用密码加密、更改默认用户密码以及调整客户端访问控制。 ... [详细]
  • 从CodeIgniter中提取图像处理组件
    本指南旨在帮助开发者在未使用CodeIgniter框架的情况下,如何独立使用其强大的图像处理功能,包括图像尺寸调整、创建缩略图、裁剪、旋转及添加水印等。 ... [详细]
  • 本文探讨了如何通过Service Locator模式来简化和优化在B/S架构中的服务命名访问,特别是对于需要频繁访问的服务,如JNDI和XMLNS。该模式通过缓存机制减少了重复查找的成本,并提供了对多种服务的统一访问接口。 ... [详细]
  • 本文记录了在Windows 8.1系统环境下,使用IIS 8.5和Visual Studio 2013部署Orchard 1.7.2过程中遇到的问题及解决方案,包括503服务不可用错误和web.config配置错误。 ... [详细]
  • 在处理大量联系人数据的批量插入操作时,发现现有方法的执行效率低下,尤其是在处理数十条记录以上时,与导出操作的速度形成鲜明对比。本文将探讨如何通过代码优化来提升批量插入联系人的效率。 ... [详细]
  • HTML:  将文件拖拽到此区域 ... [详细]
  • 在处理大数据量的SQL分页查询时,通常需要执行两次查询来分别获取数据和总记录数。本文介绍了一种优化方法,通过单次查询同时返回分页数据和总记录数,从而提高查询效率。 ... [详细]
  • 流处理中的计数挑战与解决方案
    本文探讨了在流处理中进行计数的各种技术和挑战,并基于作者在2016年圣何塞举行的Hadoop World大会上的演讲进行了深入分析。文章不仅介绍了传统批处理和Lambda架构的局限性,还详细探讨了流处理架构的优势及其在现代大数据应用中的重要作用。 ... [详细]
  • Bootstrap Paginator 分页插件详解与应用
    本文深入探讨了Bootstrap Paginator这款流行的JavaScript分页插件,提供了详细的使用指南和示例代码,旨在帮助开发者更好地理解和利用该工具进行高效的数据展示。 ... [详细]
  • 本文将详细介绍Mybatis Plus框架中@TableField注解的具体使用方法及其应用场景,帮助开发者更好地理解和利用这一强大工具进行高效的数据操作。 ... [详细]
  • java类名的作用_java下Class.forName的作用是什么,为什么要使用它?
    湖上湖返回与带有给定字符串名的类或接口相关联的Class对象。调用此方法等效于:Class.forName(className,true,currentLoader) ... [详细]
  • SQL查询数据表结构的方法
    本文介绍了如何使用SQL命令查看数据库中特定表的创建结构,这对于数据库管理和维护非常有用。文章还提供了一些额外的技巧和最佳实践。 ... [详细]
  • flea,frame,db,使用,之 ... [详细]
author-avatar
捕鱼达人2502873933
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有