java-Mybatis框架02

news/2024/9/17 15:01:48/ 标签: java, mybatis, 开发语言

1.#{} ${}区别

 #{} 是占位符,是采用编译方式向sql中传值,可以防止sql注入,如果往sql中传值,使用#{}${} 是将内容直接拼接到sql语句中,一般不用于向sql中传值,一般用于向sql中动态传递列名。
 区别:
1.底层实现不同
#{} 采用预编译方式 可以防止sql注入 更加安全${} 采用字符串拼接,直接将值拼接到sql中2.使用场景不同#{} 一般用于向sql中的列传值
${} 一般用于sql动态传递列名  例如,排序时order by  后面的列名是可以改变的。例如,select后面的列名也可以自由选择

2.特殊处理定义 resultMap

(1)定义resultMap

<resultMap id="studentMap" type="Student"><id column="id" property="id"></id><result column="num" property="num"></result><result column="name" property="name"></result><result column="gender" property="gender"></result>
</resultMap>
1. resutlMap 的 id 属性是 resutlMap 的唯一标识,本例中定义为 “studentMap”
2. resutlMap 的 type 属性是映射的 POJO 类型
3. id 标签映射主键,result 标签映射非主键
4. property 设置对象属性名称,column 映射查询结果的列名称
(2)使用resultMap
 <select id="findStudentFindById" resultMap="studentMap">select
s.id,
s.num,
s.name,
s.gender,
m.name mnamefrom student s inner join major m on s.majorid=m.id where s.id=#{id}</select>
1. 本例的输出映射使用的是 resultMap,而非 resultType
2. resultMap 引用了 studentMap

3.多表关联处理结果集

resultMap 元素中 association , collection 元素.
association 一个复杂类型的关联;许多结果将包装成这种类型
  • 嵌套结果映射 – 关联可以是 resultMap 元素,或是对其它结果映射的引用
