Mybatis操作多數(shù)據(jù)源實(shí)現(xiàn)的方法

mybatis操作多數(shù)據(jù)源實(shí)現(xiàn)的方法

本文講解"mybatis操作多數(shù)據(jù)源實(shí)現(xiàn)的方法",希望能夠解決相關(guān)問(wèn)題。

 

現(xiàn)在有一個(gè)mysql數(shù)據(jù)源和一個(gè)postgresql數(shù)據(jù)源,使用mybatis對(duì)兩個(gè)數(shù)據(jù)源進(jìn)行操作:

1. 注入多數(shù)據(jù)源

可以對(duì)兩個(gè)數(shù)據(jù)源分別實(shí)現(xiàn)其service層和mapper層,以及mybatis的配置類:

@configuration
// 這里需要配置掃描包路徑,以及sqlsessiontemplateref
@mapperscan(basepackages = "com.example.mybatisdemo.mapper.mysql", sqlsessiontemplateref = "mysqlsqlsessiontemplate")
public class mysqlmybatisconfigurer {
    /**
     * 注入mysql數(shù)據(jù)源
     */
    @bean
    @configurationproperties(prefix = "spring.datasource.mysql")
    public datasource mysqldatasource() {
        return new druiddatasource();
    }
    /**
     * 注入mysqlsqlsessionfactory
     */
    @bean
    public sqlsessionfactory mysqlsqlsessionfactory(datasource mysqldatasource) throws exception {
        sqlsessionfactorybean factorybean = new sqlsessionfactorybean();
        factorybean.setdatasource(mysqldatasource);
        // 設(shè)置對(duì)應(yīng)的mapper文件
        factorybean.setmapperlocations(new pathmatchingresourcepatternresolver().getresources("classpath:" +
                "/mappers/mysqlmapper.xml"));
        return factorybean.getobject();
    }
    /**
     * 注入mysqlsqlsessiontemplate
     */
    @bean
    public sqlsessiontemplate mysqlsqlsessiontemplate(sqlsessionfactory mysqlsqlsessionfactory) {
        return new sqlsessiontemplate(mysqlsqlsessionfactory);
    }
    /**
     * 注入mysqltransactionalmanager
     */
    @bean
    public datasourcetransactionmanager mysqltransactionalmanager(datasource mysqldatasource) {
        return new datasourcetransactionmanager(mysqldatasource);
    }
}
@configuration
// 這里需要配置掃描包路徑,以及sqlsessiontemplateref
@mapperscan(basepackages = "com.example.mybatisdemo.mapper.postgresql", sqlsessiontemplateref = "postgresqlsqlsessiontemplate")
public class postgresqlmybatisconfigurer {
    /**
     * 注入postgresql數(shù)據(jù)源
     */
    @bean
    @configurationproperties(prefix = "spring.datasource.postgresql")
    public datasource postgresqldatasource() {
        return new druiddatasource();
    }
    /**
     * 注入postgresqlsqlsessionfactory
     */
    @bean
    public sqlsessionfactory postgresqlsqlsessionfactory(datasource postgresqldatasource) throws exception {
        sqlsessionfactorybean factorybean = new sqlsessionfactorybean();
        factorybean.setdatasource(postgresqldatasource);
        // 設(shè)置對(duì)應(yīng)的mapper文件
        factorybean.setmapperlocations(new pathmatchingresourcepatternresolver().getresources("classpath:" +
                "/mappers/postgresqlmapper.xml"));
        return factorybean.getobject();
    }
    /**
     * 注入postgresqlsqlsessiontemplate
     */
    @bean
    public sqlsessiontemplate postgresqlsqlsessiontemplate(sqlsessionfactory postgresqlsqlsessionfactory) {
        return new sqlsessiontemplate(postgresqlsqlsessionfactory);
    }
    /**
     * 注入postgresqltransactionalmanager
     */
    @bean
    public datasourcetransactionmanager postgresqltransactionalmanager(datasource postgresqldatasource) {
        return new datasourcetransactionmanager(postgresqldatasource);
    }
}

