在编写基于数据库的应用时,有一个常见的需求:某一张表有个编码字段,需要按照一定的规则生成,例如:某订单编号的生成规则是:部门编号+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;