《MyBatis CRUD實戰與核心配置詳解:從基礎操作到高級應用》

一、使用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還有其他的指定方式,后面再看!!!

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/75461.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/75461.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/75461.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Java語言如何用AI實現文件報告的自動質檢?

文件報告的質量直接影響工作效率和決策的準確性&#xff0c;然而&#xff0c;傳統的文件質檢方式往往依賴人工審核&#xff0c;效率低下且容易出錯。那么&#xff0c;如何利用AI技術實現文件報告的自動化質檢呢&#xff1f; 問題1&#xff1a;質檢的目標是什么&#xff1f; 文…

es自定義ik分詞器中文詞庫實現熱更新

基于web地址的方式實現ik分詞熱更新。 操作系統&#xff1a;win 11 es version&#xff1a;8.6.2 ik version&#xff1a;8.6.2 1、創建web服務&#xff0c;并提供ik查詢詞庫接口 編寫分詞http url代碼&#xff0c;返回自定義分詞內容分詞詞庫數據來自業務需求&#xff0c;存…

鉑卡梭 智能羽翼 AI 系統:交易科技的未來引擎

突破性的 AI 交易系統 鉑卡梭(Pegasus)近期推出的 InnoFeather AI System(智能羽翼 AI 系統) 代表了金融科技領域的前沿突破。這一系統集成了先進的 機器學習算法、大數據分析 和 實時市場情緒感知,旨在幫助交易者在復雜多變的市場環境中做出更精準的決策。 智能羽翼 AI 系統的…

js中判斷對象是否包含某個屬性(元素)

在JavaScript中&#xff0c;判斷對象是否包含某個屬性&#xff08;元素&#xff09;主要有以下幾種方法&#xff0c;根據具體需求選擇合適的方式&#xff1a; 1. 使用 in 運算符 作用&#xff1a;檢查對象自身及原型鏈上是否存在指定屬性。 示例&#xff1a; javascript cons…

Anaconda和Pycharm的區別,以及如何選擇兩者

目錄 主要區別詳細說明如何選擇&#xff1f;Anaconda的使用步驟 主要區別 Anaconda 和 PyCharm 是 Python 開發中常用的兩個工具&#xff0c;但它們的定位和功能完全不同。以下是它們的主要區別&#xff1a; 對比項AnacondaPyCharm類型Python 發行版 包管理工具Python 集成開…

UE小:在Unreal Engine 5中實現多層靜態網格體遮擋拾取

問題描述 當需要拾取被多層靜態網格體遮擋的對象時&#xff0c;若所有網格體碰撞預設為BlockAll&#xff0c;需要通過特殊配置實現穿透檢測。 完整實現方案 1. 創建自定義追蹤通道 進入 ?項目設置&#xff08;Project Settings&#xff09; > 碰撞&#xff08;Collision…

wireshark抓包分析數據怎么看 wireshark使用教程_wireshark怎么看

Wireshark與Sniff Master&#xff1a;網絡抓包工具使用指南 網絡抓包分析是開發測試和網絡故障排查中不可或缺的技能。在眾多抓包工具中&#xff0c;Wireshark無疑是最流行且功能強大的選擇&#xff0c;而Sniff Master作為后起之秀&#xff0c;也因其簡潔高效的特點受到許多專…

密碼學基礎——古典密碼學

目錄 一、定義 特點&#xff1a; 二、發展階段 三、代換密碼 1.單表代換密碼 1.1愷撒密碼 1.2 移位變換 1.3 仿射變換 2.多表代換密碼 維吉尼亞密碼 四、置換密碼 柵欄密碼 一、定義 古典密碼學是指在現代密碼學出現之前&#xff0c;使用較為簡單的數學方法和手工…

【案例分享】江蘇某汽車制造廠水冷式制冷站AI節能優化方案

主要訴求&#xff1a; 對B系統進行AI節能優化&#xff1a;3臺離心機1臺螺桿機板式換熱器 優化前后對比&#xff1a; ? 優化前&#xff1a;根據人工經驗判斷冷機和板換的啟停&#xff0c;PLC固定邏輯調節參數 ?優化后&#xff1a;根據冷負荷對工況進行分類&#xff0c;自動…

vue+springboot 新增操作,前端id傳string,后端接收為long類型,報錯類型不匹配——解決方法

