MyBatis之动态SQL语句和Java本地缓存如何设计

我们在进行项目开发时,经常遇到需要根据不同的需求,对原有SQL语句的内容进行修改,原来这是一个比较头疼的问题,因为需要对原有SQL语句进行拼接、重组,费时费力还容易出错,今天我们将学习MyBatis的动态SQL功能,可以解决这个问题。

动态SQL语句简介

动态SQL语句是MyBatis的一个非常强大的功能,允许我们根据不同的需求,组合出不同的SQL语句,在select、update、insert、delete标签中都可以添加动态SQL语句。

IF标签

IF标签可以在条件成立时,在SQL语句中插入IF标签中的内容,不成立就不插入

示例:

select * from tb_user where

u_realname=#{realname}

and u_gender=#{gender}

[计算机软件著作权登记申请表分类号,app申请软件著作权多少钱](http://daduchang.net/446910.html) 上面代码中是按照User对象的各个属性进行查询,如果姓名不为空就插入姓名作为条件,如果性别不为空就插入性别作为条件。

但是我们会发现如果姓名为空,性别不为空,SQL语句就会变成:select * from tb_user where and u_gender=#{gender}

或者姓名、性别都为空就会变成:select * from tb_user where

这样都会出现语法错误,这样我们就需要下一个标签:where

Where标签

Where标签 用于配置where条件,会去掉多余的and或or,如果一个条件都不成立,会自动去掉sql中的where。
app软件著作权怎么申请流程,软件著作权可以在哪里申请
代码改为:

select * from tb_user

u_realname=#{realname}

and u_gender=#{gender}

and u_age=#{age}

申请软件著作权证书查询,软件著作权自己申请流程

这样就不会出现上面可能的错误了

Trim标签

上面where标签的功能,也可以使用trim标签实现,trim能去掉多余的前缀和后缀。

语法:

<if标签>

示例:

Set标签

用于配置update语句中的set部分,可以自动添加set关键字,删除多余的逗号
游戏软件著作权申请材料,申请软件著作权登记查询
示例:

update tb_user

u_name=#{name},

u_password=#{password},

where u_id=#{id}

也可以使用trim实现:

u_name=#{name},
申请计算机软件著作权要求,软件著作权申请能补正几次添加链接描述

Foreach标签

用于循环遍历集合或数组的值,如:按多个人姓名查询

select * from tb_user where u_realname in (‘张三’,‘李四’…)

语法:

#{变量名}

注意:集合名在Mapper接口中需要使用@Param注解配置

示例:

select * from tb_user where u_realname in

#{name}

Choose标签

Java中有if,也会有if-else、switch语句来判断多个条件,MyBatis也有Choose标签可以进行多条件判断

示例:

<select id=“findActiveBlogLike”

resultType=“Blog”>

SELECT * FROM BLOG WHERE state = ‘ACTIVE’

AND title like #{title}
软件著作权免申请费,技术开发软件双方共有软件著作权如何申请

AND author_name like #{author}

AND featured = 1

这里从第一个when进行判断,如果成立就插入条件,结束choose标签,如果不成立再判断下一个when,如果所有when都不成立,就插入otherwise中的语句。

总结

动态SQL语句是MyBatis非常重要的特性,能够让我们根据不同需求组合SQL语句,而且不容易出错。动态SQL的标签有:if、where、set、choose、foreach等。掌握好它们我们可以写出更加灵活、高效的数据库操作代码。
最近在看Mybatis的源码,刚好看到缓存这一块,Mybatis提供了一级缓存和二级缓存;一级缓存相对来说比较简单,功能比较齐全的是二级缓存,基本上满足了一个缓存该有的功能;当然如果拿来和专门的缓存框架如ehcache来对比可能稍有差距;本文带大家来整理一下实现一个本地缓存都应该需要考虑哪些东西。

考虑点

考虑点主要在数据用何种方式存储,能存储多少数据,多余的数据如何处理等几个点,下面我们来详细的介绍每个考虑点,以及该如何去实现;

1.数据结构

首要考虑的就是数据该如何存储,用什么数据结构存储,最简单的就直接用Map来存储数据;或者复杂的如redis一样提供了多种数据类型哈希,列表,集合,有序集合等,底层使用了双端链表,压缩列表,集合,跳跃表等数据结构;

2.对象上限

因为是本地缓存,内存有上限,所以一般都会指定缓存对象的数量比如1024,当达到某个上限后需要有某种策略去删除多余的数据;
软件著作权代理申请公司,软件著作权申请需多长时间添加链接描述
3.清除策略

上面说到当达到对象上限之后需要有清除策略,常见的比如有LRU(最近最少使用)、FIFO(先进先出)、LFU(最近最不常用)、SOFT(软引用)、WEAK(弱引用)等策略;

4.过期时间

除了使用清除策略,一般本地缓存也会有一个过期时间设置,比如redis可以给每个key设置一个过期时间,这样当达到过期时间之后直接删除,采用清除策略+过期时间双重保证;

5.线程安全

像redis是直接使用单线程处理,所以就不存在线程安全问题;而我们现在提供的本地缓存往往是可以多个线程同时访问的,所以线程安全是不容忽视的问题;并且线程安全问题是不应该抛给使用者去保证;

6.简明的接口

提供一个傻瓜式的对外接口是很有必要的,对使用者来说使用此缓存不是一种负担而是一种享受;提供常用的get,put,remove,clear,getSize方法即可;

7.是否持久化

这个其实不是必须的,是否需要将缓存数据持久化看需求;本地缓存如ehcache是支持持久化的,而guava是没有持久化功能的;分布式缓存如redis是有持久化功能的,memcached是没有持久化功能的;

8.阻塞机制

在看Mybatis源码的时候,二级缓存提供了一个blocking标识,表示当在缓存中找不到元素时,它设置对缓存键的锁定;这样其他线程将等待此元素被填充,而不是命中数据库;其实我们使用缓存的目的就是因为被缓存的数据生成比较费时,比如调用对外的接口,查询数据库,计算量很大的结果等等;这时候如果多个线程同时调用get方法获取的结果都为null,每个线程都去执行一遍费时的计算,其实也是对资源的浪费;比较好的办法是只有一个线程去执行,其他线程等待,计算一次就够了;但是此功能基本上都交给使用者来处理,很少有本地缓存有这种功能;

如何实现

以上大致介绍了实现一个本地缓存我们都有哪些需要考虑的地方,当然可能还有其他没有考虑到的点;下面继续看看关于每个点都应该如何去实现,重点介绍一下思路;

1.数据结构

本地缓存最常见的是直接使用Map来存储,比如guava使用ConcurrentHashMap,ehcache也是用了ConcurrentHashMap,Mybatis二级缓存使用HashMap来存储:
软件著作权申请了一定能通过么,程序代码申请软件著作权添加链接描述
Map<Object, Object> cache = new ConcurrentHashMap<Object, Object>()

Mybatis使用HashMap本身是非线程安全的,所以可以看到起内部使用了一个SynchronizedCache用来包装,保证线程的安全性;

当然除了使用Map来存储,可能还使用其他数据结构来存储,比如redis使用了双端链表,压缩列表,整数集合,跳跃表和字典;当然这主要是因为redis对外提供的接口很丰富除了哈希还有列表,集合,有序集合等功能;

2.对象上限

本地缓存常见的一个属性,一般缓存都会有一个默认值比如1024,在用户没有指定的情况下默认指定;当缓存的数据达到指定最大值时,需要有相关策略从缓存中清除多余的数据这就涉及到下面要介绍的清除策略;

3.清除策略

配合对象上限之后使用,场景的清除策略如:LRU(最近最少使用)、FIFO(先进先出)、LFU(最近最不常用)、SOFT(软引用)、WEAK(弱引用);

LRU:Least Recently Used的缩写最近最少使用,移除最长时间不被使用的对象;常见的使用LinkedHashMap来实现,也是很多本地缓存默认使用的策略;

FIFO:先进先出,按对象进入缓存的顺序来移除它们;常见使用队列Queue来实现;

LFU:Least Frequently Used的缩写大概也是最近最少使用的意思,和LRU有点像;区别点在LRU的淘汰规则是基于访问时间,而LFU是基于访问次数的;可以通过HashMap并且记录访问次数来实现;

SOFT:软引用基于垃圾回收器状态和软引用规则移除对象;常见使用SoftReference来实现;

WEAK:弱引用更积极地基于垃圾收集器状态和弱引用规则移除对象;常见使用WeakReference来实现;

4.过期时间

设置过期时间,让缓存数据在指定时间过后自动删除;常见的过期数据删除策略有两种方式:被动删除和主动删除;

被动删除:每次进行get/put操作的时候都会检查一下当前key是否已经过期,如果过期则删除,类似如下代码:

if (System.currentTimeMillis() - lastClear > clearInterval) {

clear();

}

主动删除:专门有一个job在后台定期去检查数据是否过期,如果过期则删除,这其实可以有效的处理冷数据;

5.线程安全

尽量用线程安全的类去存储数据,比如使用ConcurrentHashMap代替HashMap;或者提供相应的同步处理类,比如Mybatis提供了SynchronizedCache:

public synchronized void putObject(Object key, Object object) {

…省略…

}
软件著作权申请书范文,北京如何申请软件著作权证
@Override

public synchronized Object getObject(Object key) {

…省略…

}

6.简明的接口

提供常用的get,put,remove,clear,getSize方法即可,比如Mybatis的Cache接口:

public interface Cache {

String getId();

void putObject(Object key, Object value);

Object getObject(Object key);

Object removeObject(Object key);

void clear();

int getSize();

ReadWriteLock getReadWriteLock();

}

再来看看guava提供的Cache接口,相对来说也是比较简洁的:

public interface Cache<K, V> {
软件著作权申请规定, 软件著作权只能一个软件申请一个吗
V getIfPresent(@CompatibleWith(“K”) Object key);

V get(K key, Callable<? extends V> loader) throws ExecutionException;

ImmutableMap<K, V> getAllPresent(Iterable<?> keys);

void put(K key, V value);

void putAll(Map<? extends K, ? extends V> m);

void invalidate(@CompatibleWith(“K”) Object key);

void invalidateAll(Iterable<?> keys);

void invalidateAll();

long size();

CacheStats stats();

ConcurrentMap<K, V> asMap();

void cleanUp();

}

7.是否持久化

持久化的好处是重启之后可以再次加载文件中的数据,这样就起到类似热加载的功效;比如ehcache提供了是否持久化磁盘缓存的功能,将缓存数据存放在一个.data文件中;

diskPersistent=“false” //是否持久化磁盘缓存

redis更是将持久化功能发挥到极致,慢慢的有点像数据库了;提供了AOF和RDB两种持久化方式;当然很多情况下可以配合使用两种方式;

8.阻塞机制

除了在Mybatis中看到了BlockingCache来实现此功能,之前在看<<java并发编程实战>>的时候其中有实现一个很完美的缓存,大致代码如下:

public class Memoizerl<A, V> implements Computable<A, V> {

private final Map<A, Future> cache = new ConcurrentHashMap<A, Future>();

private final Computable<A, V> c;

public Memoizerl(Computable<A, V> c) {

this.c = c;

}

@Override

public V compute(A arg) throws InterruptedException, ExecutionException {

while (true) {福建软件著作权申请,武汉软件著作权申请

Future f = cache.get(arg);

if (f == null) {

Callable eval = new Callable() {

@Override

public V call() throws Exception {

return c.compute(arg);

}

};

FutureTask ft = new FutureTask(eval);

f = cache.putIfAbsent(arg, ft);

if (f == null) {

f = ft;

ft.run();

}

try {

return f.get();

} catch (CancellationException e) {

cache.remove(arg, f);

}

}

}

}

}
软件著作权申请天津,软件著作权申请天津添加链接描述
compute是一个计算很费时的方法,所以这里把计算的结果缓存起来,但是有个问题就是如果两个线程同时进入此方法中怎么保证只计算一次,这里最核心的地方在于使用了ConcurrentHashMap的putIfAbsent方法,http://daduchang.net/446883.html同时只会写入一个FutureTask;

总结:要设计一个本地缓存都需要考虑哪些点:数据结构,对象上限,清除策略,过期时间,线程安全,阻塞机制,实用的接口,是否持久化;当然肯定有其他考虑点,欢迎补充。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值