MyBatis 架构与原理

沙海 2021年5月24日02:37:23Java评论115字数 11327阅读37分45秒阅读模式
摘要

速读摘要

速读摘要文章源自JAVA秀-https://www.javaxiu.com/25189.html

接口层一接收到调用请求就会调用数据处理层来完成具体的数据处理。Statement操作,负责对JDBCstatement的操作,如设置参数、将Statement结果集转换成List集合。欢迎加入我的知识星球,一起探讨架构,交流源码。MyBatis、Redis、MongoDB、ES、分库分表、读写分离、SpringMVC、Webflux、权限、WebSocket、Dubbo、RabbitMQ、RocketMQ、Kafka、性能测试等等内容。文章源自JAVA秀-https://www.javaxiu.com/25189.html

原文约 7511 | 图片 11 | 建议阅读 16 分钟 | 评价反馈文章源自JAVA秀-https://www.javaxiu.com/25189.html

MyBatis 架构与原理

点击关注 ? 芋道源码 文章源自JAVA秀-https://www.javaxiu.com/25189.html

收录于话题文章源自JAVA秀-https://www.javaxiu.com/25189.html

#芋道源码文章源自JAVA秀-https://www.javaxiu.com/25189.html

83个文章源自JAVA秀-https://www.javaxiu.com/25189.html

点击上方“芋道源码”,选择“设为星标文章源自JAVA秀-https://www.javaxiu.com/25189.html

管她前浪,还是后浪?文章源自JAVA秀-https://www.javaxiu.com/25189.html

能浪的浪,才是好浪!文章源自JAVA秀-https://www.javaxiu.com/25189.html

每天 8:55 更新文章,每天掉亿点点头发...文章源自JAVA秀-https://www.javaxiu.com/25189.html

文章源自JAVA秀-https://www.javaxiu.com/25189.html

源码精品专栏文章源自JAVA秀-https://www.javaxiu.com/25189.html

 文章源自JAVA秀-https://www.javaxiu.com/25189.html

文章源自JAVA秀-https://www.javaxiu.com/25189.html

来源:jianshu.com/p/15781ec742f2文章源自JAVA秀-https://www.javaxiu.com/25189.html

MyBatis功能架构设计

MyBatis 架构与原理文章源自JAVA秀-https://www.javaxiu.com/25189.html

我们把Mybatis的功能架构分为三层:文章源自JAVA秀-https://www.javaxiu.com/25189.html

(1)API接口层:提供给外部使用的接口API,开发人员通过这些本地API来操纵数据库。接口层一接收到调用请求就会调用数据处理层来完成具体的数据处理。文章源自JAVA秀-https://www.javaxiu.com/25189.html

(2)数据处理层:负责具体的SQL查找、SQL解析、SQL执行和执行结果映射处理等。它主要的目的是根据调用的请求完成一次数据库操作。文章源自JAVA秀-https://www.javaxiu.com/25189.html

(3)基础支撑层:负责最基础的功能支撑,包括连接管理、事务管理、配置加载和缓存处理,这些都是共用的东西,将他们抽取出来作为最基础的组件。为上层的数据处理层提供最基础的支撑。文章源自JAVA秀-https://www.javaxiu.com/25189.html

框架架构

MyBatis 架构与原理文章源自JAVA秀-https://www.javaxiu.com/25189.html

这张图从上往下看。MyBatis的初始化,会从mybatis-config.xml配置文件,解析构造成Configuration这个类,就是图中的红框。文章源自JAVA秀-https://www.javaxiu.com/25189.html

(1)加载配置:配置来源于两个地方,一处是配置文件,一处是Java代码的注解,将SQL的配置信息加载成为一个个MappedStatement对象(包括了传入参数映射配置、执行的SQL语句、结果映射配置),存储在内存中。文章源自JAVA秀-https://www.javaxiu.com/25189.html

(2)SQL解析:当API接口层接收到调用请求时,会接收到传入SQL的ID和传入对象(可以是Map、JavaBean或者基本数据类型),Mybatis会根据SQL的ID找到对应的MappedStatement,然后根据传入参数对象对MappedStatement进行解析,解析后可以得到最终要执行的SQL语句和参数。文章源自JAVA秀-https://www.javaxiu.com/25189.html