我在做新增功能的時候&#xff0c;做了一個邏輯判斷&#xff0c;當前端沒有傳遞對應的id值時&#xff0c;為新增&#xff1b;有id值時&#xff0c;為修改 問題一、&#xff1a;前端idundefined&#xff0c;和我的需求不符合&#xff0c;我要id為空&#xff0c;不傳值。最后發現…

vue前端項目技術架構(第二版)

vue技術架構介紹 如下圖所示&#xff0c;展示了項目系統的軟件層次架構。該系統采用基于SOA&#xff08;面向服務架構&#xff09;思想的分層架構&#xff0c;分為四個主要層次&#xff1a;視圖層、編譯層、代碼層和數據層。 視圖層 瀏覽器&#xff1a;核心職責是解析并展示…

在服務器里面磁盤分區很簡單嗎?

不管是我們的普通辦公電腦還是服務器都是需要硬盤的&#xff0c;硬盤的類型分機械硬盤&#xff08;HDD&#xff09;和固態硬盤&#xff08;SSD&#xff09;&#xff0c;這兩種磁盤是我們平常最常見的硬盤您他們又有什么區別 機械硬盤&#xff08;HDD&#xff09; 原理&#xff…

Pycharm(十一):字符串練習題

1.輸入一個字符串&#xff0c;打印所有偶數位上的字符(下標是0&#xff0c;2&#xff0c;4&#xff0c;6...位上的字符) # 練習題1:輸入一個字符串&#xff0c;打印所有偶數位上的字符(下標是0&#xff0c;2&#xff0c;4&#xff0c;6...位上的字符) # 1.鍵盤錄入字符串&…

虛幻5入門

常用操作 運行時&#xff0c;調試相機&#xff0c;按~鍵&#xff0c;輸入ToggleDebugCamera 。進入自由視角 常用節點 gate節點&#xff1a;用于控制該流程通不通&#xff0c;執不執行。Flip Flop節點&#xff1a;反轉執行&#xff0c;一次A&#xff0c;一次B。Set Timer by…

Prompt Flow 與 DSPy:大型語言模型開發的未來

作為一名大模型算法工程師&#xff0c;我一直在探索如何更高效地開發和優化基于大型語言模型&#xff08;LLM&#xff09;的應用程序。隨著模型規模的不斷擴大和應用場景的日益復雜&#xff0c;傳統的開發方式已經難以滿足高效、靈活和可擴展的需求。幸運的是&#xff0c;Promp…

Prompt攻擊是什么

什么是Prompt攻擊 Prompt攻擊(Prompt Injection/Attack) 是指通過精心構造的輸入提示(Prompt),誘導大語言模型(LLM)突破預設安全限制、泄露敏感信息或執行惡意操作的攻擊行為。其本質是利用模型對自然語言的理解漏洞,通過語義欺騙繞過防護機制。 Prompt攻擊的精髓:學…

Python 數據類型 - 集合(set)

Python 數據類型 - 集合(set) 1. 集合簡介 集合(set)是Python中的一種無序、可變、不重復元素的容器數據類型。集合的主要用途&#xff1a; 快速成員檢測&#xff08;判斷元素是否存在&#xff09;去除重復項執行數學集合運算&#xff08;并集、交集、差集等&#xff09; 2.…

Profibus DP主站轉Modbus TCP網關配置文件制作

Profibus DP主站轉Modbus TCP網關配置文件制作 1、首先打開配置文件制作軟件SST Profibus Configuration&#xff1a; 2、點擊“Library”選擇GSD文件所在文件夾找到后點擊打開導入GSD文件: 3、點開Masters、molex選擇主站設備&#xff1a; 4、雙擊添加后地址保存默認0就可以&…

如何批量拆分Excel工作表或按行拆分Excel表格 - Excel拆分器使用方法

在數據分析和處理的日常工作中&#xff0c;Excel文件因其強大的數據管理和計算能力而廣受歡迎。然而&#xff0c;當面對龐大的Excel工作簿&#xff0c;特別是需要將其拆分為多個獨立文件時&#xff0c;傳統的操作方法往往會顯得繁瑣且效率低下。為了解決這一難題&#xff0c;Ex…

JavaScript基礎-移動端常見特效

隨著移動互聯網的發展&#xff0c;為移動設備優化的網頁變得越來越重要。JavaScript在實現移動端特有的交互體驗中扮演著關鍵角色。本文將介紹幾種常見的移動端特效&#xff0c;并通過具體的代碼示例展示如何使用JavaScript和相關技術來創建這些特效。 一、手勢識別 &#xf…