mybatis-redis項目分析

redis作為現在最優秀的key-value數據庫,非常適合提供項目的緩存服務。把redis作為mybatis的查詢緩存也是很常見的做法。在網上發現N多人是自己做的Cache,其實在mybatis的git下有一個子項目mybatis-redis;這個項目提供了redis作為mybatis查詢緩存的一個實現,下面先分析一下這個項目的實現原理,再提出幾個項目的問題:

代碼實現

該項目和大家普遍實現Mybatis的緩存方案大同小異,無非是實現Cache接口,并使用jedis操作緩存;不過該項目在設計細節上有一些區別;下面簡要分析一下源碼:

public final class RedisCache implements Cache {  
public RedisCache(final String id) {
  if (id == null) {
  throw new IllegalArgumentException("Cache instances require an ID");
}
this.id = id;
RedisConfig redisConfig = RedisConfigurationBuilder.getInstance().parseConfiguration();
pool = new JedisPool(redisConfig, redisConfig.getHost(), redisConfig.getPort(),
        redisConfig.getConnectionTimeout(), redisConfig.getSoTimeout(), redisConfig.getPassword(),
        redisConfig.getDatabase(), redisConfig.getClientName());
}

RedisCache在mybatis啟動的時候,由MyBatis的CacheBuilder創建,創建的方式很簡單,就是調用RedisCache的帶有String參數的構造方法,即RedisCache(String id);而在RedisCache的構造方法中,調用了RedisConfigurationBuilder來創建RedisConfig對象,并使用RedisConfig來創建JedisPool。
RedisConfig類繼承了JedisPoolConfig,并提供了host,port等屬性的包裝,簡單看一下RedisConfig的屬性:

public class RedisConfig extends JedisPoolConfig {
private String host = Protocol.DEFAULT_HOST;
private int port = Protocol.DEFAULT_PORT;
private int connectionTimeout = Protocol.DEFAULT_TIMEOUT;
private int soTimeout = Protocol.DEFAULT_TIMEOUT;
private String password;
private int database = Protocol.DEFAULT_DATABASE;
private String clientName;
}

RedisConfig對象是由RedisConfigurationBuilder創建的,簡單看下這個類的主要方法:

public RedisConfig parseConfiguration(ClassLoader classLoader) {
    Properties config = new Properties();
    InputStream input = classLoader.getResourceAsStream(redisPropertiesFilename);
    if (input != null) {
        try {
            config.load(input);
        } catch (IOException e) {
            throw new RuntimeException(
                    "An error occurred while reading classpath property '"
                            + redisPropertiesFilename
                            + "', see nested exceptions", e);
        } finally {
            try {
                input.close();
            } catch (IOException e) {
                // close quietly
            }
        }
    }
    RedisConfig jedisConfig = new RedisConfig();
    setConfigProperties(config, jedisConfig);
    return jedisConfig;
}

核心的方法就是parseConfiguration方法,該方法從classpath中讀取一個redis.properties文件:

host=localhost
port=6379
connectionTimeout=5000
soTimeout=5000
password=
database=0
clientName=

并將該配置文件中的內容設置到RedisConfig對象中,并返回;
接下來,就是RedisCache使用RedisConfig類創建完成JedisPool;
在RedisCache中實現了一個簡單的模板方法,用來操作Redis:

private Object execute(RedisCallback callback) {
Jedis jedis = pool.getResource();
try {
  return callback.doWithRedis(jedis);
} finally {
  jedis.close();
}
}

模板接口為RedisCallback,這個接口中就只需要實現了一個doWithRedis方法而已:

public interface RedisCallback {
    Object doWithRedis(Jedis jedis);
}

接下來看看Cache中最重要的兩個方法:putObject和getObject,通過這兩個方法來查看mybatis-redis儲存數據的格式:

@Override
public void putObject(final Object key, final Object value) {
execute(new RedisCallback() {
  @Override
  public Object doWithRedis(Jedis jedis) {
    jedis.hset(id.toString().getBytes(), key.toString().getBytes(), SerializeUtil.serialize(value));
    return null;
  }
});
}
@Override
public Object getObject(final Object key) {
return execute(new RedisCallback() {
  @Override
  public Object doWithRedis(Jedis jedis) {
    return SerializeUtil.unserialize(jedis.hget(id.toString().getBytes(), key.toString().getBytes()));
  }
});
}

可以很清楚的看到,mybatis-redis在存儲數據的時候,是使用的hash結構,把cache的id作為這個hash的key(cache的id在mybatis中就是mapper的namespace);這個mapper中的查詢緩存數據作為hash的field,需要緩存的內容直接使用SerializeUtil存儲,SerializeUtil和其他的序列化類差不多,負責對象的序列化和反序列化;

使用方式

整個mybatis-redis的關鍵代碼就這些,通過對代碼的分析,我們很容易得到mybatis-redis的使用方式:
1,在項目中添加一個redis.properties配置;
2,直接在mapper中使用<cache type="org.mybatis.caches.redis.RedisCache" />即可;

分析

通過代碼,我們可以看到在實際應用當中可能存在的一些問題:

1.默認情況下,mybatis會為每一個mapper創建一個RedisCache,而JedisPool是在RedisCache的構造方法中創建的,這就意味著會為每一個mapper創建一個JedisPool,使用意圖和開銷上面都會有問題;
2.在很多情況下,我們的應用中也會獨立使用到redis,這樣也無法讓RedisCache直接使用我們項目中可能已經存在的JedisPool;并且會造成兩個配置文件(除非我們應用也使用redis.properties);
3.RedisCache是使用hash來緩存一個Mapper中的查詢,所以我們只能通過mybatis的cache配置來控制對象的生存時間,空閑時間等屬性;而無法獨立的去配置每一個緩存區域(即每一個hash);

來源:
http://bbs.520it.com/forum.php?mod=viewthread&tid=286&extra=page%3D4

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,936評論 6 535
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,744評論 3 421
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,879評論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,181評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,935評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,325評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,384評論 3 443
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,534評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,084評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,892評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,067評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,623評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,322評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,735評論 0 27
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,990評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,800評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,084評論 2 375

推薦閱讀更多精彩內容