“Do not understand to ask”, is a new series, the main arrangement of my small book group encountered some more interesting/difficult/easy to discuss the problem, and give the problem analysis and solution, etc. Like the small friends can click to pay attention to my duck ~ ~ learn source code can see my small book ~ ~
Dragon-boat festival holiday, believe a lot of friends all in secretly learning (say holiday play, results and learning) behind my back, it’s not, just after the Dragon Boat Festival, I in the circle of a sand sculpture program apes have someone to talk about it, this problem seems quite complicated to chat, but the problem is actually very simple, the za to discuss this problem.
The original problem
Does MyBatis level 1 cache conflict with SpringFramework declarative transactions? The transaction is started in the Service and the same data is queried twice. However, the two query results are inconsistent.
Mapper selectById = selectById = selectById = selectById = selectById = selectById = selectById
If the transaction is not enabled, the result of the two requests is the same, and the console prints the SQL twice.
The preliminary analysis
Reasonable, see this problem, I immediately guessed that MyBatis level 1 cache repeated read the problem.
MyBatis level 1 cache is enabled by default and belongs to SqlSession scope. During the duration of the transaction, the database is queried only once for the same database query request, and subsequent repeated queries are retrieved from the level 1 cache. When the transaction is not enabled, the database request is sent for the same multiple database queries.
The above all belong to the basic knowledge, not much explanation. The important thing is that the entities he changes are directly queried from MyBatis’s level 1 cache. As we all know, these entities must belong to objects, and we get a reference to the object. If we change this in the Service, the level 1 cache will be affected. Therefore, the core cause of this problem is easy to find.
Problem of repetition
To illustrate the problem, let’s briefly recreate the scene.
Engineering structures,
I use SpringBoot + mybatis – spring – the boot – starter to quickly build a project, here SpringBoot version for 2.2.8, mybatis – spring – the boot for 2.1.2 – the starter version.
pom
There are three core POM dependencies:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
<version>2.1.2</version>
</dependency>
<dependency>
<groupId>com.h2database</groupId>
<artifactId>h2</artifactId>
<version>1.4.199</version>
</dependency>
Copy the code
Database configuration
We will continue to use H2 as the database for rapid problem recurrence, and just add the following configuration to application.properties to initialize an H2 database. MyBatis is configured as follows:
spring.datasource.driver-class-name=org.h2.Driver
spring.datasource.url=jdbc:h2:mem:mybatis-transaction-cache
spring.datasource.username=sa
spring.datasource.password=sa
spring.datasource.platform=h2
spring.datasource.schema=classpath:sql/schema.sql
spring.datasource.data=classpath:sql/data.sql
spring.h2.console.settings.web-allow-others=true
spring.h2.console.path=/h2
spring.h2.console.enabled=true
mybatis.type-aliases-package=com.linkedbear.demo.entity
mybatis.mapper-locations=classpath:mapper/*.xml
Copy the code
Initializing the database
The datasource’s schema and data are initialized using the datasource’s schema and data.
Schema. SQL:
create table if not exists sys_department (
id varchar(32) not null primary key.name varchar(32) not null
);
Copy the code
Data. SQL:
insert into sys_department (id.name) values ('idaaa'.'testaaa');
insert into sys_department (id.name) values ('idbbb'.'testbbb');
insert into sys_department (id.name) values ('idccc'.'testccc');
insert into sys_department (id.name) values ('idddd'.'testddd');
Copy the code
Writing test code
We used a simple single table model to quickly reproduce the scene.
entity
Create a new Department class and declare the ID and name attributes:
public class Department {
private String id;
private String name;
// getter setter toString ......
}
Copy the code
mapper
MyBatis provides dynamic proxy for statement (s), and findById (s) can be used for statement (s).
@Mapper
public interface DepartmentMapper {
Department findById(String id);
}
Copy the code
mapper.xml
Correspondingly, the interface requires XML as a reference :(annotated Mapper is not used here)
<?xml version="1.0" encoding="UTF-8" ? >
<mapper namespace="com.linkedbear.demo.mapper.DepartmentMapper">
<select id="findById" parameterType="string" resultType="department">
select * from sys_department where id = #{id}
</select>
</mapper>
Copy the code
service
Insert Mapper into Service and write a transaction-required update method to simulate the update action:
@Service
public class DepartmentService {
@Autowired
DepartmentMapper departmentMapper;
@Transactional(rollbackFor = Exception.class)
public Department update(Department department) {
Department temp = departmentMapper.findById(department.getId());
temp.setName(department.getName());
Department temp2 = departmentMapper.findById(department.getId());
System.out.println("Do two queries result in the same object?" + temp == temp2);
returntemp; }}Copy the code
controller
Inject the Service into the Controller and call the update method on the Service to trigger the test:
@RestController
public class DepartmentController {
@Autowired
DepartmentService departmentService;
@GetMapping("/department/{id}")
public Department findById(@PathVariable("id") String id) {
Department department = new Department();
department.setId(id);
department.setName(UUID.randomUUID().toString().replaceAll("-".""));
returndepartmentService.update(department); }}Copy the code
The main start class
There’s nothing special in the main startup class, just remember to start the transaction:
@EnableTransactionManagement
@SpringBootApplication
public class MyBatisTransactionCacheApplication {
public static void main(String[] args) { SpringApplication.run(MyBatisTransactionCacheApplication.class, args); }}Copy the code
Run the test
Open the admin console for the H2 database by running SpringBoot’s primary boot class in Debug mode and entering http://localhost:8080/h2 in your browser with the configuration just declared in application.properties.
SELECT * FROM SYS_DEPARTMENT; SELECT * FROM SYS_DEPARTMENT;
Below test result, enter http://localhost:8080/department/idaaa in the browser, the console print result is true, prove MyBatis level cache effect, two queries the resulting entity class object.
The solution
The solution to this problem, to put it bluntly, is to turn off level 1 caching. Here are some of the most common solutions:
- Global Off: Set
mybatis.configuration.local-cache-scope=statement
- Specify mapper off: In
mapper.xml
In the specified statementflushCache="true"
- Alternative: Add a random number to the statement SQL (too unconventional…)
select * from sys_department where #{random} = #{random}
The principle of extended
In fact, at this point, the problem has been solved, but don’t worry, consider a question: Why is level-1 caching disabled when local-cache-scope is declared as statement or mapper’s Statement tag is set to flushCache=true? Here’s how it works.
Principle of level 1 cache invalidation
In DepartmentService, perform the mapper.findByID action and eventually enter the selectOne of DefaultSqlSession:
public <T> T selectOne(String statement) {
return this.selectOne(statement, null);
}
@Override
public <T> T selectOne(String statement, Object parameter) {
// Popular vote was to return null on 0 results and throw exception on too many.
List<T> list = this.selectList(statement, parameter);
if (list.size() == 1) {
return list.get(0);
} else if (list.size() > 1) {
throw new TooManyResultsException("Expected one result (or null) to be returned by selectOne(), but found: " + list.size());
} else {
return null; }}Copy the code
You can see that the bottom line of selectOne is the selectList that was called, and then get(0) fetched the first data and returned it.
The underlying selectList has two steps: obtain the MappedStatement → execute the query, as shown in the try section of the code below:
public <E> List<E> selectList(String statement, Object parameter, RowBounds rowBounds) {
try {
MappedStatement ms = configuration.getMappedStatement(statement);
return executor.query(ms, wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER);
} catch (Exception e) {
throw ExceptionFactory.wrapException("Error querying database. Cause: " + e, e);
} finally{ ErrorContext.instance().reset(); }}Copy the code
Execute the query method, which goes into BaseExecutor, and it performs three steps: get the precompiled SQL → create the cache key → actual query.
public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException {
BoundSql boundSql = ms.getBoundSql(parameter);
CacheKey key = createCacheKey(ms, parameter, rowBounds, boundSql);
return query(ms, parameter, rowBounds, resultHandler, key, boundSql);
}
Copy the code
The cache key is designed, and its structure can be simply viewed as “statementId + SQL + parameter”. According to these three elements, a query result can be uniquely determined.
When you get to the query method here, it performs the actual query action with the cache key, as shown in this long source code:
public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {
ErrorContext.instance().resource(ms.getResource()).activity("executing a query").object(ms.getId());
if (closed) {
throw new ExecutorException("Executor was closed.");
}
// If statement is set to flushCache="true", clear level-1 cache before querying
if (queryStack == 0 && ms.isFlushCacheRequired()) {
clearLocalCache();
}
List<E> list;
try {
queryStack++;
// Check the level 1 cache first
list = resultHandler == null ? (List<E>) localCache.getObject(key) : null;
if(list ! =null) {
// If it is in the level 1 cache, it is fetched directly
handleLocallyCachedOutputParameters(ms, key, parameter, boundSql);
} else {
// If the level-1 cache does not exist, query the databaselist = queryFromDatabase(ms, parameter, rowBounds, resultHandler, key, boundSql); }}finally {
queryStack--;
}
if (queryStack == 0) {
for (DeferredLoad deferredLoad : deferredLoads) {
deferredLoad.load();
}
// issue #601
deferredLoads.clear();
// If local-cache-scope=statement is set in the global configuration, the level-1 cache is cleared
if (configuration.getLocalCacheScope() == LocalCacheScope.STATEMENT) {
// issue #482clearLocalCache(); }}return list;
}
Copy the code
In the above comment, you can see that if any of the above three solutions are configured, the level 1 cache will be invalid.
- Global Settings
local-cache-scope=statement
, even after the query into the level one cache, but stored immediately to clear, the next time or to check the database; - The statement is set
flushCache="true"
If the level 1 cache is cleared before the query, the database is still checked. - Set a random number. If the upper limit of the random number is large enough, then the probability of random to the same number is low enough, and it can also be regarded as different database requests, then the cache keys are different, and naturally will not match the cache.
All the source code covered in this article can be found on GitHub: github.com/LinkedBear/…
[all see here, friends do not want to focus on the like ah, there are source learning need to see my little book oh, learn ~ ollie to]