一、使用MyBatis完成CRUD
-
準備工作
-
創建module(Maven的普通Java模塊):mybatis-002-crud
-
pom.xml
-
打包方式jar
-
依賴:
-
mybatis依賴
-
mysql驅動依賴
-
junit依賴
-
logback依賴
-
-
-
mybatis-config.xml放在類的根路徑下
-
CarMapper.xml放在類的根路徑下
-
logback.xml放在類的根路徑下
-
提供utils.SqlSessionUtil工具類
-
1.insert(Create)
分析以下SQL映射文件中SQL語句存在的問題
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapperPUBLIC "-//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語句中的值不應該寫死,值應該是用戶提供的。之前的JDBC代碼是這樣寫的:
// JDBC中使用 ? 作為占位符。那么MyBatis中會使用什么作為占位符呢?
String sql = "insert into t_car(car_num,brand,guide_price,produce_time,car_type) values(?,?,?,?,?)";
// ......
// 給 ? 傳值。那么MyBatis中應該怎么傳值呢?
ps.setString(1,"103");
ps.setString(2,"奔馳E300L");
ps.setDouble(3,50.3);
ps.setString(4,"2022-01-01");
ps.setString(5,"燃油車");
在MyBatis中可以這樣做:
在Java程序中,將數據放到Map集合中
在sql語句中使用 #{map集合的key} 來完成傳值,#{} 等同于JDBC中的 ? ,#{}就是占位符
Java程序這樣寫:
import com.powernode.mybatis.utils.SqlSessionUtil;
import org.apache.ibatis.session.SqlSession;
import org.junit.Test;import java.util.HashMap;
import java.util.Map;/*** 測試MyBatis的CRUD*/
public class CarMapperTest {@Testpublic void testInsertCar(){// 準備數據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();// 執行SQL語句(使用map集合給sql語句傳遞數據)int count = sqlSession.insert("insertCar", map);System.out.println("插入了幾條記錄:" + count);}
}
SQL語句這樣寫:
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapperPUBLIC "-//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>
⑴.演示代碼運行
mybatis-config.xml? ?這是數據庫的的配置文件
CarMapper.xml? ?這是對應表操作的文件
測試運行文件
運行前的數據庫
#{} 的里面必須填寫map集合的key,不能隨便寫。運行測試程序,查看數據庫:
⑵如果#{}里寫的是map集合中不存在的key會有什么問題?
運行程序:
通過測試,看到程序并沒有報錯。正常執行。不過 #{kk} 的寫法導致無法獲取到map集合中的數據,最終導致數據庫表car_num插入了NULL。
在以上sql語句中,可以看到#{k1} #{k2} #{k3} #{k4} #{k5}的可讀性太差,為了增強可讀性,我們可以將Java程序做如下修改:
運行程序,查看數據庫表:
⑶使用Map集合可以傳參,那使用pojo(簡單普通的java對象)可以完成傳參嗎?測試一下:
-
第一步:定義一個pojo類Car,提供相關屬性。
package org.example1.pojo;/*** 封裝汽車相關信息的pojo類。普通的java類。*/
public class Car {// 數據庫表當中的字段應該和pojo類的屬性一一對應。// 建議使用包裝類,這樣可以防止null的問題。private Long id;private String carNum;private String brand;private Double guidePrice;private String produceTime;private String carType;@Overridepublic String toString() {return "Car{" +"id=" + id +", carNum='" + carNum + '\'' +", brand='" + brand + '\'' +", guidePrice=" + guidePrice +", produceTime='" + produceTime + '\'' +", carType='" + carType + '\'' +'}';}public Long getId() {return id;}public void setId(Long id) {this.id = id;}public String getCarNum() {return carNum;}/*public String getXyz() {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;}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 Car() {}
}
-
第二步:Java程序
-
第三步:SQL語句
-
運行程序,查看數據庫表:
⑷#{} 里寫的是POJO的屬性名,如果寫成其他的會有問題嗎?
運行程序,出現了以下異常:
錯誤信息中描述:在Car類中沒有找到a屬性的getter方法。 修改POJO類Car的代碼,只將getCarNum()方法名修改為getA(),其他代碼不變,如下:
再運行程序,查看數據庫表中數據:
經過測試得出結論:
如果采用map集合傳參,#{} 里寫的是map集合的key,如果key不存在不會報錯,數據庫表中會插入NULL。
如果采用POJO傳參,#{} 里寫的是get方法的方法名去掉get之后將剩下的單詞首字母變小寫(例如:getAge對應的是#{age},getUserName對應的是#{userName}),如果這樣的get方法不存在會報錯。
注意:其實傳參數的時候有一個屬性parameterType,這個屬性用來指定傳參的數據類型,不過這個屬性是可以省略的
<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="org.example1.pojo.Car">insert into t_car(car_num,brand,guide_price,produce_time,car_type) values(#{carNum},#{brand},#{guidePrice},#{produceTime},#{carType})
</insert>
通過這個測試,得出一個結論:嚴格意義上來說:如果使用POJO對象傳遞值的話,#{}這個大括號中到底寫什么?寫的是get方法的方法名去掉get,然后將剩下的單詞首字母小寫,然后放進去。例如:getUsername() --> #{username}例如:getEmail() --> #{email}.... 也就是說mybatis在底層給?傳值的時候,先要獲取值,怎么獲取的?調用了pojo對象的get方法。例如:car.getCarNum(),car.getCarType(),car.getBrand()
2.delete(Delete)
需求:根據car_num進行刪除。
SQL語句這樣寫:
CarMapper.xml
<delete id="deleteByCarNum">delete from t_car where car_num = #{SuiBianXie}
</delete>
Java程序這樣寫:
運行前的數據庫
運行結果:
注意:當占位符只有一個的時候,${} 里面的內容可以隨便寫。
3.update(Update)
需求:修改id=3的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代碼如下:
@Testpublic void testUpdateCarByPOJO(){// 準備數據Car car = new Car();car.setId(3L);car.setCarNum("102");car.setBrand("比亞迪漢");car.setGuidePrice(30.23);car.setProduceTime("2018-09-10");car.setCarType("電車");// 獲取SqlSession對象SqlSession sqlSession = SqlSessionUtil.openSession();// 執行SQL語句int count = sqlSession.update("updateCarByPOJO", car);System.out.println("更新了幾條記錄:" + count);sqlSession.commit();//注意要提交sqlSession.close();}
運行結果:
4.select(Retrieve)
select語句和其它語句不同的是:查詢會有一個結果集。來看mybatis是怎么處理結果集的!!!
⑴.查詢一條數據
需求:查詢id為1的Car信息 SQL語句如下:
<select id="selectCarById">select * from t_car where id = #{id}
</select>
Java程序如下:
@Testpublic void testSelectById(){SqlSession sqlSession = SqlSessionUtil.openSession();// 執行DQL語句。查詢。根據id查詢。返回結果一定是一條。// mybatis底層執行了select語句之后,一定會返回一個結果集對象:ResultSet// JDBC中叫做ResultSet,接下來就是mybatis應該從ResultSet中取出數據,封裝java對象。Object car = sqlSession.selectOne("selectCarById", 1);System.out.println(car);sqlSession.close();}
運行結果如下:
### 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'. ?【翻譯】:對于一個查詢語句來說,沒有找到查詢的結果映射。
? ? It's likely that neither a Result Type nor a Result Map was specified.?? ??? ??? ??? ??? ??? ?【翻譯】:很可能既沒有指定結果類型,也沒有指定結果映射。
以上的異常大致的意思是:對于一個查詢語句來說,你需要指定它的“結果類型”或者“結果映射”。 所以說,你想讓mybatis查詢之后返回一個Java對象的話,至少你要告訴mybatis返回一個什么類型的Java對象,可以在<select>標簽中添加resultType屬性,用來指定查詢要轉換的類型:
<select id="selectCarById" resultType="org.example1.pojo.Car">select * from t_car where id = #{id}</select>
運行結果:
運行后之前的異常不再出現了,這說明添加了resultType屬性之后,解決了之前的異常,可以看出resultType是不能省略的。 仔細觀察控制臺的日志信息,不難看出,結果查詢出了一條。并且每個字段都查詢到值了:? Row: 1, 1001, 寶馬520Li, 10.00, 2020-10-11, 燃油車
但是奇怪的是返回的Car對象,只有id和brand兩個屬性有值,其它屬性的值都是null,這是為什么呢?我們來觀察一下查詢結果列名和Car類的屬性名是否能一一對應:
查詢結果集的列名:id, car_num, brand, guide_price, produce_time, car_type
Car類的屬性名:id, carNum, brand, guidePrice, produceTime, carType
通過觀察發現:只有id和brand是一致的,其他字段名和屬性名對應不上,這是不是導致null的原因呢?我們嘗試在sql語句中使用as關鍵字來給查詢結果列名起別名試試:
<select id="selectCarById" resultType="com.powernode.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>
運行結果如下:
通過測試得知,如果當查詢結果的字段名和java類的屬性名對應不上的話,可以采用as關鍵字起別名,當然還有其它解決方案,我們后面再看。
⑵查詢多條數據
需求:查詢所有的Car信息。 SQL語句如下:
<!--雖然結果是List集合,但是resultType屬性需要指定的是List集合中元素的類型。--><select id="selectCarAll" resultType="org.example1.pojo.Car"><!--記得使用as起別名,讓查詢結果的字段名和java類的屬性名對應上。-->selectid, car_num as carNum, brand, guide_price as guidePrice, produce_time as produceTime, car_type as carTypefromt_car</select>
Java代碼如下:
@Test
public void testSelectCarAll(){// 獲取SqlSession對象SqlSession sqlSession = SqlSessionUtil.openSession();// 執行SQL語句List<Object> cars = sqlSession.selectList("selectCarAll");// 輸出結果cars.forEach(car -> System.out.println(car));
}
運行結果如下:
注意:resultType還是指定要封裝的結果集的類型。不是指定List類型,是指定List集合中元素的類型。 selectList方法:mybatis通過這個方法就可以得知你需要一個List集合。它會自動給你返回一個List集合。
5.關于SQL Mapper的namespace
在SQL Mapper配置文件中<mapper>標簽的namespace屬性可以翻譯為命名空間,這個命名空間主要是為了防止sqlId沖突的。 創建CarMapper2.xml文件,代碼如下:
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapperPUBLIC "-//mybatis.org//DTD Mapper 3.0//EN""http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="car2"><!--雖然結果是List集合,但是resultType屬性需要指定的是List集合中元素的類型。--><select id="selectCarAll" resultType="org.example1.pojo.Car"><!--記得使用as起別名,讓查詢結果的字段名和java類的屬性名對應上。-->selectid, car_num as carNum, brand, guide_price as guidePrice, produce_time as produceTime, car_type as carTypefromt_car</select></mapper>
不難看出,CarMapper.xml和CarMapper2.xml文件中都有 id="selectCarAll" 將CarMapper2.xml配置到mybatis-config.xml文件中。
CarMapper.xml
編寫Java代碼如下:
@Testpublic void testSelectCarAll(){// 獲取SqlSession對象SqlSession sqlSession = SqlSessionUtil.openSession();// 執行SQL語句List<Object> cars = sqlSession.selectList("selectCarAll");// 輸出結果cars.forEach(car -> System.out.println(car));}
運行結果如下:
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();// 執行SQL語句//List<Object> cars = sqlSession.selectList("car.selectCarAll");List<Object> cars = sqlSession.selectList("car2.selectCarAll");// 輸出結果cars.forEach(car -> System.out.println(car));
}
運行結果如下:
二、MyBatis核心配置文件詳解
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configurationPUBLIC "-//mybatis.org//DTD Config 3.0//EN""http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration><settings><setting name="logImpl" value="STDOUT_LOGGING" /></settings><environments default="development"><environment id="development"><transactionManager type="JDBC"/><dataSource type="POOLED"><property name="driver" value="com.mysql.cj.jdbc.Driver"/><property name="url" value="jdbc:mysql://localhost:3306/jdbc"/><property name="username" value="root"/><property name="password" value="abc123"/></dataSource></environment></environments><mappers><!--sql映射文件創建好之后,需要將該文件路徑配置到這里--><mapper resource="Carmapper.xml"/><mapper resource="Carmapper2.xml"/></mappers>
</configuration>
-
configuration:根標簽,表示配置信息。
-
environments:環境(多個),以“s”結尾表示復數,也就是說mybatis的環境可以配置多個數據源。
-
default屬性:表示默認使用的是哪個環境,default后面填寫的是environment的id。default的值只需要和environment的id值一致即可。
-
-
environment:具體的環境配置(主要包括:事務管理器的配置 + 數據源的配置)
-
id:給當前環境一個唯一標識,該標識用在environments的default后面,用來指定默認環境的選擇。
-
<!--一般一個數據庫會對應一個SqlSessionFactory對象。--> <!--一個環境environment會對應一個SqlSessionFactory對象-->
-
transactionManager:配置事務管理器
-
type屬性:指定事務管理器具體使用什么方式,可選值包括兩個
-
JDBC:使用JDBC原生的事務管理機制。底層原理:事務開啟conn.setAutoCommit(false); ...處理業務...事務提交conn.commit();
-
MANAGED:交給其它容器來管理事務,比如WebLogic、JBOSS等。如果沒有管理事務的容器,則沒有事務。沒有事務的含義:只要執行一條DML語句,則提交一次。
-
-
-
dataSource:指定數據源
-
type屬性:用來指定具體使用的數據庫連接池的策略,可選值包括三個
-
UNPOOLED:采用傳統的獲取連接的方式,雖然也實現Javax.sql.DataSource接口,但是并沒有使用池的思想。
-
property可以是:
-
driver 這是 JDBC 驅動的 Java 類全限定名。
-
url 這是數據庫的 JDBC URL 地址。
-
username 登錄數據庫的用戶名。
-
password 登錄數據庫的密碼。
-
defaultTransactionIsolationLevel 默認的連接事務隔離級別。
-
defaultNetworkTimeout 等待數據庫操作完成的默認網絡超時時間(單位:毫秒)
-
-
-
POOLED:采用傳統的javax.sql.DataSource規范中的連接池,mybatis中有針對規范的實現。
-
property可以是(除了包含UNPOOLED中之外):
-
poolMaximumActiveConnections 在任意時間可存在的活動(正在使用)連接數量,默認值:10
-
poolMaximumIdleConnections 任意時間可能存在的空閑連接數。
-
其它....
-
-
-
JNDI:采用服務器提供的JNDI技術實現,來獲取DataSource對象,不同的服務器所能拿到DataSource是不一樣。如果不是web或者maven的war工程,JNDI是不能使用的。
-
property可以是(最多只包含以下兩個屬性):
-
initial_context 這個屬性用來在 InitialContext 中尋找上下文(即,initialContext.lookup(initial_context))這是個可選屬性,如果忽略,那么將會直接從 InitialContext 中尋找 data_source 屬性。
-
data_source 這是引用數據源實例位置的上下文路徑。提供了 initial_context 配置時會在其返回的上下文中進行查找,沒有提供時則直接在 InitialContext 中查找。
-
-
-
-
-
mappers:在mappers標簽中可以配置多個sql映射文件的路徑。
-
mapper:配置某個sql映射文件的路徑
-
resource屬性:使用相對于類路徑的資源引用方式
-
url屬性:使用完全限定資源定位符(URL)方式
-
1.environment
<!--一般一個數據庫會對應一個SqlSessionFactory對象。-->
<!--一個環境environment會對應一個SqlSessionFactory對象-->
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configurationPUBLIC "-//mybatis.org//DTD Config 3.0//EN""http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration><!--default表示默認使用的環境。--><!--默認環境什么意思?當你使用mybatis創建SqlSessionFactory對象的時候,沒有指定環境的話,默認使用哪個環境。--><environments default="production"><!--其中的一個環境。連接的數據庫是jdbc--><!--一般一個數據庫會對應一個SqlSessionFactory對象。--><!--一個環境environment會對應一個SqlSessionFactory對象--><!--開發環境--><environment id="dev"><transactionManager type="JDBC"/><dataSource type="POOLED"><property name="driver" value="com.mysql.cj.jdbc.Driver"/><property name="url" value="jdbc:mysql://localhost:3306/jdbc"/><property name="username" value="root"/><property name="password" value="abc123"/></dataSource></environment><!--生產環境--><environment id="production"><transactionManager type="JDBC" /><dataSource type="POOLED"><property name="driver" value="com.mysql.cj.jdbc.Driver"/><property name="url" value="jdbc:mysql://localhost:3306/car"/><property name="username" value="root"/><property name="password" value="abc123"/></dataSource></environment></environments><mappers><mapper resource="CarMapper.xml"/></mappers>
</configuration>
CarMapper.xml
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapperPUBLIC "-//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 values(null,'8888','沃爾沃',30.0,'2000-11-11','燃油車')</insert>
</mapper>
test
@Testpublic void testEnvironment() throws Exception {// 獲取SqlSessionFactory對象(采用默認的方式獲取)SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();// 這種方式就是獲取的默認環境SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(Resources.getResourceAsStream("mybatis-config.xml"));SqlSession sqlSession = sqlSessionFactory.openSession();// 執行SQL語句sqlSession.insert("car.insertCar");sqlSession.commit();sqlSession.close();// 這種方式就是通過環境id來使用指定的環境SqlSessionFactory sqlSessionFactory1 = sqlSessionFactoryBuilder.build(Resources.getResourceAsStream("mybatis-config.xml"), "dev");SqlSession sqlSession1 = sqlSessionFactory1.openSession();// 執行SQL語句sqlSession1.insert("car.insertCar");sqlSession1.commit();sqlSession1.close();}
運行前數據庫
運行后:
2.transactionManager
transactionManager標簽:1.作用:配置事務管理器。指定mybatis具體使用什么方式去管理事務。2.type屬性有兩個值:第一個:JDBC: 使用原生的JDBC代碼來管理事務。conn.setAutoCommit(false);....conn.commit();第二個:MANAGED:mybatis不再負責事務的管理,將事務管理交給其它的JEE(JavaEE)容器來管理。例如:spring3. 大小寫無所謂。不缺分大小寫。但是不能寫其他值。只能是二選一:jdbc、managed4. 在mybatis中提供了一個事務管理器接口:Transaction該接口下有兩個實現類:JdbcTransactionManagedTransaction如果type="JDBC",那么底層會實例化JdbcTransaction對象。如果type="MANAGED",那么底層會實例化ManagedTransaction
當事務管理器是:JDBC
-
采用JDBC的原生事務機制:
-
開啟事務:conn.setAutoCommit(false);
-
處理業務......
-
提交事務:conn.commit();
-
當事務管理器是:MANAGED
-
交給容器去管理事務,但目前使用的是本地程序,沒有容器的支持,當mybatis找不到容器的支持時:沒有事務。也就是說只要執行一條DML語句,則提交一次。
3.dataSource
dataSource配置:1.dataSource被稱為數據源。2.dataSource作用是什么?為程序提供Connection對象。(但凡是給程序提供Connection對象的,都叫做數據源。)3.數據源實際上是一套規范。JDK中有這套規范:javax.sql.DataSource(這個數據源的規范,這套接口實際上是JDK規定的。)4.我們自己也可以編寫數據源組件,只要實現javax.sql.DataSource接口就行了。實現接口當中所有的方法。這樣就有了自己的數據源。比如你可以寫一個屬于自己的數據庫連接池(數據庫連接池是提供連接對象的,所以數據庫連接池就是一個數據源)。5.常見的數據源組件有哪些呢【常見的數據庫連接池有哪些呢】?阿里巴巴的德魯伊連接池:druidc3p0dbcp....6. type屬性用來指定數據源的類型,就是指定具體使用什么方式來獲取Connection對象:type屬性有三個值:必須是三選一。type="[UNPOOLED|POOLED|JNDI]"UNPOOLED:不使用數據庫連接池技術。每一次請求過來之后,都是創建新的Connection對象。POOLED:使用mybatis自己實現的數據庫連接池。JNDI:集成其它第三方的數據庫連接池。JNDI是一套規范。誰實現了這套規范呢?大部分的web容器都實現了JNDI規范:例如:Tomcat、Jetty、WebLogic、WebSphere,這些服務器(容器)都實現了JNDI規范。JNDI是:java命名目錄接口。Tomcat服務器實現了這個規范。
⑴當type是UNPOOLED
UNPOOLED不會使用連接池,每一次都會新建JDBC連接對象。
test
@Testpublic void testDataSource() throws Exception{SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();// sqlSessionFactory對象:一個數據一個。不要頻繁創建該對象。SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(Resources.getResourceAsStream("mybatis-config.xml"));// 通過sqlSessionFactory對象可以開啟多個會話。// 會話1SqlSession sqlSession1 = sqlSessionFactory.openSession();sqlSession1.insert("car.insertCar");sqlSession1.commit();sqlSession1.close(); // 因為要測試連接池的效果,關閉是需要的。別忘了,要不然測不出來。// 會話2SqlSession sqlSession2 = sqlSessionFactory.openSession();sqlSession2.insert("car.insertCar");sqlSession2.commit();sqlSession2.close();}
⑵type="POOLED
修改配置文件mybatis-config3.xml中的配置:
<dataSource type="POOLED">
Java測試程序不需要修改,直接執行,看控制臺輸出:
通過測試得出:UNPOOLED不會使用連接池,每一次都會新建JDBC連接對象。POOLED會使用數據庫連接池。【這個連接池是mybatis自己實現的。】
重點說一下type="POOLED"的時候,它的屬性有哪些?
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configurationPUBLIC "-//mybatis.org//DTD Config 3.0//EN""http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration><environments default="dev"><environment id="dev"><transactionManager type="JDBC"/><dataSource type="POOLED"><property name="driver" value="com.mysql.cj.jdbc.Driver"/><property name="url" value="jdbc:mysql://localhost:3306/jdbc"/><property name="username" value="root"/><property name="password" value="abc123"/><!--最大連接數--><property name="poolMaximumActiveConnections" value="3"/><!--這是一個底層設置,如果獲取連接花費了相當長的時間,連接池會打印狀態日志并重新嘗試獲取一個連接(避免在誤配置的情況下一直失敗且不打印日志),默認值:20000 毫秒(即 20 秒)。--><property name="poolTimeToWait" value="20000"/><!--強行回歸池的時間--><property name="poolMaximumCheckoutTime" value="20000"/><!--最多空閑數量--><property name="poolMaximumIdleConnections" value="1"/></dataSource></environment></environments><mappers><mapper resource="CarMapper.xml"/></mappers>
</configuration>
poolMaximumActiveConnections:最大的活動的連接數量。默認值10 poolMaximumIdleConnections:最大的空閑連接數量。默認值5
poolMaximumCheckoutTime:強行回歸池的時間。默認值20秒。
poolTimeToWait:當無法獲取到空閑連接時,每隔20秒打印一次日志,避免因代碼配置有誤,導致傻等。(時長是可以配置的) 當然,還有其他屬性。對于連接池來說,以上幾個屬性比較重要。
最大的活動的連接數量就是連接池連接數量的上限。默認值10,如果有10個請求正在使用這10個連接,第11個請求只能等待空閑連接。 最大的空閑連接數量。默認值5,如何已經有了5個空閑連接,當第6個連接要空閑下來的時候,連接池會選擇關閉該連接對象。來減少數據庫的開銷。
需要根據系統的并發情況,來合理調整連接池最大連接數以及最多空閑數量。充分發揮數據庫連接池的性能。【可以根據實際情況進行測試,然后調整一個合理的數量。】 下圖是默認配置:
在以上配置的基礎之上,可以編寫java程序測試:
@Test
public void testPool() throws Exception{SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(Resources.getResourceAsStream("mybatis-config3.xml"));for (int i = 0; i < 4; i++) {SqlSession sqlSession = sqlSessionFactory.openSession();Object selectCarByCarNum = sqlSession.selectOne("selectCarByCarNum");}
}
<select id="selectCarByCarNum" resultType="com.powernode.mybatis.pojo.Car">select id,car_num carNum,brand,guide_price guidePrice,produce_time produceTime,car_type carType from t_car where car_num = '100'
</select>
⑶type="JNDI
NDI的方式:表示對接JNDI服務器中的連接池。這種方式給了我們可以使用第三方連接池的接口。如果想使用dbcp、c3p0、druid(德魯伊)等,需要使用這種方式。
4.properties
mybatis提供了更加靈活的配置,連接數據庫的信息可以單獨寫到一個屬性資源文件中,假設在類的根路徑下創建jdbc.properties文件,配置如下:
jdbc.properties
jdbc.driver=com.mysql.cj.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/jdbc
jdbc.username=root
jdbc.password=abc123
properties兩個屬性:
resource:這個屬性從類的根路徑下開始加載。【常用的。】
url:從指定的url加載,假設文件放在d:/jdbc.properties,這個url可以寫成:file:///d:/jdbc.properties。注意是三個斜杠哦。
注意:如果不知道mybatis-config.xml文件中標簽的編寫順序的話,可以有兩種方式知道它的順序:
-
第一種方式:查看dtd約束文件。
-
第二種方式:通過idea的報錯提示信息。【一般采用這種方式】
5.mapper
mapper標簽用來指定SQL映射文件的路徑,包含多種指定方式,這里先主要看其中兩種: 第一種:resource,從類的根路徑下開始加載【比url常用】
<mappers><mapper resource="CarMapper.xml"/>
</mappers>
如果是這樣寫的話,必須保證類的根下有CarMapper.xml文件。
如果類的根路徑下有一個包叫做test,CarMapper.xml如果放在test包下的話,這個配置應該是這樣寫:
<mappers><mapper resource="test/CarMapper.xml"/>
</mappers>
第二種:url,從指定的url位置加載
假設CarMapper.xml文件放在d盤的根下,這個配置就需要這樣寫:
<mappers><mapper url="file:///d:/CarMapper.xml"/>
</mappers>
mapper還有其他的指定方式,后面再看!!!