在编写基于数据库的应用时,有一个常见的需求:某一张表有个编码字段,需要按照一定的规则生成,例如:某订单编号的生成规则是:部门编号+yyyyMMdd+四位流水号,中间部分代表当前的年月日。难点就是如何生成流水号,并且能够保证在多用户并发的情况下,保证流水号不重复。
得到流水号的方法比较简单:select max(theColumn) from theTable where theColumn like “BBXXXXXXX%”,即在该表中查询具有相同前缀(编码流水号之前的部分)的编码最大值,然后再将流水号部分+1就可以得到新的编码了。为了保证流水号不重复,我们需要锁定数据,但是如果锁定该表的话,开销太大,针对该表的增、删、改操作都不能进行。这里采用一个小技巧:我们单独建立一张新的表格,SQL语句如下:
create table LOCK_TABLE (
TABLE_NAME VARCHAR(20) not null
constraint PK_ LOCK_TABLE primary key (TABLE_NAME)
)
这个表只有一个字段,数据即需要我们生成编号的表名。我们在计算某个表的当前最大流水号之前,首先锁定LOCK_TABLE表中数据为该表名的那条数据(具体方法后文有介绍),然后再执行上面的select max(**) …… 操作,得到新的编码。请注意:上面的锁定LOCK_TABLE表中一行数据;查询最大编码是在一个事务中完成的。
锁定LOCK_TABLE表中数据为该表名的那条数据的方法:就是在普通的SQL语句中加入锁定的关键字,对于Oracle来说是: for update ;对于SQLServer来说是 with (holdlock)。
具体实现代码如下:
- import java.sql.Connection;
- import java.sql.PreparedStatement;
- import java.sql.ResultSet;
- import java.sql.SQLException;
- import java.sql.Statement;
- import org.apache.log4j.Logger;
-
- public class KeyGenerator{
-
- private Logger logger = Logger.getLogger(KeyGenerator.class);
- private Connection conn;
- private String tableName;
- private String columnName;
- private String head;
- private int numCount;
- private String lockString;
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- public KeyGenerator(Connection conn,String tableName,String columnName,String head,int numCount,String lockString) {
- this.conn=conn;
- this.tableName=tableName;
- this.columnName=columnName;
- this.head=head;
- this.numCount=numCount;
- this.lockString=lockString;
-
- }
-
- private String computeNewCode(String maxCode,String head,int numCount){
-
- String newCode="";
- if(maxCode!=null){
- int i=head.length();
- int j=maxCode.length();
- int k=j-i;
-
- String numPart=maxCode.substring(i,j);
- int theInt= new Integer(numPart).intValue();
- theInt++;
- String numString =new Integer(theInt).toString();
- k=k-numString.length();
- String temp0="";
- for(;k>0;k--){
- temp0=temp0+"0";
- }
- numString=temp0+numString;
- newCode=head+numString;
- }
- else{
- String temp0="";
- for(int k=numCount-1;k>0;k--){
- temp0=temp0+"0";
- }
- newCode=head+temp0+"1";
- }
- return newCode;
- }
-
-
- public String getKey() {
-
- String oracleLockStr=" for update ";
- String sqlServerLockStr=" with (holdlock) ";
-
- String sql1 = " SELECT * FROM " + "LOCK_TABLE ";
-
-
-
-
-
- sql1 = sql1+sqlServerLockStr;
- sql1 = sql1+" WHERE " + "TABLE_NAME" + " LIKE '" + lockString.trim() + "'";
-
- String sql2 = " SELECT MAX(" + columnName+ ") AS A FROM "+ tableName ;
- sql2 = sql2+" WHERE " + columnName + " LIKE '" + head.trim() + "%' ";
-
- PreparedStatement pstm1 = null;
- PreparedStatement pstm2 = null;
- Statement stmt = null;
- ResultSet rset1 = null;
- String maxCode="";
- String newCode="";
-
- try {
- pstm1 = conn.prepareStatement(sql1);
- pstm1.executeQuery();
- pstm2 = conn.prepareStatement(sql2);
- rset1 = pstm2.executeQuery();
- rset1.next();
- maxCode=rset1.getString("A");
-
- newCode=computeNewCode(maxCode,head,numCount);
-
- logger.info("newCode:"+newCode);
- System.out.println("newCode:"+newCode);
- return newCode;
-
- } catch (Exception e) {
- System.out.println(e);
- e.printStackTrace();
- return null;
-
- }
- finally{
- try {
- if (rset1 != null)
- rset1.close();
- } catch (SQLException e1) {
- }
-
- try {
- if (pstm1 != null)
- pstm1.close();
- if (pstm2 != null)
- pstm2.close();
- if (stmt != null)
- stmt.close();
- } catch (SQLException e1) {
- }
- }
-
- }
-
- }
分享到:
相关推荐
例如,可以使用`YEAR()`, `MONTH()`, 和 `DAY()` 函数分别获取日期的年、月、日,再结合自定义的序列号或随机数生成器来确保唯一性。 一个简单的实现可能如下: ```sql DECLARE @Year INT = YEAR(GETDATE()); ...
在IT行业中,排列组合生成器是一种常见的工具,用于生成所有可能的排列或组合,这在很多领域都有应用,比如数据分析、密码学、游戏设计、测试用例生成等。本项目专注于后端实现,使用了SpringBoot框架,这是一种流行...
不死鸟西太平洋大学文凭生成器是一款基于PHP编程语言开发的应用程序,主要目的是为了娱乐或教育目的模拟创建文凭证书。这款工具的创意源于一位网友,他可能在某个社区或平台上发布了一份演示,引起了人们的兴趣。PHP...
在数据库管理领域,MSSQL数据生成器是一个极其重要的工具,它主要用于快速生成大量测试数据,以填充SQL Server数据库。在软件开发、性能测试或者数据分析等场景中,需要大量的测试数据来模拟真实环境,此时,mssql...
本篇文章将详细探讨如何在C#中连接到SQL Server并生成访问存储过程的代码,特别是针对描述中的"第二种方式"。 首先,我们需要理解C#与SQL Server交互的基本概念。ADO.NET(.NET Framework的数据访问组件)是实现这...
利用CPU ID、MAC地址等硬件信息,结合随机数生成器,可以创建与特定设备关联的唯一编码。但这可能引发隐私问题,不适用于所有场景。 在测试这些方法时,需要关注以下指标: - **唯一性**:确保在所有测试条件下,...
《动软.Net代码生成器源码解析》 在.NET开发领域,提高开发效率和减少重复工作是每个程序员追求的目标。动软.Net代码生成器就是这样一款工具,它可以帮助开发者快速生成符合特定需求的基础代码,从而节省大量的编码...
通常,这些日志文件位于“C:\Program Files\Microsoft SQL Server\MSSQL.n\MSSQL\LOG\ERRORLOG”路径下,其中“n”是SQL Server实例的编号。 从提供的错误日志中,我们可以看出SQL Server版本为2005,它在启动时会...
### 数据库文档生成器知识点详解 #### 一、概述 数据库文档生成器是一种自动化工具,用于根据现有的数据库结构自动生成数据库文档。这类工具通常能够识别数据库中的表、字段、索引等元素,并根据这些信息创建出易于...
在"测试数据生成器"这个项目中,开发者可能已经实现了一个简单的C#程序,它能根据预设规则生成常见的测试数据,如姓名、电话号码、地址等,并可能将生成的数据保存到一个或多个文件中。这个工具对于软件测试团队来说...
双鱼林vb代码生成器 v2.0是一款Visual Basic 6.0代码生成器,目前能实现整个项目一次性生成,直接双击vbp工程文件打开生成的项目,自动代码生成的代码在project 目录! 真正面向对象设计:系统的整体设计,提供通过...
此SQL语句用于确定当前查询优化器(RBO或CBO)的使用情况。 **SQL语句**: ```sql SQL>select decode(nvl(position, -1), -1, 'RBO', 1, 'CBO') from plan_table where id = 0; ``` **解释**: - `plan_table`: 由`...
- **查询优化器**: 了解SQL Server如何解析和执行查询,以及如何利用索引提升查询速度。 - **性能监控和调优**: 使用动态管理视图(DMVs)和性能计数器进行性能诊断和优化。 7. **错误处理和日志记录** - **异常...
在SQL Server的管理界面中,用户可以通过查询分析器轻松生成数据库或表的SQL脚本。具体操作为:在查询分析器中,选中目标数据库或表,通过右键菜单选择“在文件中编写对象脚本”->“创建”,系统会自动生成对应对象...
采用客户机/浏览器(Client/Browsers)结构,ASP+SQL技术编写。可以为新生儿家长提供互联网查询,方便使用。 管理员:super 密码:www 其主要内容: ★ 系统权限管理; ★ 新建出生证; ★ 修改出生证; ★ ...
【编号查询器(c#结合xml和sql server技术)】是一个基于C#编程语言开发的简单应用程序,它利用XML数据存储和SQL Server数据库管理系统的功能,为用户提供了一个实用的查询工具。这个项目对于初学者来说是一个很好的...
11、表字典生成器让您快速地将本工具使用溶入开发中。 可能您的项目已经着手开发一段时间了,即使是这样,您依然可以让本工具为您效劳。您只需打开表字典生成工具,工具会读取数据库中所有的表及字段信息,根据您...
"plsqldev804.exe"这个文件名可能表示的是PL/SQL Developer 8.0的安装程序,"804"可能是内部版本号或者更新编号。通常,.exe文件是Windows操作系统下的可执行文件,用户可以通过运行这个文件来安装或更新PL/SQL ...