(3)SQL执行:将最终得到的SQL和参数拿到数据库进行执行,得到操作数据库的结果。文章源自JAVA秀-https://www.javaxiu.com/25189.html

(4)结果映射:将操作数据库的结果按照映射的配置进行转换,可以转换成HashMap、JavaBean或者基本数据类型,并将最终结果返回。文章源自JAVA秀-https://www.javaxiu.com/25189.html

MyBatis核心类

每一个MyBatis的应用程序的入口是SqlSessionFactoryBuilder。文章源自JAVA秀-https://www.javaxiu.com/25189.html

它的作用是通过XML配置文件创建Configuration对象(当然也可以在程序中自行创建),然后通过build方法创建SqlSessionFactory对象。没有必要每次访问Mybatis就创建一次SqlSessionFactoryBuilder,通常的做法是创建一个全局的对象就可以了。示例程序如下:文章源自JAVA秀-https://www.javaxiu.com/25189.html

private static SqlSessionFactoryBuilder sqlSessionFactoryBuilder;private static SqlSessionFactory sqlSessionFactory;private static void init() throws IOException {        String resource = "mybatis-config.xml";        Reader reader = Resources.getResourceAsReader(resource);        sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();        sqlSessionFactory = sqlSessionFactoryBuilder.build(reader);        }        org.apache.ibatis.session.Configuration 是mybatis初始化的核心。

mybatis-config.xml中的配置,最后会解析xml成Configuration这个类。文章源自JAVA秀-https://www.javaxiu.com/25189.html

SqlSessionFactoryBuilder根据传入的数据流(XML)生成Configuration对象,然后根据Configuration对象创建默认的SqlSessionFactory实例。文章源自JAVA秀-https://www.javaxiu.com/25189.html

它的主要功能是创建SqlSession对象,和SqlSessionFactoryBuilder对象一样,没有必要每次访问Mybatis就创建一次SqlSessionFactory,通常的做法是创建一个全局的对象就可以了。SqlSessionFactory对象一个必要的属性是Configuration对象,它是保存Mybatis全局配置的一个配置对象,通常由SqlSessionFactoryBuilder从XML配置文件创建。这里给出一个简单的示例:文章源自JAVA秀-https://www.javaxiu.com/25189.html

<?xml version="1.0" encoding="UTF-8" ?><!DOCTYPE configuration PUBLIC        "-//mybatis.org//DTD Config 3.0//EN"        "http://mybatis.org/dtd/mybatis-3-config.dtd"><configuration>    <!-- 配置别名 -->    <typeAliases>        <typeAlias type="org.iMybatis.abc.dao.UserDao" alias="UserDao" />        <typeAlias type="org.iMybatis.abc.dto.UserDto" alias="UserDto" />    </typeAliases>    <!-- 配置环境变量 -->    <environments default="development">        <environment id="development">            <transactionManager type="JDBC" />            <dataSource type="POOLED">                <property name="driver" value="com.mysql.jdbc.Driver" />                <property name="url" value="jdbc:mysql://127.0.0.1:3306/iMybatis?characterEncoding=GBK" />                <property name="username" value="iMybatis" />                <property name="password" value="iMybatis" />            </dataSource>        </environment>    </environments>    <!-- 配置mappers -->    <mappers>        <mapper resource="org/iMybatis/abc/dao/UserDao.xml" />    </mappers></configuration>

SqlSession对象的主要功能是完成一次数据库的访问和结果的映射,它类似于数据库的session概念,由于不是线程安全的,所以SqlSession对象的作用域需限制方法内。SqlSession的默认实现类是DefaultSqlSession,它有两个必须配置的属性:Configuration和Executor。Configuration前文已经描述这里不再多说。SqlSession对数据库的操作都是通过Executor来完成的。文章源自JAVA秀-https://www.javaxiu.com/25189.html

SqlSession :默认创建DefaultSqlSession 并且开启一级缓存,创建执行器 、赋值。文章源自JAVA秀-https://www.javaxiu.com/25189.html

