数据库[分库分表]中间件 Sharding-JDBC 源码分析 —— SQL 解析(四)之插入SQL
🙂🙂🙂关注微信公众号:【芋艿的后端小屋】有福利:
RocketMQ / MyCAT / Sharding-JDBC 所有源码分析文章列表 RocketMQ / MyCAT / Sharding-JDBC 中文注释源码 GitHub 地址 您对于源码的疑问每条留言都将得到认真回复。甚至不知道如何读源码也可以请教噢。 新的源码解析文章实时收到通知。每周更新一篇左右。 认真的源码交流微信群。
本文主要基于 Sharding-JDBC 1.5.0 正式版
- 1. 概述
- 2. InsertStatement
- 3. #parse()
- 3.1 #parseInfo()
- 3.2 #parseColumns()
- 3.3 #parseValues()
- 3.4 #parseCustomizedInsert()
- 3.5 #appendGenerateKey()
- 666. 彩蛋
1. 概述
本文前置阅读:
- 《SQL 解析(一)之词法解析》
- 《SQL 解析(二)之SQL解析》
本文分享插入SQL解析的源码实现。
不考虑 INSERT SELECT 情况下,插入SQL解析比查询SQL解析复杂度低的多的多。不同数据库在插入SQL语法上也统一的多。本文分享 MySQL 插入SQL解析器 MySQLInsertParser。
MySQL INSERT 语法一共有 3 种 :
- 第一种:
INSERT{VALUES|VALUES}
INSERT [LOW_PRIORITY | DELAYED | HIGH_PRIORITY][IGNORE]
[INTO] tbl_name
[PARTITION (partition_name,...)]
[(col_name,...)]
{VALUES | VALUE}({expr | DEFAULT},...),(...),...
[ ON DUPLICATE KEY UPDATE
col_name=expr
[, col_name=expr]...]
- 第二种:
INSERT SET
INSERT [LOW_PRIORITY | DELAYED | HIGH_PRIORITY][IGNORE]
[INTO] tbl_name
[PARTITION (partition_name,...)]
SET col_name={expr | DEFAULT},...
[ ON DUPLICATE KEY UPDATE
col_name=expr
[, col_name=expr]...]
- 第三种:
INSERT SELECT
INSERT [LOW_PRIORITY | HIGH_PRIORITY][IGNORE]
[INTO] tbl_name
[PARTITION (partition_name,...)]
[(col_name,...)]
SELECT ...
[ ON DUPLICATE KEY UPDATE
col_name=expr
[, col_name=expr]...]
Sharding-JDBC 目前支持:
- 第一种:
INSERT{VALUES|VALUES}
单条记录 - 第二种:
INSERT SET
Sharding-JDBC 插入SQL解析主流程如下:
// AbstractInsertParser.java
publicfinalInsertStatement parse(){
sqlParser.getLexer().nextToken();// 跳过 INSERT 关键字
parseInto();// 解析INTO
parseColumns();// 解析表
if(sqlParser.equalAny(DefaultKeyword.SELECT,Symbol.LEFT_PAREN)){
thrownewUnsupportedOperationException("Cannot support subquery");
}
if(getValuesKeywords().contains(sqlParser.getLexer().getCurrentToken().getType())){// 第一种插入SQL情况
parseValues();
}elseif(getCustomizedInsertKeywords().contains(sqlParser.getLexer().getCurrentToken().getType())){// 第二种插入SQL情况
parseCustomizedInsert();
}
appendGenerateKey();// 自增主键
return insertStatement;
}
Sharding-JDBC 正在收集使用公司名单:传送门。🙂 你的登记,会让更多人参与和使用 Sharding-JDBC。传送门Sharding-JDBC 也会因此,能够覆盖更多的业务场景。传送门登记吧,骚年!传送门
2. InsertStatement
插入SQL 解析结果。
publicfinalclassInsertStatementextendsAbstractSQLStatement{
/**
* 插入字段
*/
privatefinalCollection<Column> columns =newLinkedList<>();
/**
*
*/
privateGeneratedKey generatedKey;
/**
* 插入字段 下一个Token 开始位置
*/
privateint columnsListLastPosition;
/**
* 值字段 下一个Token 开始位置
*/
privateint valuesListLastPosition;
}
我们来看下
INSERT INTO t_order(uid,nickname)VALUES(?,?)
的解析结果:3. #parse()
3.1 #parseInto()
解析表。
// AbstractInsertParser.java
/**
* 解析表
*/
privatevoid parseInto(){
// 例如,Oracle,INSERT FIRST/ALL 目前不支持
if(getUnsupportedKeywords().contains(sqlParser.getLexer().getCurrentToken().getType())){
thrownewSQLParsingUnsupportedException(sqlParser.getLexer().getCurrentToken().getType());
}
sqlParser.skipUntil(DefaultKeyword.INTO);
sqlParser.getLexer().nextToken();
// 解析表
sqlParser.parseSingleTable(insertStatement);
skipBetweenTableAndValues();
}
/**
* 跳过 表 和 插入字段 中间的 Token
* 例如 MySQL :[PARTITION (partition_name,...)]
*/
privatevoid skipBetweenTableAndValues(){
while(getSkippedKeywordsBetweenTableAndValues().contains(sqlParser.getLexer().getCurrentToken().getType())){
sqlParser.getLexer().nextToken();
if(sqlParser.equalAny(Symbol.LEFT_PAREN)){
sqlParser.skipParentheses();
}
}
}
其中
#parseSingleTable()
请看《SQL 解析(二)之SQL解析》的 #parseSingleTable()
小节。3.2 #parseColumns()
解析插入字段。
// AbstractInsertParser.java
privatevoid parseColumns(){
Collection<Column> result =newLinkedList<>();
if(sqlParser.equalAny(Symbol.LEFT_PAREN)){
String tableName = insertStatement.getTables().getSingleTableName();
Optional<String> generateKeyColumn = shardingRule.getGenerateKeyColumn(tableName);// 自动生成键信息
int count =0;
do{
// Column 插入字段
sqlParser.getLexer().nextToken();
String columnName =SQLUtil.getExactlyValue(sqlParser.getLexer().getCurrentToken().getLiterals());
result.add(newColumn(columnName, tableName));
sqlParser.getLexer().nextToken();
// 自动生成键
if(generateKeyColumn.isPresent()&& generateKeyColumn.get().equalsIgnoreCase(columnName)){
generateKeyColumnIndex = count;
}
count++;
}while(!sqlParser.equalAny(Symbol.RIGHT_PAREN)&&!sqlParser.equalAny(Assist.END));
//
insertStatement.setColumnsListLastPosition(sqlParser.getLexer().getCurrentToken().getEndPosition()- sqlParser.getLexer().getCurrentToken().getLiterals().length());
//
sqlParser.getLexer().nextToken();
}
insertStatement.getColumns().addAll(result);
}
3.3 #parseValues()
解析值字段
/**
* 解析值字段
*/
privatevoid parseValues(){
boolean parsed =false;
do{
if(parsed){// 只允许INSERT INTO 一条
thrownewUnsupportedOperationException("Cannot support multiple insert");
}
sqlParser.getLexer().nextToken();
sqlParser.accept(Symbol.LEFT_PAREN);
// 解析表达式
List<SQLExpression> sqlExpressions =newLinkedList<>();
do{
sqlExpressions.add(sqlParser.parseExpression());
}while(sqlParser.skipIfEqual(Symbol.COMMA));
//
insertStatement.setValuesListLastPosition(sqlParser.getLexer().getCurrentToken().getEndPosition()- sqlParser.getLexer().getCurrentToken().getLiterals().length());
// 解析值字段
int count =0;
for(Column each : insertStatement.getColumns()){
SQLExpression sqlExpression = sqlExpressions.get(count);
insertStatement.getConditions().add(newCondition(each, sqlExpression), shardingRule);
if(generateKeyColumnIndex == count){// 自动生成键
insertStatement.setGeneratedKey(createGeneratedKey(each, sqlExpression));
}
count++;
}
sqlParser.accept(Symbol.RIGHT_PAREN);
parsed =true;
}
while(sqlParser.equalAny(Symbol.COMMA));// 字段以 "," 分隔
}
/**
* 创建 自动生成键
*
* @param column 字段
* @param sqlExpression 表达式
* @return 自动生成键
*/
privateGeneratedKey createGeneratedKey(finalColumn column,finalSQLExpression sqlExpression){
GeneratedKey result;
if(sqlExpression instanceofSQLPlaceholderExpression){// 占位符
result =newGeneratedKey(column.getName(),((SQLPlaceholderExpression) sqlExpression).getIndex(),null);
}elseif(sqlExpression instanceofSQLNumberExpression){// 数字
result =newGeneratedKey(column.getName(),-1,((SQLNumberExpression) sqlExpression).getNumber());
}else{
thrownewShardingJdbcException("Generated key only support number.");
}
return result;
}
3.4.1 GeneratedKey
自动生成键,属于分片上下文信息。
publicfinalclassGeneratedKey{
/**
* 字段
*/
privatefinalString column;
/**
* 第几个占位符
*/
privatefinalint index;
/**
* 值
*/
privatefinalNumber value;
}
3.4.2 Condition
条件对象,属于分片上下文信息。在插入SQL解析里存储影响分片的值字段。后续《SQL 路由》 会专门分享这块。
publicfinalclassCondition{
/**
* 字段
*/
@Getter
privatefinalColumn column;
// ... 省略其它属性
}
publicfinalclassColumn{
/**
* 列名
*/
privatefinalString name;
/**
* 表名
*/
privatefinalString tableName;
}
3.4 #parseCustomizedInsert()
解析第二种插入SQL:
INSERT SET
。例如:INSERT INTO test SET id =4 ON DUPLICATE KEY UPDATE name ='doubi', name ='hehe';
INSERT INTO test SET id =4, name ='hehe';
privatevoid parseInsertSet(){
do{
getSqlParser().getLexer().nextToken();
// 插入字段
Column column =newColumn(SQLUtil.getExactlyValue(getSqlParser().getLexer().getCurrentToken().getLiterals()), getInsertStatement().getTables().getSingleTableName());
getSqlParser().getLexer().nextToken();
// 等号
getSqlParser().accept(Symbol.EQ);
// 【值】表达式
SQLExpression sqlExpression;
if(getSqlParser().equalAny(Literals.INT)){
sqlExpression =newSQLNumberExpression(Integer.parseInt(getSqlParser().getLexer().getCurrentToken().getLiterals()));
}elseif(getSqlParser().equalAny(Literals.FLOAT)){
sqlExpression =newSQLNumberExpression(Double.parseDouble(getSqlParser().getLexer().getCurrentToken().getLiterals()));
}elseif(getSqlParser().equalAny(Literals.CHARS)){
sqlExpression =newSQLTextExpression(getSqlParser().getLexer().getCurrentToken().getLiterals());
}elseif(getSqlParser().equalAny(DefaultKeyword.NULL)){
sqlExpression =newSQLIgnoreExpression();
}elseif(getSqlParser().equalAny(Symbol.QUESTION)){
sqlExpression =newSQLPlaceholderExpression(getSqlParser().getParametersIndex());
getSqlParser().increaseParametersIndex();
}else{
thrownewUnsupportedOperationException("");
}
getSqlParser().getLexer().nextToken();
// Condition
if(getSqlParser().equalAny(Symbol.COMMA,DefaultKeyword.ON,Assist.END)){
getInsertStatement().getConditions().add(newCondition(column, sqlExpression), getShardingRule());
}else{
getSqlParser().skipUntil(Symbol.COMMA,DefaultKeyword.ON);
}
}while(getSqlParser().equalAny(Symbol.COMMA));// 字段以 "," 分隔
}
3.5 #appendGenerateKey()
当表设置自动生成键,并且插入SQL没写自增字段,增加该字段。例如:
// 主键为user_id
INSERT INTO t_user(nickname, age) VALUES (?,?)
后续 SQL 改写会生成该自增编号,并改写该 SQL。后续《SQL 改写》 会专门分享这块。
privatevoid appendGenerateKey(){
// 当表设置自动生成键,并且插入SQL没写自增字段
String tableName = insertStatement.getTables().getSingleTableName();
Optional<String> generateKeyColumn = shardingRule.getGenerateKeyColumn(tableName);
if(!generateKeyColumn.isPresent()||null!= insertStatement.getGeneratedKey()){
return;
}
// ItemsToken
ItemsToken columnsToken =newItemsToken(insertStatement.getColumnsListLastPosition());
columnsToken.getItems().add(generateKeyColumn.get());
insertStatement.getSqlTokens().add(columnsToken);
// GeneratedKeyToken
insertStatement.getSqlTokens().add(newGeneratedKeyToken(insertStatement.getValuesListLastPosition()));
}
3.5.1 GeneratedKeyToken
自增主键标记对象。
publicfinalclassGeneratedKeyTokenimplementsSQLToken{
/**
* 开始位置
*/
privatefinalint beginPosition;
}
666. 彩蛋
😈 是不是比《SQL 解析(三)之插入SQL》简单很多。
道友,可否分享一波【本文】到朋友圈。
继续加油更新.
阅读原文 最新评论
推荐文章
作者最新文章
你可能感兴趣的文章
Copyright Disclaimer: The copyright of contents (including texts, images, videos and audios) posted above belong to the User who shared or the third-party website which the User shared from. If you found your copyright have been infringed, please send a DMCA takedown notice to [email protected]. For more detail of the source, please click on the button "Read Original Post" below. For other communications, please send to [email protected].
版权声明:以上内容为用户推荐收藏至CareerEngine平台,其内容(含文字、图片、视频、音频等)及知识版权均属用户或用户转发自的第三方网站,如涉嫌侵权,请通知[email protected]进行信息删除。如需查看信息来源,请点击“查看原文”。如需洽谈其它事宜,请联系[email protected]。
版权声明:以上内容为用户推荐收藏至CareerEngine平台,其内容(含文字、图片、视频、音频等)及知识版权均属用户或用户转发自的第三方网站,如涉嫌侵权,请通知[email protected]进行信息删除。如需查看信息来源,请点击“查看原文”。如需洽谈其它事宜,请联系[email protected]。