java-Mybatis框架02

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.rhkb.cn/news/410792.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

OpenCV绘图函数(6)绘制椭圆函数ellipse()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 画出一个简单的或粗的椭圆弧或者填充一个椭圆扇形。 函数 cv::ellipse 使用更多的参数可以画出椭圆轮廓、填充的椭圆、椭圆弧或填充的椭圆扇形。…

ISO 26262中的失效率计算:SN 29500-3 Expected values for discrete semiconductors

目录 概要 1 基准条件下的失效率 2 失效率转换 2.1 失效率预测模型 2.2 电压应力系数 2.2.1 电压应力系数计算模型 2.2.2 电压应力系数计算 2.3 温度应力系数 2.3.1 温度应力系数计算模型 2.3.2 温度应力系数计算 2.4 漂移灵敏度系数 3 任务剖面应力系数 4 早期失…

【STM32】时钟体系

一、时钟体系 为什么需要时钟&#xff1f; 时钟可以为系统提供精确的定时&#xff0c;比如时间显示&#xff0c;定时器&#xff0c;pwm… 为芯片各功能模块提供工作势能,使能各组管脚工作&#xff0c;如果不使能&#xff0c;管脚无法工作 同步数据传输 给单片机提供一个时…

【使用 Python 进行截图的两种方法】

在 Python 中&#xff0c;可以使用 pyautogui 和 Pillow 进行截图 使用 pyautogui 进行截图时&#xff0c;其提供了方便的函数。例如&#xff0c;使用 pyautogui.screenshot() 函数可以获取整个屏幕的截图&#xff0c;该函数返回一个包含屏幕截图的图像对象。如果不想截取整个…

齐护【百度AI对话】编程系统文心一言大语音模型对话ESP32图形化Mixly编程Scratch编程Arduino

齐护【百度AI对话】编程系统 一、前言 ​ 在这个日新月异的时代&#xff0c;AI的触角已延伸至互联网、金融、医疗、教育等每一个角落&#xff0c;其影响力不容忽视。从日常中的智能推荐到医疗前沿的精准诊断&#xff0c;从定制化教育到智能化的投资策略&#xff0c;AI正以前所…

Xilinx高速接口之GTP

简介 开坑计划中&#xff0c;主要参考ug482 主要讲解结构以及原语 以及时钟路由和一些其他的 GTP_COMMON还好&#xff0c;需要设置的不多&#xff0c;原语也短&#xff0c; GTP_CHANNEL需要设置的东西真多&#xff0c;原语也长 还好有官方参考例程以及自动生成的原语例化 不然…

PDF文件的读取与合并:使用PyPDF2与ReportLab

目录 一、PyPDF2库基础 1.1 PyPDF2简介 1.2 安装PyPDF2 1.3 读取PDF内容 1.4 合并PDF文件 二、ReportLab库基础 2.1 ReportLab简介 2.2 安装ReportLab 2.3 使用ReportLab生成PDF文本 2.4 ReportLab 与 PyPDF2 结合使用 三、注意事项与最佳实践 3.1 文本提取的局限性…

【Linux】初步识操作系统

linux专栏&#xff1a;《Linux入门系列》 系列文章&#xff1a;gdb-调试器初入门&#xff08;简单版使用&#xff09; 编辑器vim入门&#xff08;概念模式转换技巧&#xff09; 目录 1. 概念 2. 设计操作系统的目的 3. 定位&#xff1a;操作系统负责管理 4. 如何理解管理 …

科研绘图系列:R语言单细胞差异基因四分图(Quad plot)

介绍 在单细胞分析领域,为了探究不同分组间同一细胞类型的基因表达差异,研究者们常采用四分图(Quad Plot)作为分析工具。该图形的横轴代表比较组1,而纵轴代表比较组2。通过这种布局,四分图能够有效地展示两组间共有的差异表达基因,从而为深入理解细胞类型在不同条件下的…

Android 突破边界:解密google Partner机制获取Resource

在 Android 应用开发中&#xff0c;除了可以查找系统中的特定 APK 并获取其信息外&#xff0c;还可以通过 Partner 机制获取 Partner APK 的资源文件&#xff0c;同时这种机制也是一种跨进程的通信方式。本文将进一步探讨这些内容。 1.Partner apk注册特定的action广播 /** M…