在配置類中,分別注入了一個(gè)事務(wù)管理器transactionmanager,這個(gè)和事務(wù)管理是相關(guān)的。在使用@transactional注解時(shí),需要配置其value屬性指定對(duì)應(yīng)的事務(wù)管理器。

2. 動(dòng)態(tài)數(shù)據(jù)源

spring中提供了abstractroutingdatasource抽象類,可以用于動(dòng)態(tài)地選擇數(shù)據(jù)源。

public abstract class abstractroutingdatasource extends abstractdatasource implements initializingbean {
    @nullable
    private map targetdatasources;
    @nullable
    private object defaulttargetdatasource;
    private boolean lenientfallback = true;
    private datasourcelookup datasourcelookup = new jndidatasourcelookup();
    @nullable
    private map resolveddatasources;
    @nullable
    private datasource resolveddefaultdatasource;
    // 略
}

通過(guò)源碼可以看到,該抽象類實(shí)現(xiàn)了initializingbean接口,并在其afterpropertiesset方法中將數(shù)據(jù)源以的形式放入一個(gè)map中。

public void afterpropertiesset() {
    if (this.targetdatasources == null) {
        throw new illegalargumentexception("property 'targetdatasources' is required");
    } else {
        this.resolveddatasources = collectionutils.newhashmap(this.targetdatasources.size());
        this.targetdatasources.foreach((key, value) -> {
            object lookupkey = this.resolvespecifiedlookupkey(key);
            datasource datasource = this.resolvespecifieddatasource(value);
            // 將數(shù)據(jù)源以的形式放入map中
            this.resolveddatasources.put(lookupkey, datasource);
        });
        if (this.defaulttargetdatasource != null) {
            this.resolveddefaultdatasource = this.resolvespecifieddatasource(this.defaulttargetdatasource);
        }
    }
}

該類中還有一個(gè)determinetargetdatasource方法,是根據(jù)lookupkey從map中獲取對(duì)應(yīng)的數(shù)據(jù)源,如果沒(méi)有獲取到,則使用默認(rèn)的數(shù)據(jù)源。

protected datasource determinetargetdatasource() {
    assert.notnull(this.resolveddatasources, "datasource router not initialized");
    object lookupkey = this.determinecurrentlookupkey();
    // 根據(jù)lookupkey從map中獲取對(duì)應(yīng)的數(shù)據(jù)源
    datasource datasource = (datasource)this.resolveddatasources.get(lookupkey);
    if (datasource == null && (this.lenientfallback || lookupkey == null)) {
        datasource = this.resolveddefaultdatasource;
    }
    if (datasource == null) {
        throw new illegalstateexception("cannot determine target datasource for lookup key [" + lookupkey + "]");
    } else {
        return datasource;
    }
}

lookupkey是通過(guò)determinetargetdatasource方法獲取到的,而它是一個(gè)抽象方法,我們要做的就是通過(guò)實(shí)現(xiàn)這個(gè)方法,來(lái)控制獲取到的數(shù)據(jù)源。

@nullable
protected abstract object determinecurrentlookupkey();

(1) 創(chuàng)建并注入動(dòng)態(tài)數(shù)據(jù)源

創(chuàng)建abstractroutingdatasource的子類,實(shí)現(xiàn)determinecurrentlookupkey方法

public class routingdatasource extends abstractroutingdatasource {
    @override
    protected object determinecurrentlookupkey() {
        return datasourcecontextholder.get();
    }
}

這里的datasourcecontextholder是一個(gè)操作threadlocal對(duì)象的工具類

public class datasourcecontextholder {
    /**
     * 數(shù)據(jù)源上下文
     */
    private static final threadlocal contextholder = new threadlocal<>();
    /**
     * 設(shè)置數(shù)據(jù)源類型
     */
    public static void set(datasourcetype type) {
        contextholder.set(type);
    }
    /**
     * 獲取數(shù)據(jù)源類型
     *
     * @return datasourcetype
     */
    public static datasourcetype get() {
        return contextholder.get();
    }
    /**
     * 使用mysql數(shù)據(jù)源
     */
    public static void mysql() {
        set(datasourcetype.mysql);
    }
    /**
     * 使用postgresql數(shù)據(jù)源
     */
    public static void postgresql() {
        set(datasourcetype.postgresql);
    }
    public static void remove() {
        contextholder.remove();
    }
}

