MyBatis實現(xiàn)CRUD的示例代碼
準(zhǔn)備工作
創(chuàng)建module(Maven的普通Java模塊):mybatis-002-crud
pom.xml
- 打包方式j(luò)ar
- 依賴:
- mybatis依賴
- mysql驅(qū)動依賴
- junit依賴
- logback依賴
mybatis-config.xml放在類的根路徑下CarMapper.xml放在類的根路徑下logback.xml放在類的根路徑下- 提供
com.study.mybatis.utils.SqlSessionUtil工具類 - 創(chuàng)建測試用例:
com.study.mybatis.CarMapperTest
1 insert(Create)
分析以下SQL映射文件中SQL語句存在的問題
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<!--namespace先隨便寫-->
<mapper namespace="car">
<insert id="insertCar">
insert into t_car(car_num, brand, guide_price, produce_time, car_type) values ('103', '奔馳E300L', 50.3, '2022-01-01', '燃油車')
</insert>
</mapper>
存在的問題是:SQL語句中的值不應(yīng)該寫死,值應(yīng)該是用戶提供的。之前的JDBC代碼是這樣寫的:
// JDBC中使用 ? 作為占位符。那么MyBatis中會使用什么作為占位符呢? String sql = "insert into t_car(car_num,brand,guide_price,produce_time,car_type) values(?,?,?,?,?)"; // ...... // 給 ? 傳值。那么MyBatis中應(yīng)該怎么傳值呢? ps.setString(1,"103"); ps.setString(2,"奔馳E300L"); ps.setDouble(3,50.3); ps.setString(4,"2022-01-01"); ps.setString(5,"燃油車");
在MyBatis中可以這樣做:
在Java程序中,將數(shù)據(jù)放到Map集合中在sql語句中使用 #{map集合的key} 來完成傳值,#{} 等同于JDBC中的 ? ,#{}就是占位符Java程序這樣寫:
package com.study.mybatis;
import com.study.mybatis.utils.SqlSessionUtil;
import org.apache.ibatis.session.SqlSession;
import org.junit.Test;
import java.util.HashMap;
import java.util.Map;
/**
* 測試MyBatis的CRUD
* @author sqnugy
* @version 1.0
* @since 1.0
*/
public class CarMapperTest {
@Test
public void testInsertCar(){
// 準(zhǔn)備數(shù)據(jù)
Map<String, Object> map = new HashMap<>();
map.put("k1", "103");
map.put("k2", "奔馳E300L");
map.put("k3", 50.3);
map.put("k4", "2020-10-01");
map.put("k5", "燃油車");
// 獲取SqlSession對象
SqlSession sqlSession = SqlSessionUtil.openSession();
// 執(zhí)行SQL語句(使用map集合給sql語句傳遞數(shù)據(jù))
int count = sqlSession.insert("insertCar", map);
System.out.println("插入了幾條記錄:" + count);
}
}
SQL語句這樣寫:
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<!--namespace先隨便寫-->
<mapper namespace="car">
<insert id="insertCar">
insert into t_car(car_num,brand,guide_price,produce_time,car_type) values(#{k1},#{k2},#{k3},#{k4},#{k5})
</insert>
</mapper>
**#{} 的里面必須填寫map集合的key,不能隨便寫。**運行測試程序,查看數(shù)據(jù)庫:

如果#{}里寫的是map集合中不存在的key會有什么問題?
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="car">
<insert id="insertCar">
insert into t_car(car_num,brand,guide_price,produce_time,car_type) values(#{kk},#{k2},#{k3},#{k4},#{k5})
</insert>
</mapper>
運行程序:


通過測試,看到程序并沒有報錯。正常執(zhí)行。不過 #{kk} 的寫法導(dǎo)致無法獲取到map集合中的數(shù)據(jù),最終導(dǎo)致數(shù)據(jù)庫表car_num插入了NULL。
在以上sql語句中,可以看到#{k1} #{k2} #{k3} #{k4} #{k5}的可讀性太差,為了增強(qiáng)可讀性,我們可以將Java程序做如下修改:
Map<String, Object> map = new HashMap<>();
// 讓key的可讀性增強(qiáng)
map.put("carNum", "103");
map.put("brand", "奔馳E300L");
map.put("guidePrice", 50.3);
map.put("produceTime", "2020-10-01");
map.put("carType", "燃油車");
SQL語句做如下修改,這樣可以增強(qiáng)程序的可讀性:
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="car">
<insert id="insertCar">
insert into t_car(car_num,brand,guide_price,produce_time,car_type) values(#{carNum},#{brand},#{guidePrice},#{produceTime},#{carType})
</insert>
</mapper>
運行程序,查看數(shù)據(jù)庫表:

使用Map集合可以傳參,那使用pojo(簡單普通的java對象)可以完成傳參嗎?測試一下:
- 第一步:定義一個pojo類Car,提供相關(guān)屬性。
package com.study.mybatis.pojo;
/**
* POJOs,簡單普通的Java對象。封裝數(shù)據(jù)用的。
* @author sqnugy
* @version 1.0
* @since 1.0
*/
public class Car {
private Long id;
private String carNum;
private String brand;
private Double guidePrice;
private String produceTime;
private String carType;
@Override
public String toString() {
return "Car{" +
"id=" + id +
", carNum='" + carNum + '\'' +
", brand='" + brand + '\'' +
", guidePrice=" + guidePrice +
", produceTime='" + produceTime + '\'' +
", carType='" + carType + '\'' +
'}';
}
public Car() {
}
public Car(Long id, String carNum, String brand, Double guidePrice, String produceTime, String carType) {
this.id = id;
this.carNum = carNum;
this.brand = brand;
this.guidePrice = guidePrice;
this.produceTime = produceTime;
this.carType = carType;
}
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getCarNum() {
return carNum;
}
public void setCarNum(String carNum) {
this.carNum = carNum;
}
public String getBrand() {
return brand;
}
public void setBrand(String brand) {
this.brand = brand;
}
public Double getGuidePrice() {
return guidePrice;
}
public void setGuidePrice(Double guidePrice) {
this.guidePrice = guidePrice;
}
public String getProduceTime() {
return produceTime;
}
public void setProduceTime(String produceTime) {
this.produceTime = produceTime;
}
public String getCarType() {
return carType;
}
public void setCarType(String carType) {
this.carType = carType;
}
}
- 第二步:Java程序
@Test
public void testInsertCarByPOJO(){
// 創(chuàng)建POJO,封裝數(shù)據(jù)
Car car = new Car();
car.setCarNum("103");
car.setBrand("奔馳C200");
car.setGuidePrice(33.23);
car.setProduceTime("2020-10-11");
car.setCarType("燃油車");
// 獲取SqlSession對象
SqlSession sqlSession = SqlSessionUtil.openSession();
// 執(zhí)行SQL,傳數(shù)據(jù)
int count = sqlSession.insert("insertCarByPOJO", car);
System.out.println("插入了幾條記錄" + count);
}
- 第三步:SQL語句
<insert id="insertCarByPOJO">
<!--#{} 里寫的是POJO的屬性名-->
insert into t_car(car_num,brand,guide_price,produce_time,car_type) values(#{carNum},#{brand},#{guidePrice},#{produceTime},#{carType})
</insert>
- 運行程序,查看數(shù)據(jù)庫表:

#{} 里寫的是POJO的屬性名,如果寫成其他的會有問題嗎?
<insert id="insertCarByPOJO">
insert into t_car(car_num,brand,guide_price,produce_time,car_type) values(#{a},#{brand},#{guidePrice},#{produceTime},#{carType})
</insert>
運行程序,出現(xiàn)了以下異常:

錯誤信息中描述:在Car類中沒有找到a屬性的getter方法。
修改POJO類Car的代碼,只將getCarNum()方法名修改為getA(),其他代碼不變,如下:

再運行程序,查看數(shù)據(jù)庫表中數(shù)據(jù):

經(jīng)過測試得出結(jié)論:
如果采用map集合傳參,#{} 里寫的是map集合的key,如果key不存在不會報錯,數(shù)據(jù)庫表中會插入NULL。
如果采用POJO傳參,#{} 里寫的是get方法的方法名去掉get之后將剩下的單詞首字母變小寫(例如:getAge對應(yīng)的是#{age},getUserName對應(yīng)的是#{userName}),如果這樣的get方法不存在會報錯。
注意:其實傳參數(shù)的時候有一個屬性parameterType,這個屬性用來指定傳參的數(shù)據(jù)類型,不過這個屬性是可以省略的
<insert id="insertCar" parameterType="java.util.Map">
insert into t_car(car_num,brand,guide_price,produce_time,car_type) values(#{carNum},#{brand},#{guidePrice},#{produceTime},#{carType})
</insert>
<insert id="insertCarByPOJO" parameterType="com.study.mybatis.pojo.Car">
insert into t_car(car_num,brand,guide_price,produce_time,car_type) values(#{carNum},#{brand},#{guidePrice},#{produceTime},#{carType})
</insert>
2 delete(Delete)
需求:根據(jù)car_num進(jìn)行刪除。
SQL語句這樣寫:
<delete id="deleteByCarNum">
delete from t_car where car_num = #{SuiBianXie}
</delete>
Java程序這樣寫:
@Test
public void testDeleteByCarNum(){
// 獲取SqlSession對象
SqlSession sqlSession = SqlSessionUtil.openSession();
// 執(zhí)行SQL語句
int count = sqlSession.delete("deleteByCarNum", "102");
System.out.println("刪除了幾條記錄:" + count);
}
運行結(jié)果:

注意:當(dāng)占位符只有一個的時候,${} 里面的內(nèi)容可以隨便寫。
3 update(Update)
需求:修改id=34的Car信息,car_num為102,brand為比亞迪漢,guide_price為30.23,produce_time為2018-09-10,car_type為電車修改前:

SQL語句如下:
<update id="updateCarByPOJO">
update t_car set
car_num = #{carNum}, brand = #{brand},
guide_price = #{guidePrice}, produce_time = #{produceTime},
car_type = #{carType}
where id = #{id}
</update>
Java代碼如下:
@Test
public void testUpdateCarByPOJO(){
// 準(zhǔn)備數(shù)據(jù)
Car car = new Car();
car.setId(34L);
car.setCarNum("102");
car.setBrand("比亞迪漢");
car.setGuidePrice(30.23);
car.setProduceTime("2018-09-10");
car.setCarType("電車");
// 獲取SqlSession對象
SqlSession sqlSession = SqlSessionUtil.openSession();
// 執(zhí)行SQL語句
int count = sqlSession.update("updateCarByPOJO", car);
System.out.println("更新了幾條記錄:" + count);
}
運行結(jié)果:


當(dāng)然了,如果使用map傳數(shù)據(jù)也是可以的。
4 select(Retrieve)
select語句和其它語句不同的是:查詢會有一個結(jié)果集。來看mybatis是怎么處理結(jié)果集的?。?!
查詢一條數(shù)據(jù)
需求:查詢id為1的Car信息
SQL語句如下:
<select id="selectCarById">
select * from t_car where id = #{id}
</select>
Java程序如下:
@Test
public void testSelectCarById(){
// 獲取SqlSession對象
SqlSession sqlSession = SqlSessionUtil.openSession();
// 執(zhí)行SQL語句
Object car = sqlSession.selectOne("selectCarById", 1);
System.out.println(car);
}
運行結(jié)果如下:
### Error querying database. Cause: org.apache.ibatis.executor.ExecutorException:
A query was run and no Result Maps were found for the Mapped Statement 'car.selectCarById'. 【翻譯】:對于一個查詢語句來說,沒有找到查詢的結(jié)果映射。
It's likely that neither a Result Type nor a Result Map was specified. 【翻譯】:很可能既沒有指定結(jié)果類型,也沒有指定結(jié)果映射。
以上的異常大致的意思是:對于一個查詢語句來說,你需要指定它的“結(jié)果類型”或者“結(jié)果映射”。
所以說,你想讓mybatis查詢之后返回一個Java對象的話,至少你要告訴mybatis返回一個什么類型的Java對象,可以在<select>標(biāo)簽中添加resultType屬性,用來指定查詢要轉(zhuǎn)換類型:
<select id="selectCarById" resultType="com.study.mybatis.pojo.Car">
select * from t_car where id = #{id}
</select>
運行結(jié)果:

運行后之前的異常不再出現(xiàn)了,這說明添加了resultType屬性之后,解決了之前的異常,可以看出resultType是不能省略的。
仔細(xì)觀察控制臺的日志信息,不難看出,結(jié)果查詢出了一條。并且每個字段都查詢到值了:Row: 1, 100, 寶馬520Li, 41.00, 2022-09-01, 燃油車
但是奇怪的是返回的Car對象,只有id和brand兩個屬性有值,其它屬性的值都是null,這是為什么呢?我們來觀察一下查詢結(jié)果列名和Car類的屬性
名是否能一一對應(yīng):
查詢結(jié)果集的列名:id, car_num, brand, guide_price, produce_time, car_type
Car類的屬性名:id, carNum, brand, guidePrice, produceTime, carType
通過觀察發(fā)現(xiàn):只有id和brand是一致的,其他字段名和屬性名對應(yīng)不上,這是不是導(dǎo)致null的原因呢?我們嘗試在sql語句中使用as關(guān)鍵字來給查
詢結(jié)果列名起別名試試:
<select id="selectCarById" resultType="com.study.mybatis.pojo.Car">
select
id, car_num as carNum, brand, guide_price as guidePrice, produce_time as produceTime, car_type as carType
from
t_car
where
id = #{id}
</select>
運行結(jié)果如下:

通過測試得知,如果當(dāng)查詢結(jié)果的字段名和java類的屬性名對應(yīng)不上的話,可以采用as關(guān)鍵字起別名,當(dāng)然還有其它解決方案,我們后面再看。
查詢多條數(shù)據(jù)
需求:查詢所有的Car信息。
SQL語句如下:
<!--雖然結(jié)果是List集合,但是resultType屬性需要指定的是List集合中元素的類型。-->
<select id="selectCarAll" resultType="com.study.mybatis.pojo.Car">
<!--記得使用as起別名,讓查詢結(jié)果的字段名和java類的屬性名對應(yīng)上。-->
select
id, car_num as carNum, brand, guide_price as guidePrice, produce_time as produceTime, car_type as carType
from
t_car
</select>
Java代碼如下:
@Test
public void testSelectCarAll(){
// 獲取SqlSession對象
SqlSession sqlSession = SqlSessionUtil.openSession();
// 執(zhí)行SQL語句
List<Object> cars = sqlSession.selectList("selectCarAll");
// 輸出結(jié)果
cars.forEach(car -> System.out.println(car));
}
運行結(jié)果如下:

5 關(guān)于SQL Mapper的namespace
在SQL Mapper配置文件中<mapper>標(biāo)簽的namespace屬性可以翻譯為命名空間,這個命名空間主要是為了防止sqlId沖突的。
創(chuàng)建CarMapper2.xml文件,代碼如下:
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="car2">
<select id="selectCarAll" resultType="com.study.mybatis.pojo.Car">
select
id, car_num as carNum, brand, guide_price as guidePrice, produce_time as produceTime, car_type as carType
from
t_car
</select>
</mapper>
不難看出,CarMapper.xml和CarMapper2.xml文件中都有 id="selectCarAll"將CarMapper2.xml配置到mybatis-config.xml文件中。
<mappers> <mapper resource="CarMapper.xml"/> <mapper resource="CarMapper2.xml"/> </mappers>
編寫Java代碼如下:
@Test
public void testNamespace(){
// 獲取SqlSession對象
SqlSession sqlSession = SqlSessionUtil.openSession();
// 執(zhí)行SQL語句
List<Object> cars = sqlSession.selectList("selectCarAll");
// 輸出結(jié)果
cars.forEach(car -> System.out.println(car));
}
運行結(jié)果如下:
org.apache.ibatis.exceptions.PersistenceException:
### Error querying database. Cause: java.lang.IllegalArgumentException:
selectCarAll is ambiguous in Mapped Statements collection (try using the full name including the namespace, or rename one of the entries)
【翻譯】selectCarAll在Mapped Statements集合中不明確(請嘗試使用包含名稱空間的全名,或重命名其中一個條目)
【大致意思是】selectCarAll重名了,你要么在selectCarAll前添加一個名稱空間,要有你改個其它名字。
Java代碼修改如下:
@Test
public void testNamespace(){
// 獲取SqlSession對象
SqlSession sqlSession = SqlSessionUtil.openSession();
// 執(zhí)行SQL語句
//List<Object> cars = sqlSession.selectList("car.selectCarAll");
List<Object> cars = sqlSession.selectList("car2.selectCarAll");
// 輸出結(jié)果
cars.forEach(car -> System.out.println(car));
}
運行結(jié)果如下:

到此這篇關(guān)于MyBatis實現(xiàn)CRUD的示例代碼的文章就介紹到這了,更多相關(guān)MyBatis實現(xiàn)CRUD內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Mybatis實現(xiàn)增刪改查(CRUD)實例代碼
- Mybatis實現(xiàn)數(shù)據(jù)的增刪改查實例(CRUD)
- Spring Boot整合Mybatis并完成CRUD操作的實現(xiàn)示例
- Spring boot整合Mybatis實現(xiàn)級聯(lián)一對多CRUD操作的完整步驟
- 基于Mybatis-Plus的CRUD的實現(xiàn)
- MyBatis Plus配置日志CRUD的使用詳解
- MybatisPlus,無XML分分鐘實現(xiàn)CRUD操作
- Mybatis-Plus 通用CRUD的詳細(xì)操作
- mybatisplus?復(fù)合主鍵(多主鍵)?CRUD示例詳解
相關(guān)文章
java kafka寫入數(shù)據(jù)到HDFS問題
這篇文章主要介紹了java kafka寫入數(shù)據(jù)到HDFS問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-08-08
迅速掌握J(rèn)ava容器中常用的ArrayList類與Vector類用法
這篇文章主要介紹了Java容器中常用的ArrayList類與Vector類用法,文中只對其最基本的功能給出了示例代碼,需要的朋友可以參考下2015-11-11
java?kafka如何動態(tài)設(shè)置用戶讀寫權(quán)限
這篇文章主要介紹了java?kafka如何動態(tài)設(shè)置用戶讀寫權(quán)限問題,具有很好的參考家價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-08-08
詳解spring cloud使用Hystrix實現(xiàn)單個方法的fallback
本篇文章主要介紹了詳解spring cloud-使用Hystrix實現(xiàn)單個方法的fallback,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-01-01
SpringBoot詳解shiro過濾器與權(quán)限控制
當(dāng)shiro被運用到web項目時,shiro會自動創(chuàng)建一些默認(rèn)的過濾器對客戶端請求進(jìn)行過濾。比如身份驗證、授權(quán)的相關(guān)的,這篇文章主要介紹了shiro過濾器與權(quán)限控制2022-07-07
SpringBoot中的Spring Cloud Hystrix原理和用法詳解
在Spring Cloud中,Hystrix是一個非常重要的組件,Hystrix可以幫助我們構(gòu)建具有韌性的分布式系統(tǒng),保證系統(tǒng)的可用性和穩(wěn)定性,在本文中,我們將介紹SpringBoot中的Hystrix,包括其原理和如何使用,需要的朋友可以參考下2023-07-07
Java實現(xiàn)curl調(diào)用帶參數(shù)接口方法
本文主要介紹了Java實現(xiàn)curl調(diào)用帶參數(shù)接口方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2025-04-04