Collection  一个复杂类型的集合( 关联元素处理一对多关联
  • 嵌套结果映射 – 集合可以是 resultMap 元素,或是对其它结果映射的引用
关联查询方式
1.关联查询方式一:直接多表关联查询获得学生及专业的信息
2.关联查询方式二:先查询主表(学生表),再通过学生表中的majorid与专业表关联,查询专业表
,得到Major对象,最后将得到的Major对象封装到学生对象中。

4.动态SQL

对SQL语句进行逻辑判断

1.if

<select id="teachers" resultType="Teacher">select * from teacher<where><if test="num!=null">num= #{num}</if><if test="name!=null">and   name= #{name}</if><if test="gender!=null">and     gender= #{gender}</if></where>

语法:<if test=""></if>

注意:test属性成立,执行if标签体,不成立就不执行

2.where

<select id="teachers" resultType="Teacher">select * from teacher<where><if test="num!=null">num= #{num}</if><if test="name!=null">and   name= #{name}</if><if test="gender!=null">and     gender= #{gender}</if></where>
</select>

语法:<where></where>

注意:当where标签中的if语句有条件成立时, 就会动态的添加where 关键字, 还可以动态删除where 紧跟着的关键字,例如 and or

3.trim

<select id="teachers" resultType="Teacher">select * from teacher<trim prefix="where" prefixOverrides="and| or"><if test="num!=null">num= #{num}</if><if test="name!=null">or  name= #{name}</if><if test="gender!=null">and     gender= #{gender}</if></trim>
</select>

语法<trim prefix="where" prefixOverrides=""> </trim>

作用: trim 当判断条件成立时,可以自定义前缀关键字和后缀关键字

prefix="where"

prefixOverrides="and|or" 覆盖指定关键字

4.choose when otherwise

<select id="teachers" resultType="Teacher">select * from teacher<trim prefix="where" prefixOverrides="and| or"><choose><when test="name!=null">name=#{name}</when><otherwise>name='李老师'</otherwise></choose></trim>
</select>

语法:

<choose> 

 <when test="判断条件">

   执行的sql语句

    </when>

<otherwise> 

执行其他的sql语句

</otherwise>

</choose>

作用:MyBatis 提供了 choose 元素,它有点像 Java 中的 switch 语句

5.set

可以把最后一个逗号去掉
<update id="updateTeacher" parameterType="Teacher">
update teacher
<set><if test="num!=null">num=#{num},</if><if test="name!=null">name=#{name},</if><if test="gender!=null">gender=#{gender}</if>
</set>
where id=#{id}</update>

也可以用trim实现

<update id="updateTeacher" parameterType="Teacher">update teacher<trim prefix="set" suffixOverrides=","><if test="num!=null">num=#{num},</if><if test="name!=null">name=#{name},</if><if test="gender!=null">gender=#{gender}</if></trim>where id=#{id}
</update>

6.foreach

主要用在构建 in 条件中,它可以在 SQL 语句中进行迭代一个集合。foreach 元素的属性主要有 item,index,collection,open,separator,close。
item 表示集合中每一个元素进行迭代时的别名。
index 指定一个名字,用于 表示在迭代过程中,每次迭代到的位置。
open 表示该语句以什么开始。
separator 表示在每次进行迭代之间以什么符号作为分隔符。
close 表示以什 么结束。
  在使用 foreach 的时候最关键的也是最容易出错的就是 collection 属性,该属性是必须指定的,但是在不同情况下,该属性的值是不一样的。
如果传入的是单参数且参数类型是一个 List 的时候,collection 属 性值为 list
如果传入的是单参数且参数类型是一个 array 数组的时候, collection 的属性值为 array
例如,可以用foreach来批量删除教师表的信息,用Integer[]数组来存放id,foreach循坏拿到id,
放入数组中

array:


<delete id="deleteTeacher">

delete from teacher where id in

<foreach item="id" collection="array" open="(" separator="," close=")">

#{id} </foreach>

</delete>

List:


<select id="findTeacher" resultType="Teacher">

select

<foreach item="col" collection="list" separator=",">

${col}

</foreach>

from teacher

</select>

5.特殊符号处理

mybatis 中的 xml 文件中,存在一些特殊的符号,比如:<、>、"、&、<> 等,正常书写 mybatis 会报错,需要对这些符号进行转义。具体转义如下所示:
特殊字符 转义字符
<      &lt
>     &gt
"      &quot
’      &apos
&     &amp
除了可以使用上述转义字符外,还可以使用<![CDATA[]]>来包裹特殊字符。如 下所示:
<if test="id != null">
AND <![CDATA[ id <> #{id} ]]>
</if>
<![CDATA[ ]]>是 XML 语法。在 CDATA 内部的所有内容都会被解析器忽略。
问题:那就是 <if> </if> <where> </where> <choose> </choose> <trim> </trim> 等这些标签都不会被解析,所以 我们只把有特殊字符的语句放在 <![CDATA[ ]]> 尽量缩小<![CDATA[ ]]> 的范围。

6.mybatis 一级缓存二级缓存

缓存作用:是为了减去数据库的压力,提高查询性能

缓存原理:是从数据库中查询出来的对象在使用完后不要销毁,而是存储在内存(缓存) 中,当再次需要获取该对象时,直接从内存(缓存)中直接获取,不再向数据库 执行 select 语句,从而减少了对数据库的查询次数,因此提高了数据库的性能。(先从缓存中查询数据,缓存中如果没有,去数据库查询,查询到后把数据放到缓存中,下次直接从缓存中获取

1.mybatis一级缓存

(1)定义:在参数和 SQL 完全一样的情况下,我们使用同一个 SqlSession 对象调 用一个 Mapper 方法,往往只执行一次 SQL,因为使用 SelSession 第一次查询 后,MyBatis 会将其放在缓存中,以后再查询的时候,如果没有声明需要刷新, 并且缓存没有超时的情况下,SqlSession 都会取出当前缓存的数据,而不会再 次发送 SQL 到数据库。(默认是SqlSession级别的,在同一个SqlSession中查询到数据先缓存到SqlSession对象中 第二次查询数据时,先从SqlSession中查询,如果有直接返回,没有,再去查询数据库

(2)一级缓存生命周期:
开始于SqlSession创建

结束于SqlSession的关闭
如果执行期间新增,修改,删除操作,也会清空当前SqlSession对象中缓存数据。调用sqlSession.clearCache(),强制清空一切缓存数据。

2.mybatis二级缓存

(1)定义:二级缓存是 SqlSessionFactory 级别的,根据 mapper 的 namespace 划分区域的,相同 namespace 的 mapper 查询的数据缓存在同一个区域,如果使 用 mapper 代理方法每个 mapper 的 namespace 都不同,此时可以理解为二 级缓存区域是根据 mapper 划分。 每次查询会先从缓存区域查找,如果找不到则从数据库查询,并将查询到数 据写入缓存。Mybatis 内部存储缓存使用一个 HashMap,key 为 hashCode+sqlId+Sql 语句。value 为从查询出来映射生成的 java 对象。

作用: 二级缓存是SqlSessionFactory级别的,可以让多个SqlSession共享数据

注意

* mybatis默认没有开启二级缓存,使用时需要配置开启。

* 如果开启二级缓存,当SqlSession关闭时,会将一级缓存数据存储到二级缓存中 , 其他的SqlSession就会从二级缓存中查询到之前的SqlSession数据。

sqlSession 执行 insert、update、delete 等操作 commit 提交后会清空缓存区 域,防止脏读。
二级缓存机制设计及工作模式如下图所示:

(2)配置二级缓存设置

<1>启用二级缓存

在mybaits.xml文件中配置全局二级缓存开关

如下代码所示,当 cacheEnabled 设置为 true 时启用二级缓存,设置为 false 时禁用二级缓存。

<setting name="cacheEnabled" value="true"/>
<2>对象序列化
将所有的 POJO 类实现序列化接口 Java.io. Serializable(这里我用的是Teacher类)
<3>配置映射文件
在 TeacherMapper 映射文件中添加<cache />/<cache></cache>,表示此 mapper 开启二级缓存。
当 SqlSeesion 关闭时,会将数据存入到二级缓存.
注意:可以在使用的方法上配置是否开启二级缓存 使用useCache=""属性可以来开启或关闭二级缓存

3.mybatis架构

7.Java反射

mybatis中,给了什么类型,mybatis就可以将结果封装映射到给定的类的对象中
<select id="findAdminByld" parameterType="int" resultType="Admin">
<select id="teachers" resultType="com.ffyc.mybatispro.model.Teacher">

框架需要只写一套程序,就可以处理给其的任意类,

框架是如何做到写一套程序,就可以处理任意类了?

以前已知类名的使用方式,可以看做是正向使用类.

框架需要对任意类处理时,只是知道类的名字,通过类的名字动态才去获取类中的信息.

把这种对类的使用方式,称为反向使用.

1.java反射概念及作用

Java反射:是在运行状态中,对于任意一个类,都能够知道这个 类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个 方法和属性;这种动态获取信息以及动态调用对象的方法的功能称为 java语言的反射机制。(在运行状态中在仅知道一个类名时,就可以动态获得类中信息,创建对象,调用对象成员的机制称为java反射机制)
作用  动态获取类的信息。

2.java反射相关API

2.1.Class类(java.lang.Class)

(1)概念:Class类表示当前正在运行中的类和接口

注意;一旦class文件被加载到内存,就会为其创建一个Class对象。任何类被 使用时都会创建一个Class对象;

Class类是Java反射机制的基础,通过Class类,可以得到一个类的基本 信息。

(2)获得Class类的方法

Object类中的getClass方法:适用于通过对象获得Class实例的情况
类名.class方式:适用于通过类名获得Class实例的情况
Class类的静态方法 forName(String name)
(3)java创建对象的四种方式
1.new  +对象名
2.反序列化
3.反射机制  获取到对象的包名和类名
4.对象克隆

2.2 Constructor类

Constructor是获取对象的构造方法的类

获取Constructor的实例:

1.通过Class实例获得,Class类中定义了如下方法

Constructor<T> getConstructor(Class... parameterTypes)
通过指定参数类型,返回构造方法

(1)getConstructor()  获得指定的公共构造方法

(2).getConstructors();//获得所有公共的构造方法

(3).getDeclaredConstructor();//获得类中任意的构造方法(包含私有)

(4)getDeclaredConstructors();//获得类中所有的构造方法(包含私有)

2.Constructor类可以通过getXXX方法获得构造方法的基本信息.

getName:返回构造方法的名字
除了获得构造方法的基本信息,还可以创建实例
newInstance(Object... initargs) :创建实例

2.3 Field类

Field类作用是获取到对象的成员变量

1.获得Field类实例:

 public Field getField(String name)通过指定Field名字,返回 Field实例.   注意Field的访问权限

(1)getField("成员变量");//获得指定名称的公共的成员变量

(2)getFields() 获得所有的公共成员变量

(3)getDeclaredField( "成员变量t");//获得指定名称的成员变量,包含私有的

(4)getDeclaredFields( );//获得所有的成员变量,包含私有的

2.Field类的作用

  Field类将类的属性进行封装,可以获得属性的基本信息、属性的值,也
可以对属性进行赋值.
  getName:返回属性的名字
  set:设置属性值

2.4  Method类

1.获取Method类的实例:

Method getMethod(String name, Class... parameterTypes) : 通过指定方法名,参数类型,返回一个Method实例
2.Method类作用
1.  Method类将类中的方法进行封装,可以动态获得方法的信息,例如
● getName:获得方法名字
● getParameterTypes:获得方法参数类型
2.除了动态获得方法信息外,Method还能动态调用某一个对象的具体方法
● invoke(Object obj, Object... args) :使用obj调用该方法,参数为args

8.反射案例

1.演示通过属性的get和set方法,对类中的私有属性进行取值,操作。
User类:
public class Test3 {public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {//使用反射机制时,只知道类的名称(包名+类名)String classname="com.ffyc.javaReflect.User";/* String classname="com.ffyc.javaReflect.Car";*///        1.通过类名获得到类的Class对象Class clazz= Class.forName(classname);//        2.通过类的Class对象,创建对象Object object=clazz.newInstance()/*模拟从数据库拿到的数据*/HashMap<String,String> map=new HashMap<>();map.put("account", "admin");map.put("password", "111");Field[] fields=clazz.getDeclaredFields();//获得类中所有成员变量for(Field field:fields){field.setAccessible(true);//允许访问操作私有的属性field.set(object,map.get(field.getName()) );}System.out.println(object);}
}

2.自定义java对象转json工具类

类同上

public class JsonUtil {public  static String objectToJson(Object object) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {// 获取到类的Class对象Class clazz=object.getClass();
String json="{";
//获取类中所有属性Field[] declaredFields =clazz.getDeclaredFields();for(Field field:declaredFields){
//生成属性get方法名字String getmethod = "get" + field.getName().substring(0, 1).toUpperCase() + field.getName().substring(1);//获得方法对象Method method=clazz.getMethod(getmethod);
//            调用方法String value=(String)method.invoke(object);
//      把属性名称和和值拼接成键值json+=field.getName()+":"+value+",";}json=json.substring(0,json.length()-1);
json+="}";return json;}public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {User user=new User();user.setAccount("admin");user.setPassword("111");Car car=new Car();car.setName("bbb");car.setColor("red");System.out.println(JsonUtil.objectToJson(user));System.out.println(JsonUtil.objectToJson(car));}
}

9.反射机制的优缺点

优点:

 1. 增加程序的灵活性,可以在运行的过程中动态对类进行修改和操作
 2.提高代码的复用率,比如动态代理
 3.可以在运行时轻松获取任意一个类的方法、属性,并且还能通过反射进行动态调用
缺点:
 1. 反射会涉及到动态类型的解析,导致性能要比非反射调用更低
 2. 使用反射技术通常要在一个没有安全限制的程序运行 .
 3. 反射可以绕过一些限制访问的属性或者方法,可能会导致破坏代码本身的抽象 性

http://www.ppmy.cn/news/1517059.html

相关文章

代理IP的来源、工作原理与应用

在数字化时代&#xff0c;代理IP已经成为网络活动中不可或缺的一部分&#xff0c;尤其在跨境电商、网络营销、网络爬虫等领域&#xff0c;代理IP发挥着至关重要的作用。本文将深入探讨代理IP的来源、工作原理、应用以及潜在风险&#xff0c;帮助读者更好地理解并合理利用这一网…

Git下载安装配置

Git的下载与安装 Git是一种分布式版本控制系统&#xff0c;用于跟踪文件和文件夹的变化。它最初由Linus Torvalds开发&#xff0c;用于管理Linux内核的源代码。Git的设计目标是&#xff1a;速度快、开发效率高、数据完整性和可靠性强。 Git通过创建一个存储库&#xff08;rep…

(一) 初入MySQL 【认识和部署】

前置资源 一、数据库概述 1.1、数据库基本概念 数据(Data) 描述事物的符号记录称为数据。数字、文字、图形、图像、声音、档案记录等都是数据。数据是以“记录”的形式按照统一的格式进行存储的&#xff0c;而不是杂乱无章的。 相同格式和类型的数据统一存放在一起&#xff0…

Python实现RSA加解密算法

目录 深入了解RSA加密算法一、RSA算法概述1.1 关键步骤1.2 安全性分析 二、RSA算法的Python实现2.1 辅助函数2.2 密钥生成2.3 加密与解密2.4 使用示例 三、总结 深入了解RSA加密算法 RSA&#xff08;Rivest-Shamir-Adleman&#xff09;是一种非对称加密算法&#xff0c;由Ron …

如何快速熟悉一个软件

当入职一家新公司时&#xff0c;必然要熟悉一些新的软件项目&#xff0c;这个时候&#xff0c;如何才能快速熟悉软件项目&#xff0c;以达到快速上手的目的。 1大忌 还记得自己刚毕业的第一份工作&#xff0c;一上来就去看代码&#xff0c;看代码的细节。在看代码的时候&…

[C++]set和map的介绍及使用

关于set和map的接口函数部分&#xff0c;只重点介绍一些相较于别的容器有特殊地方的接口&#xff0c;set和map的接口可以触类旁通。 一、概念 &#xff08;一&#xff09;、关联式容器 关联式容器存储的元素是一个个的键值对<key,value>。通过键&#xff08;key&#x…

MATLAB 生成指定范围、角度、厚度的含噪平面点云(77)

模拟生成点云并可视化显示,可以验证算法有效性,尤其是针对验证算法的某方面 MATLAB 生成指定范围、角度、厚度的含噪平面点云(77) 一、算法介绍二、使用步骤1.代码2.效果一、算法介绍 如题,模拟生成一组平面点云,含有噪声点,确定算法稳定性,可以指定生成平面的范围,厚…

Java集合—Map系列集合(习题一)

文章目录 Java集合—Map集合&#xff08;习题&#xff09;1.使用泛型修改根据学员姓名找学员对象2.运用Map的三种遍历方式进行遍历迭代器遍历键值对遍历增强遍历 综合要求 Java集合—Map集合&#xff08;习题&#xff09; 1.使用泛型修改根据学员姓名找学员对象 2.运用Map的三…

【JAVA]DAY 2在网页中输出日期和时间,实时还是静止?

一、如何输出日期文本 使用document.write(Date()); 会在网页中输出当前的日期和时间。在 2024 年 8 月 28 日星期三执行这段代码&#xff0c;可能会输出类似 “Wed Aug 28 2024 [具体时间]” 这样的内容。 Date()是 JavaScript 中的一个内置对象&#xff0c;用于处理日期和…

UE5 多个类选择界面生成

在Unreal Engine 5 (UE5) 中&#xff0c;如果你想要创建一个可以选择多个类的界面&#xff0c;你可以使用SClassPicker小部件。以下是一个简单的例子&#xff0c;展示如何在UE5的编辑器模块中创建一个自定义的编辑器工具栏按钮&#xff0c;并打开一个类选择器。 #include &quo…

论文写作遇到的问题——个人记录用

1.实验结果图绘制 python画图字体设置 Science Plots使用中中文配置的问题 11种 Matplotlib 科研论文图表教程 2.论文写作格式 word公式居中、编号右对齐、自动编号、交叉引用 mathtype操作合集&#xff0c;使用大全 arxiv.org的文章引用格式 LaTex的下载与安装&#x…

[CLIP-VIT-L + Qwen] 多模态大模型源码阅读 - DataSet篇

[CLIP-VIT-L Qwen] 多模态大模型源码阅读 - DataSet篇 前情提要源码解读完整代码逐行解读导包readjson函数data_collate函数ImageCaptionDataset类&#xff08;init函数&#xff09;ImageCaptionDataset类&#xff08;readImage函数&#xff09; 参考repo:WatchTower-Liu/VLM-…

Java中Objecy类

没有成员变量 也就只有无参 的构造方法 /*** ClassName Test* author gyf* Date 2024/8/28 10:32* Version V1.0* Description : */ public class Test {public static void main(String[] args) {// toString()Object object new Object();System.out.println(object);String…

网络安全新视角:人工智能在防御中的最新应用

人工智能在网络安全中的最新应用 概述 人工智能&#xff08;AI&#xff09;在网络安全领域的应用正日益成熟&#xff0c;它通过机器学习和深度学习技术&#xff0c;为网络安全带来了革命性的变革。AI技术不仅能够自动化、智能化地检测、分析和应对安全威胁&#xff0c;还能够…

Jenkins:自动化的魔法师,打造无缝CI/CD流水线

标题&#xff1a;“Jenkins&#xff1a;自动化的魔法师&#xff0c;打造无缝CI/CD流水线” 在当今快速发展的软件开发领域&#xff0c;持续集成&#xff08;Continuous Integration, CI&#xff09;和持续部署&#xff08;Continuous Deployment, CD&#xff09;已经成为提升开…

Docker续1:

一、打包传输 1.打包 [rootlocalhost ~]# systemctl start docker [rootlocalhost ~]# docker save -o centos.tar centos:latest [rootlocalhost ~]# ls anaconda-ks.cfg centos.tar 2.传输 [rootlocalhost ~]# scp centos.tar root192.168.1.100:/root 3.删除镜像 [r…

总结:Python语法

Python中的字典、列表和数组是三种常用的数据结构&#xff0c;它们各自有不同的用途和特性。 字典&#xff08;Dictionary&#xff09; 字典是一种无序的、可变的数据结构&#xff0c;它存储键值对&#xff08;key-value pairs&#xff09;。字典中的每个元素都是一个键值对&…

flink--会话模式与应用模式

flink-会话模式部署 会话情况&#xff1a; 添加依赖 <properties><flink.version>1.17.2</flink.version> </properties> ​ <dependencies><dependency><groupId>org.apache.flink</groupId><artifactId>flink-strea…

CSS属性

一、CSS列表样式 1、list-style-type属性&#xff08;列表项标记&#xff09; CSS列表属性允许我们设置不同的列表项标记。 在HTML中&#xff0c;有​两种类型​的列表&#xff1a; ​无序列表​&#xff08;<ul>&#xff09; - 列表项目用​项目符号​标记​有序列表…

【Linux】自动化构建工具makefile

目录 背景 makefile简单编写 .PHONY makefile中常用选项 makefile的自动推导 背景 会不会写makefile&#xff0c;从一个侧面说明了一个人是否具备完成大型工程的能力 ​ ◉ 一个工程中的源文件不计数&#xff0c;其按类型、功能、模块分别放在若干个目录中&#xff0c;mak…