通過(guò)調(diào)用datasourcecontextholder.mysql()或者datasourcecontextholder.postgresql()就能修改contextholder的值,從而在動(dòng)態(tài)數(shù)據(jù)源的determinetargetdatasource方法中就能獲取到對(duì)應(yīng)的數(shù)據(jù)源。

在數(shù)據(jù)源配置類中,將mysql和postgresql的數(shù)據(jù)源設(shè)置到動(dòng)態(tài)數(shù)據(jù)源的map中,并注入容器。

@configuration
public class datasourceconfigurer {
    @bean
    @configurationproperties(prefix = "spring.datasource.mysql")
    public datasource mysqldatasource() {
        return new druiddatasource();
    }
    @bean
    @configurationproperties(prefix = "spring.datasource.postgresql")
    public datasource postgresqldatasource() {
        return new druiddatasource();
    }
    @bean
    public routingdatasource routingdatasource(datasource mysqldatasource, datasource postgresqldatasource) {
        map datasources = new hashmap<>();
        datasources.put(datasourcetype.mysql, mysqldatasource);
        datasources.put(datasourcetype.postgresql, postgresqldatasource);
        routingdatasource routingdatasource = new routingdatasource();
        routingdatasource.setdefaulttargetdatasource(mysqldatasource);
        // 設(shè)置數(shù)據(jù)源
        routingdatasource.settargetdatasources(datasources);
        return routingdatasource;
    }
}

(2) mybatis配置類

由于使用了動(dòng)態(tài)數(shù)據(jù)源,所以只需要編寫(xiě)一個(gè)配置類即可。

@configuration
@mapperscan(basepackages = "com.example.mybatisdemo.mapper", sqlsessiontemplateref = "sqlsessiontemplate")
public class mybatisconfigurer {
    // 注入動(dòng)態(tài)數(shù)據(jù)源
    @resource
    private routingdatasource routingdatasource;
    @bean
    public sqlsessionfactory sqlsessionfactory() throws exception {
        sqlsessionfactorybean sqlsessionfactorybean = new sqlsessionfactorybean();
        sqlsessionfactorybean.setdatasource(routingdatasource);
        // 這里可以直接設(shè)置所有的mapper.xml文件
        sqlsessionfactorybean.setmapperlocations(new pathmatchingresourcepatternresolver().getresources("classpath" +
                ":mappers/*.xml"));
        return sqlsessionfactorybean.getobject();
    }
    @bean
    public sqlsessiontemplate sqlsessiontemplate(sqlsessionfactory sqlsessionfactory) {
        return new sqlsessiontemplate(sqlsessionfactory);
    }
    @bean
    public datasourcetransactionmanager transactionalmanager(datasource mysqldatasource) {
        return new datasourcetransactionmanager(mysqldatasource);
    }
}

(3) 使用注解簡(jiǎn)化數(shù)據(jù)源切換

我們雖然可以使用datasourcecontextholder類中的方法進(jìn)行動(dòng)態(tài)數(shù)據(jù)源切換,但是這種方式有些繁瑣,不夠優(yōu)雅。可以考慮使用注解的形式簡(jiǎn)化數(shù)據(jù)源切換。
我們先定義兩個(gè)注解,表示使用mysql數(shù)據(jù)源或postgresql數(shù)據(jù)源:

@target({elementtype.type, elementtype.method})
@retention(retentionpolicy.runtime)
public @interface mysql {
}
@target({elementtype.type, elementtype.method})
@retention(retentionpolicy.runtime)
public @interface postgresql {
}

再定義一個(gè)切面,當(dāng)使用了注解時(shí),會(huì)先調(diào)用切換數(shù)據(jù)源的方法,再執(zhí)行后續(xù)邏輯。