优雅实现远程调用-OpenFeign

目录 OpenFeign介绍 OpenFeign最佳实践 我再来遇到个问题&#xff0c;我创建的com.xx.xxx包&#xff0c;放到一起了&#xff0c;不是那种一个在一个下面的那种 Nacos&#xff0c;只要看见这种什么网络报错啥的&#xff0c;java.net.SocketException: Network is unreachabl…

Vatee万腾平台:打造企业智能化转型的坚实后盾

在当今这个日新月异的数字化时代&#xff0c;企业智能化转型已成为不可逆转的趋势。面对激烈的市场竞争和快速变化的市场需求&#xff0c;如何高效、稳定地完成这一转型&#xff0c;成为众多企业亟需解决的问题。而Vatee万腾平台&#xff0c;正是这样一位引领企业智能化转型的坚…

MATLAB进阶:应用微积分

今天我们继续学习matlab中的应用微积分 求导&#xff08;微分&#xff09; 1、数值微分 n维向量x(xi&#xff0c;x,… x)的差分定义为n-1维向量△x(X2-X1&#xff0c;X3-X2&#xff0c;…&#xff0c;Xn- Xn-1)。 diff(x) 如果x是向量&#xff0c;返回向量x的差分如果x是矩…

《机器学习》—— OpenCV 对图片的各种操作(均值、方框、高斯、中值滤波处理)

文章目录 1、对有椒盐噪声的图片进行均值、方框、高斯、中值滤波处理2、给图像边缘增加边框3、对图片进行阈值化操作 1、对有椒盐噪声的图片进行均值、方框、高斯、中值滤波处理 均值滤波 cv2.blur是 OpenCV 库中的一个函数&#xff0c;用于对图像进行均值模糊处理。这个函数通…

webserver.h详解

webserver.h头文件 成员变量&#xff1a; 目的是提供一个完整的Web服务器框架&#xff0c;包括网络监听、事件处理、数据库连接管理、线程池管理等功能。 #ifndef WEBSERVER_H #define WEBSERVER_H#include <sys/socket.h> #include <netinet/in.h> #include <…

算法学习-基础数据结构

基础数据结构 一.栈 1.普通栈 套路&#xff1a;从前往后遍历 需要考虑相邻元素 有消除操作 栈。 2.单调栈 二.队列 1.普通队列 2.优先队列 三.Trie 使用场景&#xff1a;可以求某个字符串在众多字符串中出现的次数&#xff0c;以某个字符串为前缀出现的次数 Trie中…

一. 初始 Spring Boot

一. 初始 Spring Boot 文章目录 一. 初始 Spring Boot1. Spring Boot 是什么&#xff1f;2. Spring Boot 官方文档地址3. 第一个 Spring Boot 程序3.1 我的环境工具配置 4. 第一个 SpringBoot 程序解释说明5. Spring&#xff0c;SpringMVC&#xff0c; SpringBoot 三者的关系6.…

Linux中的常见命令——用户管理命令

1、useradd添加新用户 基本语法 语法功能描述useradd 用户名添加新用户useradd -g 组名 用户名添加新用户到某个组 实操案例 1、添加一个新用户【此时的用户是没有密码的】 [rootcentos100 ~]# cd /home [rootcentos100 home]# ls www zss [rootcentos100 home]# useradd…

设置虚拟机使用主机以太网而不是WiF连接

虚拟机使用主机的以太网连接而不是Wi-Fi连接&#xff0c;可以通过在虚拟化软件中配置虚拟机的网络设置来实现。以下是一些常见的虚拟化软件&#xff08;如VMware和VirtualBox&#xff09;中设置虚拟机网络以使用以太网连接的步骤&#xff1a; 一、VMware中设置 1、打开虚拟网…

查询系统操作指南

在年会等大型活动中&#xff0c;快速准确地查询桌号和座位号对于参与者来说至关重要。本文将详细介绍如何使用查询系统来实现这一目的。步骤一&#xff1a;电脑端信息上传 1. 访问官网&#xff1a;打开云分组的官方网站。 2. 登录账户&#xff1a;使用微信扫码的方式进行登录。…