mybatis特殊符号处理
在 mybatis 中的 xml 文件中,存在一些特殊的符号,比如:<、>、"、&、<>等,正常书写 mybatis 会报错,需要对这些符号进行转义。具体转义如下所示:
特殊字符 转义字符
< <
> >
" "
’ '
& &
除此之外,还可以使用印射文件特殊处理,因为sql印射文件时xml类型文件,在转义上面有些符号无法读取例如<>&,我们可以使用<![CDATA[]]>来包裹特殊字符。例如
select * from student
where id <![CDATA[ > ]]>#{id}
传到mysql中的语句就是,问好就是预编译传过来的数据
select * from student
where id > ?
mybatis一二级缓存
缓存
数据缓存 让数据离我们执行程序更近,让程序可以快速获得数据
缓存(cache)的作用是为了减去数据库的压力,提高查询性能。缓存实现的原理是从数据库中查询出来的对象在使用完后不要销毁,而是存储在内存(缓存)中,当再次需要获取该对象时,直接从内存(缓存)中直接获取,不再向数据库执行 select 语句,从而减少了对数据库的查询次数,因此提高了数据库的性能。
一级缓存
在同一个sqlsession中,第一次执行一个查询语句,会将查询到的数据加载到java程序中,在第二次执行此查询语句时,会先查询缓存里面是否有此数据,如果有就不会在向mysql里面查询,如果没有才会继续向mysql里面发送请求,缓解了mysql了压力。
运行结果
Id为1的sql执行了一次第二次执行的时候先在缓存里面查找里面有数据就不会在向后端查询,并且id为1的major对象的地址是一样的就说明是从缓存中获得的。值得注意的是增删改会提交事务,提交事务后同一个sqlsession下查询语句也会执行两次,getsqlsession.clearCache();调用这个方法会清除sqlsession的缓存。
二级缓存
二级缓存是 SqlSessionFactory 级别的,根据 mapper 的 namespace 划分区域的,相同 namespace 的 mapper 查询的数据缓存在同一个区域,如果使用 mapper 代理方法每个 mapper 的 namespace 都不同,此时可以理解为二级缓存区域是根据 mapper 划分。每次查询会先从缓存区域查找,如果找不到则从数据库查询,并将查询到数据写入缓存。Mybatis 内部存储缓存使用一个 HashMap,key 为hashCode+sqlId+Sql 语句。value 为从查询出来映射生成的 java 对象。sqlSession 执行 insert、update、delete 等操作 commit 提交后会清空缓存区域,防止脏读。
运行结果
运行结果,打印台打印的结果不一样的原因是major开启了序列化,虽然对象地址发生变化,但是对象的数据是一样的。
开启二级缓存
在myatis中默认开启的是一级缓存,二级缓存需要自己设置开启
在mybatis.xml全局变量设置中开启二级缓存
<setting name="cacheEnabled" value="true"/>
需要缓存数据的java类进行序列化
在相应的印射文件中加入
<cache></cache>标签开启二级缓存
Cache标签相关api设置
java反射机制
反射是java设计的灵魂
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
要想解剖一个类,必须先要获取到该类的字节码文件对象。而解剖使用的就是Class类中的方法.所以先要获取到每一个字节码文件对应的Class类型的对象.
反射就是把java各个部分印射成一个个的java对象
例如:一个car类,有成员变量,方法,构造方法等,
public class car {
private String name;
private String color;
@Override
public String toString() {
return "car{" +
"name='" + name + '\'' +
", color='" + color + '\'' +
'}';
}
public car() {
System.out.println("无参构造方法");
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getColor() {
return color;
}
public void setColor(String color) {
this.color = color;
}
public void run(){
System.out.println("跑");
}
}
利用反射机制可以将car中的各个部分印射成一个个对象
获取javaClass类的方法
//常规创建对象
car a =new car();
a.run();
//利用java反射机制创建Class对象
String s="com.ffyc.javaResult.car";
//方式1通过Class类的静态方法forname获得Class对象
Class aClass = Class.forName(s);
System.out.println(aClass);
//方式2知道类名直接调用静态方法class,所有类都继承object类
Class<car> carClass = car.class;
System.out.println(aClass==carClass);
//方式3知道对象直接调用getclass得到class对象
car a1=new car();
Class<? extends car> aClass1 = a1.getClass();
System.out.println(aClass==aClass1);
利用Constructor对象创建对象
//利用java反射机制创建Class对象
String s="com.ffyc.javaResult.car";
Class<?> aClass = Class.forName(s);
//方式1利用Class对象创建car对象
Object o = aClass.newInstance();
System.out.println(o);
//方式2 利用Constructor对象创建car对象
Constructor constructor = aClass.getConstructor();//不包含私有的
Object o1 = constructor.newInstance();
System.out.println(o1);
// java反射机制也可以对私有的方法和变量进行操作aClass.getDeclaredConstructor();可以对私有的也进行操作
// 一般不建议操作私有成员,打破的封装性
通过Field对象得到对象的属性
//利用java反射机制创建Class对象
String s="com.ffyc.javaResult.car";
Class<?> aClass = Class.forName(s);
Object o = aClass.newInstance();
Field[] fields = aClass.getDeclaredFields();//获得所有成员变量包括私有属性
HashMap hashMap=new HashMap();
hashMap.put("name", "bm");
hashMap.put("color","red");
for (Field f:fields){
f.setAccessible(true);//允许访问私有变量
f.set(o,hashMap.get(f.getName()));
}
System.out.println(o);
通过Method对象得到对象的方法
//利用java反射机制创建Class对象
String s="com.ffyc.javaResult.car";
Class<?> aClass = Class.forName(s);
Object o = aClass.newInstance();
Method run = aClass.getMethod("run");
run.invoke(o);//将car类中的run方法放进o这个对象中,这段代码相当于执行了run这个方法
//利用java反射机制创建Class对象
String s="com.ffyc.javaResult.car";
Class<?> aClass = Class.forName(s);
HashMap hashMap=new HashMap();
hashMap.put("name", "bm");
hashMap.put("color","red");
Object o = aClass.newInstance();
Field[] declaredFields = aClass.getDeclaredFields();
for (Field field:declaredFields){
String m="set"+field.getName().substring(0,1).toUpperCase()+field.getName().substring(1);//获得标准的set方法
System.out.println(m);
Method method = aClass.getMethod(m, field.getType());
method.invoke(o,hashMap.get(field.getName()));//调用执行方法
}
System.out.println(o);