@component
@aspect
public class datasourceaspect {
    @pointcut("@within(com.example.mybatisdemo.aop.mysql) || @annotation(com.example.mybatisdemo.aop.mysql)")
    public void mysqlpointcut() {
    }
    @pointcut("@within(com.example.mybatisdemo.aop.postgresql) || @annotation(com.example.mybatisdemo.aop.postgresql)")
    public void postgresqlpointcut() {
    }
    @before("mysqlpointcut()")
    public void mysql() {
        datasourcecontextholder.mysql();
    }
    @before("postgresqlpointcut()")
    public void postgresql() {
        datasourcecontextholder.postgresql();
    }
}

在使用動(dòng)態(tài)數(shù)據(jù)源的事務(wù)操作時(shí)有兩個(gè)需要注意的問(wèn)題:

問(wèn)題一    同一個(gè)事務(wù)操作兩個(gè)數(shù)據(jù)源

mybatis使用executor執(zhí)行sql時(shí)需要獲取連接,baseexecutor類中的getconnection方法調(diào)用了springmanagedtransaction中的getconnection方法,這里優(yōu)先從connection字段獲取連接,如果connection為空,才會(huì)調(diào)用openconnection方法,并把連接賦給connection字段。

也就是說(shuō),如果你使用的是同一個(gè)事務(wù)來(lái)操作兩個(gè)數(shù)據(jù)源,那拿到的都是同一個(gè)連接,會(huì)導(dǎo)致數(shù)據(jù)源切換失敗。

protected connection getconnection(log statementlog) throws sqlexception {
    connection connection = this.transaction.getconnection();
    return statementlog.isdebugenabled() ? connectionlogger.newinstance(connection, statementlog, this.querystack) : connection;
}
public connection getconnection() throws sqlexception {
    if (this.connection == null) {
        this.openconnection();
    }
    return this.connection;
}
private void openconnection() throws sqlexception {
    this.connection = datasourceutils.getconnection(this.datasource);
    this.autocommit = this.connection.getautocommit();
    this.isconnectiontransactional = datasourceutils.isconnectiontransactional(this.connection, this.datasource);
    logger.debug(() -> {
        return "jdbc connection [" + this.connection + "] will" + (this.isconnectiontransactional ? " " : " not ") + "be managed by spring";
    });
}

問(wèn)題二      兩個(gè)獨(dú)立事務(wù)分別操作兩個(gè)數(shù)據(jù)源

(1) 在開(kāi)啟事務(wù)的時(shí)候,datasourcetransactionmanager中的dobegin方法會(huì)先獲取connection,并保存到connectionholder中,將數(shù)據(jù)源和connectionholder的對(duì)應(yīng)關(guān)系綁定到transactionsynchronizationmanager中。

protected void dobegin(object transaction, transactiondefinition definition) {
    datasourcetransactionmanager.datasourcetransactionobject txobject = (datasourcetransactionmanager.datasourcetransactionobject)transaction;
    connection con = null;
    try {
        if (!txobject.hasconnectionholder() || txobject.getconnectionholder().issynchronizedwithtransaction()) {
            // 獲取連接
            connection newcon = this.obtaindatasource().getconnection();
            if (this.logger.isdebugenabled()) {
                this.logger.debug("acquired connection [" + newcon + "] for jdbc transaction");
            }
            // 保存到connectionholder中
            txobject.setconnectionholder(new connectionholder(newcon), true);
        }
        txobject.getconnectionholder().setsynchronizedwithtransaction(true);
        // 從connectionholder獲取連接
        con = txobject.getconnectionholder().getconnection();
        // 略
        // 將數(shù)據(jù)源和connectionholder的關(guān)系綁定到transactionsynchronizationmanager中
        if (txobject.isnewconnectionholder()) {
            transactionsynchronizationmanager.bindresource(this.obtaindatasource(), txobject.getconnectionholder());
        }
         // 略
}

(2) transactionsynchronizationmanager的bindresource方法將數(shù)據(jù)源和connectionholder的對(duì)應(yīng)關(guān)系存入線程變量resources中。

public abstract class transactionsynchronizationmanager {
    // 線程變量
    private static final threadlocal                
相關(guān)文章
亚洲国产精品第一区二区,久久免费视频77,99V久久综合狠狠综合久久,国产免费久久九九免费视频