当前位置: 首页 > news >正文

58同城盐城网站建设网站建设违约责任

58同城盐城网站建设,网站建设违约责任,上海网络关键词优化,微信 网站界面 模板目录 一、前言 二、如何配置 三、源码解读 四、自定义 一、前言 在一次项目功能测试中,发现有个sql在其他嵌入式数据库中执行正常,但是在mysql中执行失败,发现是因为有个字段在mysql中是关键字,需要使用反引号(&…

目录

一、前言

二、如何配置

三、源码解读

四、自定义


一、前言

        在一次项目功能测试中,发现有个sql在其他嵌入式数据库中执行正常,但是在mysql中执行失败,发现是因为有个字段在mysql中是关键字,需要使用反引号(``)包起来才行,但是,反引号在嵌入式数据库(如hsql)中是不支持的,这就导致,同一条sql具有不同的写法。

        让一个项目支持不同的数据库在企业开发中是一个比较常见的需求。由于不同的数据库支持的sql语法稍有差别,所以某些功能需要根据数据库的不同书写不同的sql语句。对于这种需求,首先能够想到的解决方案就是针对不同的数据库维护不同的mapper.xml文件,但是这种方案会严重增加开发和维护的成本。因为不同数据库支持的语法大部分都是相同的,不同的毕竟是少数,我们希望只重写不同的部分而重用相同的部分。

        针对这种情况,MyBatis提供了解决方案,即databaseIdProvider和databaseId。通过MyBatis提供的这种功能,我们就只需要维护一套mapper.xml文件便可。

        下面先讲解如何配置,然后在从源码层面对这个功能进行解读,最后探讨一下如何通过自定义来定制这个功能。

二、如何配置

        配置databaseIdProvider(在mybatis的配置文件中配置,比如mybatis-config.xml)

<databaseIdProvider type="DB_VENDOR"><property name="MySQL" value="mysql" /><property name="HSQL Database Engine" value="hsql" /><property name="DM DBMS" value="dm" /><property name="Derby" value="derby" />
</databaseIdProvider>

        上述配置用于决定当前databaseId的名称。在每一个property标签中,name代表数据库的productName(DatabaseMetaData#getDatabaseProductName()),value是用户自定义的databaseId名称。mybatis在初始化的时候会根据所使用的数据源得到当前databaseId的名称,得到的databaseId的名称供mybatis选择映射文件中相应的语句。比如我们使用的是Mysql数据库,则得到的databaseId名称为“mysql”。

配置databaseId

<!-- mysql数据库生效 -->
<select id="select" resultMap="XXCResultMap" databaseId="mysql">
select name, status, passwords, `keys` from test
</select><!-- 非mysql数据库则生效 -->
<select id="select" resultMap="XXCResultMap" >
select name, status, passwords, keys from test
</select>

        官方文档对databaseId的解释为:

“如果配置了 databaseIdProvider,MyBatis 会加载所有的不带 databaseId 或匹配当前 databaseId 的语句;如果带或者不带的语句都有,则不带的会被忽略”。

        也就是说如果在mybatis的配置文件中没有配置 databaseIdProvider,则在映射文件中配置的databaseId不会生效。

        由于我们使用的是Mysql数据库,得到的databaseId为“mysql”,所以上述的映射片段会在生效。

三、源码解读

        上面我们讲解了如何配置,下面我们从源码的角度来剖析一下mybatis是如何实现这个功能的。

databaseIdprovider的默认实现是VendorDatabaseIdProvider,接口为DatabaseIdProvider。

先看DatabaseIdProvider的定义:

public interface DatabaseIdProvider {default void setProperties(Properties p) {}String getDatabaseId(DataSource var1) throws SQLException;
}

接口很简单,就两个方法。

void setProperties(Properties p) 该方法会把databaseIdprovider标签中配置的属性传递进来。

String getDatabaseId(DataSource dataSource) 该方法会将返回一个databaseId。

        再看一下默认实现类VendorDatabaseIdProvider的源码:

package org.apache.ibatis.mapping;import java.sql.Connection;
import java.sql.DatabaseMetaData;
import java.sql.SQLException;
import java.util.Iterator;
import java.util.Properties;
import java.util.Map.Entry;
import javax.sql.DataSource;
import org.apache.ibatis.logging.Log;
import org.apache.ibatis.logging.LogFactory;public class VendorDatabaseIdProvider implements DatabaseIdProvider {private Properties properties;public VendorDatabaseIdProvider() {}public String getDatabaseId(DataSource dataSource) {if (dataSource == null) {throw new NullPointerException("dataSource cannot be null");} else {try {return this.getDatabaseName(dataSource);} catch (Exception var3) {VendorDatabaseIdProvider.LogHolder.log.error("Could not get a databaseId from dataSource", var3);return null;}}}public void setProperties(Properties p) {this.properties = p;}private String getDatabaseName(DataSource dataSource) throws SQLException {String productName = this.getDatabaseProductName(dataSource);if (this.properties != null) {Iterator var3 = this.properties.entrySet().iterator();Entry property;do {if (!var3.hasNext()) {return null;}property = (Entry)var3.next();} while(!productName.contains((String)property.getKey()));return (String)property.getValue();} else {return productName;}}private String getDatabaseProductName(DataSource dataSource) throws SQLException {Connection con = dataSource.getConnection();Throwable var3 = null;String var5;try {DatabaseMetaData metaData = con.getMetaData();var5 = metaData.getDatabaseProductName();} catch (Throwable var14) {var3 = var14;throw var14;} finally {if (con != null) {if (var3 != null) {try {con.close();} catch (Throwable var13) {var3.addSuppressed(var13);}} else {con.close();}}}return var5;}private static class LogHolder {private static final Log log = LogFactory.getLog(VendorDatabaseIdProvider.class);private LogHolder() {}}
}

       看下不同数据库的驱动getDatabaseProductName()返回什么:

         默认实现就是根据数据库的getDatabaseProductName去匹配在mybatis配置文件中配置的属性,然后选出databaseId。

        至此,我们分析完了databaseIdprovider的源码,下面接着看一下databaseId是如何发挥作用的。mybatis在解析映射文件的时候,首先会根据mybatis的配置文件解析出当前数据库对应的databaseId,然后根据mybatis的映射文件判断配置了databaseId属性的语句是否和接卸出的databaseId匹配,核心代码如下: 

// XMLStatementBuilder类中
private boolean databaseIdMatchesCurrent(String id, String databaseId, String requiredDatabaseId) {if (requiredDatabaseId != null) {return requiredDatabaseId.equals(databaseId);} else if (databaseId != null) {return false;} else {id = this.builderAssistant.applyCurrentNamespace(id, false);if (!this.configuration.hasStatement(id, false)) {return true;} else {MappedStatement previous = this.configuration.getMappedStatement(id, false);return previous.getDatabaseId() == null;}}}

四、自定义

这里的自定义只是针对databaseIdProvider的自定义,对映射文件中的databaseId不能自定义也没自定义的必要。

通过上述的分析,可以看出,如果要自定义databaseIdProvider只要实现接口DatabaseIdProvider便可。然后在mybatis的配置文件中将databaseIdProvider的type置为自定义的实现便可。

自定义的databaseIdProvider为:

public class MyDatavbaseIdProvider implements DatabaseIdProvider {Properties props = null;@Overridepublic void setProperties(Properties p) {//p代表mybatis中针对databaseIdProvider配置的属性props = p;}@Overridepublic String getDatabaseId(DataSource dataSource) throws SQLException {//根据使用的数据源,返回不同的databaseId。这里没有给出具体实现return null;}
}

使用自定义的databaseIdProvider,则mybatis配置文件中的配置需要做相应的调整。

<databaseIdProvider type="xxx.MyDatavbaseIdProvider"><property name="DB2" value="db2" /><property name="Oracle" value="oracle" /><property name="Adaptive Server Enterprise" value="sybase" /><property name="MySQL" value="mysql" />
</databaseIdProvider>

或者直接用通过xml给VendorDatabaseIdProvider注入:

http://www.dtcms.com/a/475916.html

相关文章:

  • asp网站上传重庆网站建设leco tec
  • 要服务网站建设wordpress顶部加横幅
  • 自己怎么做返利网站什么软件做美食视频网站
  • 企业品牌网站建设怎么做宁夏省住房城乡建设厅网站
  • 分享影视资源的网站怎么做青岛网站建设选圣城
  • 雄安微网站开发北京做网站设计公司
  • 营销网站建设企业asp网站安全怎么做
  • 建设银行网站官网h5设计是什么
  • 个人网站维护费用百度知道怎么赚钱
  • 官方网站如何建设网址导航建站
  • 手机网站制作公司报价wordpress 社交网站吗
  • 自己买服务器能在wordpress建网站国外校园网站建设分析
  • 网站建设 参照 标准规范网页制作与设计先学什么
  • 免费中英文网站模板重庆设计有限公司
  • 青岛网站建设推广公司哪家好莆田网站建设费用
  • 做经营行网站需要什么wordpress 设置网站目录
  • 云南凡科建站哪家好免费2级域名注册
  • 网站设计师介绍wordpress如何做301跳转
  • html在线编辑网站备案 网站建设计划书
  • 王也头像高清帅气seo短视频网页入口引流推广
  • 做网站的整体风格确定方式产品设计招聘网站
  • 做网站备案照片的要求如何用凡科网建立一个网站
  • 做网站公司北京缙云网站建设
  • 怎么样自己创建网站怎么做网站外推
  • 网站打不开了什么原因企业网站建设457
  • Kotlin 异步数据流三剑客:Flow、Channel、StateFlow 深度解析
  • 公司网站怎么修改著名的网站制作公司
  • asp.net 建立网站吗软件开发包含网站开发吗
  • 英文网站cms企业形象宣传策划方案
  • 桥南做网站上海网站快速优化排名