SqlSession有一个重要的方法getMapper,顾名思义,这个方式是用来获取Mapper对象的。什么是Mapper对象?根据Mybatis的官方手册,应用程序除了要初始并启动Mybatis之外,还需要定义一些接口,接口里定义访问数据库的方法,存放接口的包路径下需要放置同名的XML配置文件。文章源自JAVA秀-https://www.javaxiu.com/25189.html

SqlSession的getMapper方法是联系应用程序和Mybatis纽带,应用程序访问getMapper时,Mybatis会根据传入的接口类型和对应的XML配置文件生成一个代理对象,这个代理对象就叫Mapper对象。应用程序获得Mapper对象后,就应该通过这个Mapper对象来访问Mybatis的SqlSession对象,这样就达到里插入到Mybatis流程的目的。文章源自JAVA秀-https://www.javaxiu.com/25189.html

SqlSession session= sqlSessionFactory.openSession();        UserDao userDao = session.getMapper(UserDao.class);        UserDto user = new UserDto();        user.setUsername("iMybatis");        List<UserDto> users = userDao.queryUsers(user);    public interface UserDao {    public List<UserDto> queryUsers(UserDto user) throws Exception;        }        <?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="org.iMybatis.abc.dao.UserDao">            <select id="queryUsers" parameterType="UserDto" resultType="UserDto"                    useCache="false">                <![CDATA[        select * from t_user t where t.username = #{username}        ]]>            </select>        </mapper>

Executor对象在创建Configuration对象的时候创建,并且缓存在Configuration对象里。Executor对象的主要功能是调用StatementHandler访问数据库,并将查询结果存入缓存中(如果配置了缓存的话)。文章源自JAVA秀-https://www.javaxiu.com/25189.html

StatementHandler是真正访问数据库的地方,并调用ResultSetHandler处理查询结果。文章源自JAVA秀-https://www.javaxiu.com/25189.html

处理查询结果。文章源自JAVA秀-https://www.javaxiu.com/25189.html

MyBatis成员层次&职责

MyBatis 架构与原理文章源自JAVA秀-https://www.javaxiu.com/25189.html

  1. SqlSession 作为MyBatis工作的主要顶层API,表示和数据库交互的会话,完成必要数据库增删改查功能文章源自JAVA秀-https://www.javaxiu.com/25189.html

  2. Executor MyBatis执行器,是MyBatis 调度的核心,负责SQL语句的生成和查询缓存的维护文章源自JAVA秀-https://www.javaxiu.com/25189.html

  3. StatementHandler 封装了JDBC Statement操作,负责对JDBCstatement的操作,如设置参数、将Statement结果集转换成List集合。文章源自JAVA秀-https://www.javaxiu.com/25189.html

  4. ParameterHandler 负责对用户传递的参数转换成JDBC Statement 所需要的参数文章源自JAVA秀-https://www.javaxiu.com/25189.html

  5. ResultSetHandler *负责将JDBC返回的ResultSet结果集对象转换成List类型的集合;文章源自JAVA秀-https://www.javaxiu.com/25189.html

  6. TypeHandler 负责java数据类型和jdbc数据类型之间的映射和转换文章源自JAVA秀-https://www.javaxiu.com/25189.html

  7. MappedStatement MappedStatement维护了一条<select|update|delete|insert>节点的封文章源自JAVA秀-https://www.javaxiu.com/25189.html

  8. SqlSource 负责根据用户传递的parameterObject,动态地生成SQL语句,将信息封装到BoundSql对象中,并返回文章源自JAVA秀-https://www.javaxiu.com/25189.html

  9. BoundSql 表示动态生成的SQL语句以及相应的参数信息文章源自JAVA秀-https://www.javaxiu.com/25189.html

  10. Configuration MyBatis所有的配置信息都维持在Configuration对象之中文章源自JAVA秀-https://www.javaxiu.com/25189.html

title: Mybatis架构与原理 date: 2021-05-23 tags: categories: 精进 permalink: Fight/Mybatis-architecture-and-principle author: 消失er from_url: jianshu.com/p/15781ec742f2 wechat_url:文章源自JAVA秀-https://www.javaxiu.com/25189.html

MyBatis功能架构设计

MyBatis 架构与原理文章源自JAVA秀-https://www.javaxiu.com/25189.html

我们把Mybatis的功能架构分为三层:文章源自JAVA秀-https://www.javaxiu.com/25189.html

(1)API接口层:提供给外部使用的接口API,开发人员通过这些本地API来操纵数据库。接口层一接收到调用请求就会调用数据处理层来完成具体的数据处理。文章源自JAVA秀-https://www.javaxiu.com/25189.html

(2)数据处理层:负责具体的SQL查找、SQL解析、SQL执行和执行结果映射处理等。它主要的目的是根据调用的请求完成一次数据库操作。文章源自JAVA秀-https://www.javaxiu.com/25189.html

(3)基础支撑层:负责最基础的功能支撑,包括连接管理、事务管理、配置加载和缓存处理,这些都是共用的东西,将他们抽取出来作为最基础的组件。为上层的数据处理层提供最基础的支撑。文章源自JAVA秀-https://www.javaxiu.com/25189.html

框架架构

MyBatis 架构与原理文章源自JAVA秀-https://www.javaxiu.com/25189.html

这张图从上往下看。MyBatis的初始化,会从mybatis-config.xml配置文件,解析构造成Configuration这个类,就是图中的红框。文章源自JAVA秀-https://www.javaxiu.com/25189.html

(1)加载配置:配置来源于两个地方,一处是配置文件,一处是Java代码的注解,将SQL的配置信息加载成为一个个MappedStatement对象(包括了传入参数映射配置、执行的SQL语句、结果映射配置),存储在内存中。文章源自JAVA秀-https://www.javaxiu.com/25189.html

(2)SQL解析:当API接口层接收到调用请求时,会接收到传入SQL的ID和传入对象(可以是Map、JavaBean或者基本数据类型),Mybatis会根据SQL的ID找到对应的MappedStatement,然后根据传入参数对象对MappedStatement进行解析,解析后可以得到最终要执行的SQL语句和参数。文章源自JAVA秀-https://www.javaxiu.com/25189.html

(3)SQL执行:将最终得到的SQL和参数拿到数据库进行执行,得到操作数据库的结果。文章源自JAVA秀-https://www.javaxiu.com/25189.html

(4)结果映射:将操作数据库的结果按照映射的配置进行转换,可以转换成HashMap、JavaBean或者基本数据类型,并将最终结果返回。文章源自JAVA秀-https://www.javaxiu.com/25189.html

MyBatis核心类

每一个MyBatis的应用程序的入口是SqlSessionFactoryBuilder。文章源自JAVA秀-https://www.javaxiu.com/25189.html

它的作用是通过XML配置文件创建Configuration对象(当然也可以在程序中自行创建),然后通过build方法创建SqlSessionFactory对象。没有必要每次访问Mybatis就创建一次SqlSessionFactoryBuilder,通常的做法是创建一个全局的对象就可以了。示例程序如下:文章源自JAVA秀-https://www.javaxiu.com/25189.html

private static SqlSessionFactoryBuilder sqlSessionFactoryBuilder;private static SqlSessionFactory sqlSessionFactory;private static void init() throws IOException {        String resource = "mybatis-config.xml";        Reader reader = Resources.getResourceAsReader(resource);        sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();        sqlSessionFactory = sqlSessionFactoryBuilder.build(reader);        }        org.apache.ibatis.session.Configuration 是mybatis初始化的核心。

mybatis-config.xml中的配置,最后会解析xml成Configuration这个类。文章源自JAVA秀-https://www.javaxiu.com/25189.html

SqlSessionFactoryBuilder根据传入的数据流(XML)生成Configuration对象,然后根据Configuration对象创建默认的SqlSessionFactory实例。文章源自JAVA秀-https://www.javaxiu.com/25189.html

它的主要功能是创建SqlSession对象,和SqlSessionFactoryBuilder对象一样,没有必要每次访问Mybatis就创建一次SqlSessionFactory,通常的做法是创建一个全局的对象就可以了。SqlSessionFactory对象一个必要的属性是Configuration对象,它是保存Mybatis全局配置的一个配置对象,通常由SqlSessionFactoryBuilder从XML配置文件创建。这里给出一个简单的示例:文章源自JAVA秀-https://www.javaxiu.com/25189.html

<?xml version="1.0" encoding="UTF-8" ?><!DOCTYPE configuration PUBLIC        "-//mybatis.org//DTD Config 3.0//EN"        "http://mybatis.org/dtd/mybatis-3-config.dtd"><configuration>    <!-- 配置别名 -->    <typeAliases>        <typeAlias type="org.iMybatis.abc.dao.UserDao" alias="UserDao" />        <typeAlias type="org.iMybatis.abc.dto.UserDto" alias="UserDto" />    </typeAliases>    <!-- 配置环境变量 -->    <environments default="development">        <environment id="development">            <transactionManager type="JDBC" />            <dataSource type="POOLED">                <property name="driver" value="com.mysql.jdbc.Driver" />                <property name="url" value="jdbc:mysql://127.0.0.1:3306/iMybatis?characterEncoding=GBK" />                <property name="username" value="iMybatis" />                <property name="password" value="iMybatis" />            </dataSource>        </environment>    </environments>    <!-- 配置mappers -->    <mappers>        <mapper resource="org/iMybatis/abc/dao/UserDao.xml" />    </mappers></configuration>

SqlSession对象的主要功能是完成一次数据库的访问和结果的映射,它类似于数据库的session概念,由于不是线程安全的,所以SqlSession对象的作用域需限制方法内。SqlSession的默认实现类是DefaultSqlSession,它有两个必须配置的属性:Configuration和Executor。Configuration前文已经描述这里不再多说。SqlSession对数据库的操作都是通过Executor来完成的。文章源自JAVA秀-https://www.javaxiu.com/25189.html

SqlSession :默认创建DefaultSqlSession 并且开启一级缓存,创建执行器 、赋值。文章源自JAVA秀-https://www.javaxiu.com/25189.html

SqlSession有一个重要的方法getMapper,顾名思义,这个方式是用来获取Mapper对象的。什么是Mapper对象?根据Mybatis的官方手册,应用程序除了要初始并启动Mybatis之外,还需要定义一些接口,接口里定义访问数据库的方法,存放接口的包路径下需要放置同名的XML配置文件。文章源自JAVA秀-https://www.javaxiu.com/25189.html

SqlSession的getMapper方法是联系应用程序和Mybatis纽带,应用程序访问getMapper时,Mybatis会根据传入的接口类型和对应的XML配置文件生成一个代理对象,这个代理对象就叫Mapper对象。应用程序获得Mapper对象后,就应该通过这个Mapper对象来访问Mybatis的SqlSession对象,这样就达到里插入到Mybatis流程的目的。文章源自JAVA秀-https://www.javaxiu.com/25189.html

SqlSession session= sqlSessionFactory.openSession();        UserDao userDao = session.getMapper(UserDao.class);        UserDto user = new UserDto();        user.setUsername("iMybatis");        List<UserDto> users = userDao.queryUsers(user);    public interface UserDao {    public List<UserDto> queryUsers(UserDto user) throws Exception;        }        <?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="org.iMybatis.abc.dao.UserDao">            <select id="queryUsers" parameterType="UserDto" resultType="UserDto"                    useCache="false">                <![CDATA[        select * from t_user t where t.username = #{username}        ]]>            </select>        </mapper>

Executor对象在创建Configuration对象的时候创建,并且缓存在Configuration对象里。Executor对象的主要功能是调用StatementHandler访问数据库,并将查询结果存入缓存中(如果配置了缓存的话)。文章源自JAVA秀-https://www.javaxiu.com/25189.html

StatementHandler是真正访问数据库的地方,并调用ResultSetHandler处理查询结果。文章源自JAVA秀-https://www.javaxiu.com/25189.html

处理查询结果。文章源自JAVA秀-https://www.javaxiu.com/25189.html

MyBatis成员层次&职责

MyBatis 架构与原理文章源自JAVA秀-https://www.javaxiu.com/25189.html

  1. SqlSession 作为MyBatis工作的主要顶层API,表示和数据库交互的会话,完成必要数据库增删改查功能文章源自JAVA秀-https://www.javaxiu.com/25189.html

  2. Executor MyBatis执行器,是MyBatis 调度的核心,负责SQL语句的生成和查询缓存的维护文章源自JAVA秀-https://www.javaxiu.com/25189.html

  3. StatementHandler 封装了JDBC Statement操作,负责对JDBCstatement的操作,如设置参数、将Statement结果集转换成List集合。文章源自JAVA秀-https://www.javaxiu.com/25189.html

  4. ParameterHandler 负责对用户传递的参数转换成JDBC Statement 所需要的参数文章源自JAVA秀-https://www.javaxiu.com/25189.html

  5. ResultSetHandler *负责将JDBC返回的ResultSet结果集对象转换成List类型的集合;文章源自JAVA秀-https://www.javaxiu.com/25189.html

  6. TypeHandler 负责java数据类型和jdbc数据类型之间的映射和转换文章源自JAVA秀-https://www.javaxiu.com/25189.html

  7. MappedStatement MappedStatement维护了一条<select|update|delete|insert>节点的封文章源自JAVA秀-https://www.javaxiu.com/25189.html

  8. SqlSource 负责根据用户传递的parameterObject,动态地生成SQL语句,将信息封装到BoundSql对象中,并返回文章源自JAVA秀-https://www.javaxiu.com/25189.html

  9. BoundSql 表示动态生成的SQL语句以及相应的参数信息文章源自JAVA秀-https://www.javaxiu.com/25189.html

  10. Configuration MyBatis所有的配置信息都维持在Configuration对象之中文章源自JAVA秀-https://www.javaxiu.com/25189.html

欢迎加入我的知识星球,一起探讨架构,交流源码。加入方式,长按下方二维码噢文章源自JAVA秀-https://www.javaxiu.com/25189.html

MyBatis 架构与原理文章源自JAVA秀-https://www.javaxiu.com/25189.html

已在知识星球更新源码解析如下:文章源自JAVA秀-https://www.javaxiu.com/25189.html

MyBatis 架构与原理文章源自JAVA秀-https://www.javaxiu.com/25189.html

MyBatis 架构与原理文章源自JAVA秀-https://www.javaxiu.com/25189.html

MyBatis 架构与原理文章源自JAVA秀-https://www.javaxiu.com/25189.html

MyBatis 架构与原理文章源自JAVA秀-https://www.javaxiu.com/25189.html

最近更新《芋道 SpringBoot 2.X 入门》系列,已经 20 余篇,覆盖了 MyBatis、Redis、MongoDB、ES、分库分表、读写分离、SpringMVC、Webflux、权限、WebSocket、Dubbo、RabbitMQ、RocketMQ、Kafka、性能测试等等内容。文章源自JAVA秀-https://www.javaxiu.com/25189.html

提供近 3W 行代码的 SpringBoot 示例,以及超 4W 行代码的电商微服务项目。文章源自JAVA秀-https://www.javaxiu.com/25189.html

获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。文章源自JAVA秀-https://www.javaxiu.com/25189.html

文章源自JAVA秀-https://www.javaxiu.com/25189.html

文章有帮助的话,在看,转发吧。谢谢支持哟 (*^__^*)
文章源自JAVA秀-https://www.javaxiu.com/25189.html

阅读原文文章源自JAVA秀-https://www.javaxiu.com/25189.html

继续阅读
速蛙云 - 极致体验,强烈推荐!!!购买套餐就免费送各大视频网站会员!快速稳定、独家福利社、流媒体稳定解锁!速度快,全球上网、视频、游戏加速、独立IP均支持!基础套餐性价比很高!这里不多说,我一直正在使用,推荐购买:https://www.javaxiu.com/59919.html
weinxin
资源分享QQ群
本站是JAVA秀团队的技术分享社区, 会经常分享资源和教程; 分享的时代, 请别再沉默!
沙海
  • 版权声明:本站是JAVA秀团队的技术分享社区,我们会经常分享资源和教程。
  • 转载请注明:MyBatis 架构与原理 - JAVA秀 ☜(ˆ▽ˆ)
匿名

发表评论

匿名网友 填写信息

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定