目錄
一、引言
二、MyBatis動態(tài)SQL
2.1.if元素使用
2.2.foreach元素使用
三、MyBatis模糊查詢
①使用#{字段名}
②使用${字段名}
③使用concat{'%',#{字段名},'%'}
總結(jié)
四、MyBatis結(jié)果映射
4.1.案例演示
4.1.1.resultType進(jìn)行結(jié)果映射
4.1.2.resultMap進(jìn)行結(jié)果映射
一、引言
在當(dāng)今的軟件開發(fā)環(huán)境中,數(shù)據(jù)庫的使用已經(jīng)成為了一項(xiàng)基礎(chǔ)且必不可少的技能。而在處理數(shù)據(jù)庫相關(guān)的任務(wù)時,SQL查詢語句無疑是最為常用的工具之一。然而,隨著應(yīng)用程序的復(fù)雜性不斷增加,我們往往需要編寫更加復(fù)雜的SQL查詢語句以滿足需求。這就引出了我們今天要討論的主題——MyBatis。
在這篇文章中,我們將深入探討MyBatis的兩個重要特性:動態(tài)SQL和模糊查詢。動態(tài)SQL是MyBatis的一個強(qiáng)大功能,它允許我們在XML映射文件中編寫動態(tài)生成的SQL語句。模糊查詢則是我們在處理大量數(shù)據(jù)時常用的一種查詢方式,它可以幫助我們快速地找出符合特定條件的記錄。
此外,我們還將討論MyBatis的結(jié)果映射功能。結(jié)果映射是將數(shù)據(jù)庫查詢結(jié)果映射到Java對象的過程,它是MyBatis的核心功能之一。通過結(jié)果映射,我們可以將復(fù)雜的數(shù)據(jù)庫表結(jié)構(gòu)轉(zhuǎn)換為Java中的對象,使得程序的設(shè)計(jì)更加直觀和簡單。
掌握MyBatis的這些高級特性,不僅可以提高我們的開發(fā)效率,也可以讓我們在面試中脫穎而出。因此,無論你是已經(jīng)有一定經(jīng)驗(yàn)的開發(fā)者,還是剛剛?cè)腴T的學(xué)習(xí)者,都值得花時間去學(xué)習(xí)和理解這些知識。在接下來的文章中,我們將詳細(xì)介紹這些特性的使用方法和注意事項(xiàng),希望對你有所幫助。
二、MyBatis動態(tài)SQL
2.1.if元素使用
在以往的編寫中我們肯定寫過這樣的代碼
update t_mvc_book bname=?,btype=?,bprice=?,..... where bid=?
假如我們前臺jsp沒有傳參數(shù)bname到后臺,那么會怎么樣?
update t_mvc_book bname=null,btype='科幻小說',bprice=9.9,..... where bid=7
就會造成這樣的結(jié)果,原本我們不傳遞參數(shù)只是不想改變原有的字段值,沒成想不傳反而變?yōu)閚ull
有些人就會想那我傳個參數(shù)來行不行呢?行,但是沒必要。如果我們使用Mybatis里面的動態(tài)SQL,這個問題就迎刃而解了,它會把我們的sql語句變?yōu)閤ml的方式,通過if標(biāo)簽判斷,如果為空就不會拼接判斷條件里的內(nèi)容。
<update id="updateByPrimaryKeySelective" parameterType="com.csdn.xw.model.Book" >
update t_mvc_book
<set >
<if test="bname != null" >
bname = #{bname,jdbcType=VARCHAR},
</if>
<if test="price != null" >
price = #{price,jdbcType=REAL},
</if>
</set>
where bid = #{bid,jdbcType=INTEGER}
</update>
2.2.foreach元素使用
我們再來假設(shè)一個場景: 我們寫代碼的過程中肯定寫過對某個事物的批量刪除
可能是這樣遍歷數(shù)組然后調(diào)用刪除方法
for int id :ids
? ? ? orderItemsDao.delete(id)
也可能是拼接的方式將需要刪除的id做個拼接
delete from t_oa_order where id in(...)
如果是使用mybatis我們就可以這樣寫
<select id="selectByBids" resultType="com.csdn.xw.model.Book" parameterType="java.lang.Integer" >
select
<include refid="Base_Column_List" />
from t_mvc_book
where bid in
<foreach collection="bids" item="bid" open="(" close=")" separator=",">
#{bid}
</foreach>
</select>
現(xiàn)在來測試一下
BookMapper
List<Book> selectByBids(@Param("bids") List bids);
BookBiz
List<Book> selectByBids(List bids);
BookBizImpl
@Override
public List<Book> selectByBids(List bids) {
return bookMapper.selectByBids(bids);
}
demo測試類
@Test
public void selectByBids() {
System.out.println("測試的查詢的方法");
List<Integer> integers = Arrays.asList(new Integer[]{55, 56, 57, 58});
bookBiz.selectByBids(integers).forEach(System.out::println);
}
測試結(jié)果:
三、MyBatis模糊查詢
為了突出我們的效果展示,我們增加一個配置文件log4j2.xml
<?xml version="1.0" encoding="UTF-8"?>
<!-- status : 指定log4j本身的打印日志的級別.ALL< Trace < DEBUG < INFO < WARN < ERROR
< FATAL < OFF。 monitorInterval : 用于指定log4j自動重新配置的監(jiān)測間隔時間,單位是s,最小是5s. -->
<Configuration status="WARN" monitorInterval="30">
<Properties>
<!-- 配置日志文件輸出目錄 ${sys:user.home} -->
<Property name="LOG_HOME">/root/workspace/lucenedemo/logs</Property>
<property name="ERROR_LOG_FILE_NAME">/root/workspace/lucenedemo/logs/error</property>
<property name="WARN_LOG_FILE_NAME">/root/workspace/lucenedemo/logs/warn</property>
<property name="PATTERN">%d{yyyy-MM-dd HH:mm:ss.SSS} [%t-%L] %-5level %logger{36} - %msg%n</property>
</Properties>
<Appenders>
<!--這個輸出控制臺的配置 -->
<Console name="Console" target="SYSTEM_OUT">
<!-- 控制臺只輸出level及以上級別的信息(onMatch),其他的直接拒絕(onMismatch) -->
<ThresholdFilter level="trace" onMatch="ACCEPT"
onMismatch="DENY" />
<!-- 輸出日志的格式 -->
<!-- %d{yyyy-MM-dd HH:mm:ss, SSS} : 日志生產(chǎn)時間 %p : 日志輸出格式 %c : logger的名稱
%m : 日志內(nèi)容,即 logger.info("message") %n : 換行符 %C : Java類名 %L : 日志輸出所在行數(shù) %M
: 日志輸出所在方法名 hostName : 本地機(jī)器名 hostAddress : 本地ip地址 -->
<PatternLayout pattern="${PATTERN}" />
</Console>
<!--文件會打印出所有信息,這個log每次運(yùn)行程序會自動清空,由append屬性決定,這個也挺有用的,適合臨時測試用 -->
<!--append為TRUE表示消息增加到指定文件中,false表示消息覆蓋指定的文件內(nèi)容,默認(rèn)值是true -->
<File name="log" fileName="logs/test.log" append="false">
<PatternLayout
pattern="%d{yyyy-MM-dd HH:mm:ss.SSS} [%t] %-5level %logger{36} - %msg%n" />
</File>
<!-- 這個會打印出所有的info及以下級別的信息,每次大小超過size, 則這size大小的日志會自動存入按年份-月份建立的文件夾下面并進(jìn)行壓縮,作為存檔 -->
<RollingFile name="RollingFileInfo" fileName="${LOG_HOME}/info.log"
filePattern="${LOG_HOME}/$${date:yyyy-MM}/info-%d{yyyy-MM-dd}-%i.log">
<!--控制臺只輸出level及以上級別的信息(onMatch),其他的直接拒絕(onMismatch) -->
<ThresholdFilter level="info" onMatch="ACCEPT"
onMismatch="DENY" />
<PatternLayout
pattern="%d{yyyy-MM-dd HH:mm:ss.SSS} [%t] %-5level %logger{36} - %msg%n" />
<Policies>
<!-- 基于時間的滾動策略,interval屬性用來指定多久滾動一次,默認(rèn)是1 hour。 modulate=true用來調(diào)整時間:比如現(xiàn)在是早上3am,interval是4,那么第一次滾動是在4am,接著是8am,12am...而不是7am. -->
<!-- 關(guān)鍵點(diǎn)在于 filePattern后的日期格式,以及TimeBasedTriggeringPolicy的interval, 日期格式精確到哪一位,interval也精確到哪一個單位 -->
<!-- log4j2的按天分日志文件 : info-%d{yyyy-MM-dd}-%i.log -->
<TimeBasedTriggeringPolicy interval="1"
modulate="true" />
<!-- SizeBasedTriggeringPolicy:Policies子節(jié)點(diǎn), 基于指定文件大小的滾動策略,size屬性用來定義每個日志文件的大小. -->
<!-- <SizeBasedTriggeringPolicy size="2 kB" /> -->
</Policies>
</RollingFile>
<RollingFile name="RollingFileWarn" fileName="${WARN_LOG_FILE_NAME}/warn.log"
filePattern="${WARN_LOG_FILE_NAME}/$${date:yyyy-MM}/warn-%d{yyyy-MM-dd}-%i.log">
<ThresholdFilter level="warn" onMatch="ACCEPT"
onMismatch="DENY" />
<PatternLayout
pattern="%d{yyyy-MM-dd HH:mm:ss.SSS} [%t] %-5level %logger{36} - %msg%n" />
<Policies>
<TimeBasedTriggeringPolicy />
<SizeBasedTriggeringPolicy size="2 kB" />
</Policies>
<!-- DefaultRolloverStrategy屬性如不設(shè)置,則默認(rèn)為最多同一文件夾下7個文件,這里設(shè)置了20 -->
<DefaultRolloverStrategy max="20" />
</RollingFile>
<RollingFile name="RollingFileError" fileName="${ERROR_LOG_FILE_NAME}/error.log"
filePattern="${ERROR_LOG_FILE_NAME}/$${date:yyyy-MM}/error-%d{yyyy-MM-dd-HH-mm}-%i.log">
<ThresholdFilter level="error" onMatch="ACCEPT"
onMismatch="DENY" />
<PatternLayout
pattern="%d{yyyy-MM-dd HH:mm:ss.SSS} [%t] %-5level %logger{36} - %msg%n" />
<Policies>
<!-- log4j2的按分鐘 分日志文件 : warn-%d{yyyy-MM-dd-HH-mm}-%i.log -->
<TimeBasedTriggeringPolicy interval="1"
modulate="true" />
<!-- <SizeBasedTriggeringPolicy size="10 MB" /> -->
</Policies>
</RollingFile>
</Appenders>
<!--然后定義logger,只有定義了logger并引入的appender,appender才會生效 -->
<Loggers>
<!--過濾掉spring和mybatis的一些無用的DEBUG信息 -->
<logger name="org.springframework" level="INFO"></logger>
<logger name="org.mybatis" level="INFO"></logger>
<!-- 第三方日志系統(tǒng) -->
<logger name="org.springframework" level="ERROR" />
<logger name="org.hibernate" level="ERROR" />
<logger name="org.apache.struts2" level="ERROR" />
<logger name="com.opensymphony.xwork2" level="ERROR" />
<logger name="org.jboss" level="ERROR" />
<!-- 配置日志的根節(jié)點(diǎn) -->
<root level="all">
<appender-ref ref="Console" />
<appender-ref ref="RollingFileInfo" />
<appender-ref ref="RollingFileWarn" />
<appender-ref ref="RollingFileError" />
</root>
</Loggers>
</Configuration>
log4j2.xml文件應(yīng)該位于項(xiàng)目的類路徑根目錄下,即src/main/resources目錄中。
小貼士:
Log4j2的配置文件,用于配置日志輸出的格式、級別、目標(biāo)等信息。Log4j2不再支持Log4j 1.x中的.properties文件的配置方式,而是支持通過json、xml或者jsn的方式進(jìn)行配置。
具體來說,log4j2.xml文件中包含了以下幾個部分:
- Configuration:配置信息區(qū)域,包括Appenders、Loggers等。
- Appender:輸出目標(biāo)區(qū)域,包括控制臺、文件、數(shù)據(jù)庫等。
- Logger:日志記錄區(qū)域,包括包名、類名、日志級別等。
在MyBatis中有三種模糊查詢的方法分別是:
①使用#{字段名}
<select id="like1" resultType="com.csdn.xw.model.Book" parameterType="java.lang.String" >
select
<include refid="Base_Column_List" />
from t_mvc_book
where bname like #{bname}
</select>
測試結(jié)果:?
②使用${字段名}
<select id="like2" resultType="com.csdn.xw.model.Book" parameterType="java.lang.String" >
select
<include refid="Base_Column_List" />
from t_mvc_book
where bname like ${bname}
</select>
測試結(jié)果:??
③使用concat{'%',#{字段名},'%'}
<select id="like3" resultType="com.csdn.xw.model.Book" parameterType="java.lang.String" >
select
<include refid="Base_Column_List" />
from t_mvc_book
where bname like concat('%',#{bname},'%')
</select>
測試結(jié)果:?
總結(jié)
想必你也看出了,這三種的不同之處,下面我來給你總結(jié)一下。
mybatis中的$與#的區(qū)別
①$是占位符傳參,#是預(yù)處理SQL
②外在形式:$傳參不帶引號,#自帶引號
可能你還沒感受到有沒有引號沒什么太大的區(qū)別,下面我來給你演示一下
假如我要根據(jù)部門編號查出該部門的所有人員信息,利用$與#來做測試
#:select*from t_oa_employee where deptid=#{deptid}
????????select*from t_oa_employee where deptid='10011'
$:select*from t_oa_employee where deptid=${deptid}
??????????select*from t_oa_employee where deptid=10011
這么一看好像確實(shí)沒有什么區(qū)別,那如果前臺惡意傳參呢?
http://localhost:8080/employee?deptid=10011 or 1=1
那這樣不就把全公司的人員信息都查出來了
③$傳參存在sql注入,#不存在
④$可以做動態(tài)列,完成動態(tài)sql的開發(fā)
我們開發(fā)過程中肯定有遇到過,當(dāng)前字段不滿足我們需求,然后我們修改數(shù)據(jù)庫字段的時候,如果改成正確還好,如果錯誤那就芭比Q了,動態(tài)列就可以做到傳入的參數(shù)當(dāng)作需要查詢的列來操作例如:
假設(shè)我們有一個表名為user_info,包含字段id、name、age、gender等,現(xiàn)在需要根據(jù)傳入的參數(shù)來動態(tài)生成查詢的列,可以使用如下的語句
<select id="getUserInfo" resultType="map"> SELECT ${columnName}, ${columnAge}, ${columnGender} FROM user_info </select>
其中,columnName、columnAge、columnGender分別代表需要查詢的列名,可以在Java代碼中定義一個Map變量,將列名作為key,對應(yīng)的值作為value傳入即可。例如
Map<String, Object> columnMap = new HashMap<>(); columnMap.put("columnName", "name"); columnMap.put("columnAge", 18); columnMap.put("columnGender", "男");
然后將columnMap傳入MyBatis的mapper中調(diào)用即可。
最后小編還是建議大家一般能用 # 的就別用 $ ,最常用的還是第三種。
四、MyBatis結(jié)果映射
在使用MyBatis中擁有多個場景,返回的結(jié)果是多樣的,resultType/resultMap
①返回單表的對應(yīng)的實(shí)體類,僅有一個查詢結(jié)果,可以用resultType/resultMap。
②返回單表的對應(yīng)的實(shí)體類,有多個查詢結(jié)果,可以用resultType/resultMap。
③返回多表對應(yīng)結(jié)果,僅有一個查詢結(jié)果,通常用resultType也可以用resultMap。
④返回多表對應(yīng)結(jié)果,有多個查詢結(jié)果,通常用resultType也可以用resultMap。
⑤返回單個列段,僅有一個查詢結(jié)果,就用resultType。
⑥返回單個列段,有多個查詢結(jié)果,就用resultType。
? 總結(jié)就是在Mybatis中結(jié)果集的處理分為兩種:
resultMap:適合使用返回值是自定義實(shí)體類的情況
resultType:適合使用返回值的數(shù)據(jù)類型是非自定義的,即jdk的提供的類型
?如果是單表的情況下,resultType與resultMap都可以使用。
1 使用resultMap返回映射關(guān)系,指的是實(shí)體類與數(shù)據(jù)庫字段的關(guān)系
2 使用resultType返回List<T>
3 使用resultType返回單個對象
4 使用resultType返回List<Map>【適用于多表查詢返回結(jié)果集】
5 使用resultType返回Map<String,Object>【適用于多表查詢返回單個結(jié)果集】
4.1.案例演示
4.1.1.resultType進(jìn)行結(jié)果映射
假設(shè)我們有一個用戶表user,包含以下字段:id、name、age、gender、email?,F(xiàn)在需要根據(jù)這些字段查詢用戶信息并返回一個User對象。
首先,我們需要定義一個User類,用于存儲查詢結(jié)果:
public class User {
private int id;
private String name;
private int age;
private String gender;
private String email;
// getter和setter方法省略
}
然后,在MyBatis的mapper文件中,我們可以使用resultType來映射查詢結(jié)果到User對象上。具體配置如下:
<select id="getUserById" resultType="com.example.User">
SELECT id, name, age, gender, email
FROM user
WHERE id = #{id}
</select>
在上面的配置中,我們使用了resultType屬性來指定查詢結(jié)果映射到的Java對象的全限定名。這樣,當(dāng)執(zhí)行查詢語句時,MyBatis會自動將查詢結(jié)果映射到指定的Java對象上。
4.1.2.resultMap進(jìn)行結(jié)果映射
假設(shè)我們有一個訂單表order,包含以下字段:id、user_id、product_id、price、quantity?,F(xiàn)在需要根據(jù)這些字段查詢訂單信息并返回一個Order對象。
首先,我們需要定義一個Order類,用于存儲查詢結(jié)果:
public class Order {
private int id;
private int userId;
private int productId;
private double price;
private int quantity;
// getter和setter方法省略
}
然后,在MyBatis的mapper文件中,我們可以使用resultMap來映射查詢結(jié)果到Order對象上。具體配置如下:
<resultMap id="OrderResultMap" type="com.example.Order">
<id property="id" column="id" />
<result property="userId" column="user_id" />
<result property="productId" column="product_id" />
<result property="price" column="price" />
<result property="quantity" column="quantity" />
</resultMap>
<select id="getOrderById" resultMap="OrderResultMap">
SELECT id, user_id, product_id, price, quantity
FROM order
WHERE id = #{id}
</select>
在上面的配置中,我們定義了一個名為"OrderResultMap"的resultMap,它的type屬性指定了要映射的Java對象的全限定名。接下來,我們?yōu)槊總€字段指定了對應(yīng)的屬性名和數(shù)據(jù)庫表中的列名。最后,在查詢語句中引用這個resultMap即可將查詢結(jié)果映射到Order對象上。
需要注意的是,如果查詢結(jié)果中的某個字段在Java對象中沒有對應(yīng)的屬性,那么該字段將被映射為null。此外,如果查詢結(jié)果中的某個字段在Java對象中有多個對應(yīng)的屬性,那么該字段的值將被映射為一個列表或數(shù)組。
到這里我的分享就結(jié)束了,歡迎到評論區(qū)探討交流!!
如果覺得有用的話還請點(diǎn)個贊吧??? ?文章來源:http://www.zghlxwxcb.cn/news/detail-665128.html
文章來源地址http://www.zghlxwxcb.cn/news/detail-665128.html
到了這里,關(guān)于MyBatis進(jìn)階:掌握MyBatis動態(tài)SQL與模糊查詢、結(jié)果映射,讓你在面試中脫穎而出??!的文章就介紹完了。如果您還想了解更多內(nèi)容,請?jiān)谟疑辖撬阉鱐OY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!