mysql 8 添加虚拟列 作者: juoliii 时间: 2024-03-25 分类: 数据库 评论 ```sql ALTER TABLE toolbox_littleqc ADD COLUMN result varchar(100) AS ( case when gc<0.395 or gc>0.425 then '不合格' when pct>0.048 then '不合格' when meanInsertSize>500 or meanInsertSize<300 then '不合格' when dups>0.28 then '不合格' when mappedLambda>0.05 then '不合格' when chrxy>=3.5 and chrxy<=10 then '不合格' when mappedAllbam<95 then '不合格' else '合格' end ) VIRTUAL; ```
mysql8以下版本行列转置_mysql 转置 group_concat 作者: juoliii 时间: 2023-05-10 分类: 开发,数据库 评论 一、多行转多列: 1、什么是行转列 所谓的行转列是指把数据表中具有相同key值的多行value数据,转换为使用一个key值的多列数据,使每一行数据中,一个key对应多个value。 行转列完成后,在视觉上的效果就是:表中的总行数减少了,但是列数增加了。 如下所示的转换过程就是一个简单的行转列的过程: ![](http://www.netman.vip/blog/usr/uploads/2023/05/2714121753.png) 2、行转列实验示例 在进行实验之前,我们需要先准备好我们的实验环境,准备好表和表中的初始化数据。 准备初始化表结构 ``` CREATE TABLE `student_x` ( `id` int(11) DEFAULT NULL, `name` varchar(255) DEFAULT NULL, `class` varchar(255) DEFAULT NULL, `score` int(255) DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; ``` 准备初始化表中的数据 ``` INSERT INTO `student_x`(`id`, `name`, `class`, `score`) VALUES (1, '张三', '数学', 78); INSERT INTO `student_x`(`id`, `name`, `class`, `score`) VALUES (2, '张三', '英语', 93); INSERT INTO `student_x`(`id`, `name`, `class`, `score`) VALUES (3, '张三', '语文', 65); INSERT INTO `student_x`(`id`, `name`, `class`, `score`) VALUES (4, '李四', '数学', 87); INSERT INTO `student_x`(`id`, `name`, `class`, `score`) VALUES (5, '李四', '英语', 90); INSERT INTO `student_x`(`id`, `name`, `class`, `score`) VALUES (6, '李四', '语文', 76); INSERT INTO `student_x`(`id`, `name`, `class`, `score`) VALUES (7, '李四', '历史', 69); ``` 进行行转列的实验环境最后如下所示: ![](http://www.netman.vip/blog/usr/uploads/2023/05/3829845119.png) **示例一:使用聚合函数** 我们在这个示例中,使用聚合函数sum、max、min、avg 来完成我们的行转列的需求,具体的实验如下。 1、下面是我们需要完成行转列的效果图: ![](http://www.netman.vip/blog/usr/uploads/2023/05/3428199625.png) 使用case when 语句来拼装新的数据列 ``` select name, case when class = '数学' then score else null end as math_score, case when class = '英语' then score else null end as engilsh_score, case when class = '语文' then score else null end as chinese_score, case when class = '历史' then score else null end as history_score from student_x; ``` ![](http://www.netman.vip/blog/usr/uploads/2023/05/931526811.png) 2、基于上面的效果图,我们需要把数据结果,按照name列进行聚合,让姓名相同的数据行合并为同一行来展示,同时,每一列的科目中,只有一行数据是有成绩的,其他行成绩都是空null,所以他们结合使用max函数,可以达到合并行,并且每列的科目成绩不会因为合并行而影响到最后的每一科目的成绩。实现上述转换的SQL语句如下: ``` select name, max(case when class = '数学' then score else null end) as math_score, max(case when class = '英语' then score else null end) as engilsh_score, max(case when class = '语文' then score else null end) as chinese_score, max(case when class = '历史' then score else null end) as history_score from student_x group by name; ``` ![](http://www.netman.vip/blog/usr/uploads/2023/05/954626026.png) 3、我们上面使用了case when 语句来判断,其实if 语句也可以达到case when 语句的效果。如下是使用if 语句的结果: ``` select name, max(if(class = '数学', score, null)) as math_score, max(if(class = '英语', score, null)) as engilsh_score, max(if(class = '语文', score, null)) as chinese_score, max(if(class = '历史', score, null)) as history_score from student_x group by name; ``` ![](http://www.netman.vip/blog/usr/uploads/2023/05/421655214.png) 4、实现过程分析 这里我们使用了聚合函数max,把每一个学生的姓名作为key,进行分组统计。 因为每一个学生对应每一门科目的成绩只有一行记录,所以我们使用聚合函数sum统计后的每一科目的成绩,仍然是该科目单独的成绩。 如果这里每一个学生对应每一门科目有多个成绩记录,这里就不能使用聚合函数max了,如果使用max,最后的结果将是每一个学生对应每一门科目成绩的最大值。 这里之所以使用max的目的是为了达到一个分组的效果。这里的max可以使用sum、min、avg等聚合函数替换掉,它们三个的效果和sum函数的效果在这里是一样的。 5、总结:上面的这样的实现方式,使大家经常使用的,也是大家最熟悉的一种方式。但是这样的写法有一个问题,就是当我们的科目名称变动或者增加或者减少的时候,我们SQL语句也需要作出对应的修改。因为我们在SQL语句中已经使用了hard code 硬编码的方式把科目的名称给写死了,所以这样的SQL不太灵活。 我们可以参考使用下面的几种写法,每一种写法稍微有点不同,但是这些方式基本都能满足我们的需求。 **示例二:使用group\_concat函数** 上面使用的聚合函数是一种类似于硬编码的方式来实现,为了避免硬编码,我们可以使用gourp\_concat 函数来实现,但是这个函数所能实现的效果和使用聚合函数的方式实现的效果有一点点差异,区别的地方就是在展现转换后的列的时候不太一样。具体可以看下面的效果。 我们要达到的效果图如下: ![](http://www.netman.vip/blog/usr/uploads/2023/05/4256457474.png) 1、要达到如上的效果图,我们使用group\_concat函数即可直接完成,SQL语句如下: ``` select name, group_concat(score separator ',') as 'scores' from student_x group by name; ``` ![](http://www.netman.vip/blog/usr/uploads/2023/05/196809403.png) 2、我们可以发现上面的转换后的列是合并在一个列中显示的,我们肯能不太清楚各个逗号分隔的成绩分别代表哪个科目。所以我们需要在转换后的列中增加显示科目的名称,达到如下的效果: ![](http://www.netman.vip/blog/usr/uploads/2023/05/1550031553.png) 达到上图所示的效果图的SQL语句如下所示: ``` select name, group_concat(class separator ',') as classes, group_concat(score separator ',') as scores from student_x group by name; ``` ![](http://www.netman.vip/blog/usr/uploads/2023/05/1741146169.png) 3、此外我们还可以科目名称和科目成绩放在一起,然后以逗号分隔显示,这样能够更清楚的看到每一个科目对应的成绩是多少。达到如下图所示的效果: ![](http://www.netman.vip/blog/usr/uploads/2023/05/1607382651.png) 如下的SQL就可以满足上面图片所示的效果: ``` select name, group_concat(class,':',score separator ',') as 'class:score' from student_x group by name; ``` ![](http://www.netman.vip/blog/usr/uploads/2023/05/2345883143.png) 4、总结:使用group\_concat 函数所能达到的效果,和使用聚合函数相比,它不需要因为科目的变动而修改SQL语句,所以它可能更灵活一些。同时,它也有自己的缺点。那就是它的列展现的方式和使用聚合函数相比,不能分开展现,只能以指定的分隔符来连接起来放在一列展现。 当然对于如果我们需要使用分开的列来展现转换后的各个科目的成绩,我们基于group\_concat函数转换后的结果,再结合substring 函数或者substring\_index 函数来对结果列进行截取和拆分来实现把一列分为多列展示的效果。但是这样的效果有有点hard code 函数的效果了,因为在截取字符串的时候,需要使用指定从哪里开始截取,截取到哪里,截取第几个元素等问题。 二、列转行的示例 **示例一:使用union all功能** 1、在进行列转行的实验之前,我们需要先准备好我们的实验环境,准备好表和表中的初始化数据。表结构和初始化数据如下: 准备初始化表结构 ``` CREATE TABLE `student_y` ( `id` int(11) DEFAULT NULL, `name` varchar(255) DEFAULT NULL, `math_score` bigint(255) DEFAULT NULL, `engilsh_score` bigint(255) DEFAULT NULL, `chinese_score` bigint(255) DEFAULT NULL, `history_score` bigint(255) DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; ``` 准备初始化表中的数据 ``` INSERT INTO `student_y`(`id`, `name`, `math_score`, `engilsh_score`, `chinese_score`, `history_score`) VALUES (1, '张三', 78, 93, 65, NULL); INSERT INTO `student_y`(`id`, `name`, `math_score`, `engilsh_score`, `chinese_score`, `history_score`) VALUES (2, '李四', 87, 90, 76, 69); ``` 2、进行列转行的实验环境最后如下所示: 3、实验环境准备后之后,接下来我们开始使用union all的功能来实现列转行的功能。 我们要达到的列转行的转换效果如下: 我们可以对所有学生的每一个科目的成绩进行单独查询,结果如下: ``` select name, math_score as score from student_y; select name, engilsh_score as score from student_y; select name, chinese_score as score from student_y; select name, history_score as score from student_y; ``` 其他三个查询结果类似,不再展示。 4、基于上面的每一个查询结果,把最后的结果使用union all 关键词合并在一起,效果如下: ``` select name, math_score as score from student_y union all select name, engilsh_score as score from student_y union all select name, chinese_score as score from student_y union all select name, history_score as score from student_y; ``` 5、此时,我们发现结果中已经大概实现了列转为行的需求。但是顺序没有达到要求,每个人的各个科目的成绩应该挨着,但是目前是没有挨着的。所以我们需要在基于上面的查询结果,外面在包裹一层查询,增加一个order by语句在外层查询中,即可得到我们想要的顺序。如下所示: ``` select * from ( select name, math_score as score from student_y union all select name, engilsh_score as score from student_y union all select name, chinese_score as score from student_y union all select name, history_score as score from student_y ) as x order by name; ``` 此时的结果已经很接近我们的最后想要的结果了,但是我们发现,每个学生的成绩我们不能区分各个科目的成绩是多少,所以我们需要把科目也纳入到结果集中,也就是我们在查询的时候,要把各个成绩对应的列名称也包含到查询的字段中。于是就有了如下的SQL ``` select * from ( select name, 'math_score' as class, math_score as score from student_y union all select name, 'engilsh_score' as class, engilsh_score as score from student_y union all select name, 'chinese_score' as class, chinese_score as score from student_y union all select name, 'history_score' as class, history_score as score from student_y ) as x order by name,class; ``` 6、总结:这里我们采用了union all的功能,把所有学生每个科目的成绩单独查询出来,然后把结果集继续合并。最后达到我们想要的列转为行的效果。但是我们发现此时的SQL语句中,和前面我们在进行行转列的时候,使用聚合函数的方式来实现行转列的方式类似,使用了hard code的硬编码,如果科目名称或数目发生改变,我们的SQL语句也需要跟着动态的去修改,这是这种方式的缺陷。但是优点就是比较容易理解。 **示例二:使用substring_index函数** 有的时候,我们的列转行的表,并不是像前面示例一那样,所有的列都是分开的。它有时候会是这样的数据内容,多个字段是通过一个特定的符号链接起来的。 1、准备实验环境用的表结构: ``` CREATE TABLE `student_y2` ( `id` int(11) DEFAULT NULL, `name` varchar(255) DEFAULT NULL, `scores` varchar(255) DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; ``` 初始化使用用的数据: ``` INSERT INTO `student_y2`(`id`, `name`, `scores`) VALUES (1, '张三', '78,93,65'); INSERT INTO `student_y2`(`id`, `name`, `scores`) VALUES (2, '李四', '87,90,76,69'); ``` 2、环境准备好之后,我们开始基于这样的一个表,来进行列转行的操作。 使用substring_index(str, delim, count) 函数来拆分字段,这里需要理解一下substring_index() 函数的功能。 基于上面解释的该函数的功能,我们可以得到如下的结果: ``` select x.name, x.scores, char_length(x.scores) as length_scores, char_length(replace(x.scores, ',', '')) as length_scores_with_out_comma, char_length(x.scores) - char_length(replace(x.scores, ',', '')) + 1 as colum_num, substring_index(x.scores, ',', 1) as score1, substring_index(x.scores, ',', 2) as score12, substring_index(x.scores, ',', 3) as score123, substring_index(x.scores, ',', 4) as score1234, substring_index(substring_index(x.scores, ',', 1), ',', -1) as colum1, substring_index(substring_index(x.scores, ',', 2), ',', -1) as colum2, substring_index(substring_index(x.scores, ',', 3), ',', -1) as colum3, substring_index(substring_index(x.scores, ',', 4), ',', -1) as colum4 from student_y2 as x; ``` a、如果count是正数,那么就是从左往右数,第N个分隔符的左边的全部内容; b、相反,如果是负数,那么就是从右边开始数,第N个分隔符右边的所有内容; c、如果想获取某个字符串中间的某个字符串,那就先得到左边的全部,然后基于截取后的字符串在得到右边的全部就可以得到想要的字符串。 3、从上面的结果可以看出,我们可以使用substring\_index函数来截取原始表中的score的值,根据逗号进行截取。问题的关键是我们需要使用一个序列来动态的判断该解决到哪个位置。而MySQL中刚好有一个系统表mysql.help\_topic ,这个表可以提供我们需要的序列0,1,2,3,4,5,6…n 。基于这个表,我们可以构建出如下的SQL和结果: ``` select * from mysql.help_topic y; ``` ``` select x.name, substring_index(substring_index(x.scores, ',', y.help_topic_id + 1), ',', -1) as 'score' from student_y2 as x join mysql.help_topic y on y.help_topic_id < (length(x.scores) - length(replace(x.scores, ',', '')) + 1); ``` 此时我们使用了mysql.help\_topic表的主键值help\_topic\_id的值,关联的时候使用了小于的关系,这就会每次都循环查并判断表中的数据,知道不满足关联的条件,这个技巧很巧妙。如果我们不使用mysql.help\_topic ,我们自己创建一个一行一列的表,表中的数据是从0,1,2,3,4,5,6…n 一直自增的数据,也可以达到同样的效果。 4、总结:我们使用了substring\_index 函数结合一个序列表mysql.help\_topic 来完成了动态的将列转行的功能。这种方式不会因为列中的数据增加或减少而导致SQL需要重新,它可普适性比较好,可以应对大多数是列转行的需求。但是它缺点是不太容易理解,SQL语句的技巧性比较强。并且对于输入的原始行数据也要特殊的格式要求,要求是使用特定的符号链接的字符串作为一个列的数据出现在原始的输入数据中。 **最后总结** 我们在文章的开始,首选针对行和列相互转行的概念做了简单澄清,因为有很多人会把行转列和列转行给混淆。行转列和列转行的最明显的区分如下: 行转列,最后的结果中行变少了,列变多了。 列转行,最后的结果中列变少了,行变多了。 紧接着,我们分别针对行转列和列转行进行了实验演示。 在行转列的实验中,我们采用了三种方式来做演示。 使用聚合函数max来实现行转列的需求。 优点:便于理解,也是大家经常或首先想到的方式。 缺点:如果key值变动相应的SQL语句需要作出对应的调整。 使用group\_concat来实现行转列的需求。 优点:key值的变动不会导致SQL的重写。 缺点:行转列之后的结果,列不是分开显示的,是以某种符号间隔连接在一起组从字符串展示的。 使用动态拼装SQL语句块的方式来实现行转列的需求。 优点:key值的变动不会导致SQL的重写,同时列的展示方式和使用max聚合函数的展示结果一致。 缺点:SQL语句分多步,过程有点复杂,不太容易理解。 在列转行的实验中,我们采用了两种方式来做演示。 使用union all功能来实现列转行的需求。 优点:便于理解,这样的方式使我们首先能想到的实现方式。 缺点:key值的变动将导致SQL需要重写。 使用substring\_index函数结合连续自增序列来完成列转行的需求。 优点:key值的变动不会导致SQL的重写。 缺点:不太容易理解,里面有很巧妙的SQL技巧。对于输入列有特殊的要求,需要使用指定的字符连接在一起组成字符串作为输入的多列。 最后一点来个首位呼应,使用union all 可以满足多列转多行,但是如果sql代码比较复杂,就是使sql代码过于臃肿,然后我就想到了可以把那些重复用的代码当成一个中间表来使用。这样可以很大程度的减少代码的总数量。mysql创建中间表,有两种方法:1、with as (适合mysql8版本以上)2、CREATE TEMPORARY TABLE test,都可以使用,但是会存在一个问题(假如你只有read权限,就不能使用这个方法) 如果你使mysql低版本而且又是readuser,那么就只能用substring\_index函数和mysql自带的help\_topic表格来进行多列转多行的操作了。如果只是简单查询,sql代码较少,还是比较推荐使用union all 毕竟简单又好用。 以上就是mysql查询结果行列转置的一些总结,大家有更好的方法,烦请留言告知,谢谢!
MyBatis主配置文件解析 作者: juoliii 时间: 2023-05-07 分类: 开发,数据库 评论 具体请参见官网 http://www.mybatis.org/mybatis-3/zh/configuration.html ## MyBatis配置文件中大标签configuration下子标签包括: configuration |--- properties |--- settings |--- typeAliases |--- typeHandlers |--- objectFactory |--- plugins |--- environments |--- |--- environment |--- |--- |--- transactionManager |--- |--- |\_\_ dataSource |\_\_ mappers ## 1 properties属性 properties和java的.properties的配置文件有关。配置properties的resource指定.properties的路径,然后再在properties标签下配置property的name和value,则可以替换.properties文件中相应属性值。 ```xml ``` ## 2 settings设置 这是MyBatis 修改操作运行过程细节的重要的步骤。下方这个表格描述了这些设置项、含义和默认值。 ------------ 设置项描述允许值默认值cacheEnabled对在此配置文件下的所有cache 进行全局性开/关设置。true | falsetruelazyLoadingEnabled全局性设置懒加载。如果设为‘false’,则所有相关联的都会被初始化加载。true | falsetrueaggressiveLazyLoading当设置为‘true’的时候,懒加载的对象可能被任何懒属性全部加载。否则,每个属性都按需加载。true | falsetruemultipleResultSetsEnabled允许和不允许单条语句返回多个数据集(取决于驱动需求)true | falsetrueuseColumnLabel使用列标签代替列名称。不同的驱动器有不同的作法。参考一下驱动器文档,或者用这两个不同的选项进行测试一下。true | falsetrueuseGeneratedKeys允许JDBC 生成主键。需要驱动器支持。如果设为了true,这个设置将强制使用被生成的主键,有一些驱动器不兼容不过仍然可以执行。true | falsefalseautoMappingBehavior指定MyBatis 是否并且如何来自动映射数据表字段与对象的属性。PARTIAL将只自动映射简单的,没有嵌套的结果。FULL 将自动映射所有复杂的结果。NONE,PARTIAL,FULLPARTIALdefaultExecutorType配置和设定执行器,SIMPLE 执行器执行其它语句。REUSE 执行器可能重复使用prepared statements 语句,BATCH执行器可以重复执行语句和批量更新。SIMPLEREUSEBATCHSIMPLEdefaultStatementTimeout设置一个时限,以决定让驱动器等待数据库回应的多长时间为超时正整数Not Set(null) 例如: ```xml ``` ## 3 typeAliases类型别名 类型别名是Java 类型的简称。 它仅仅只是关联到XML 配置,简写冗长的JAVA 类名。例如: ```xml ``` 使用这个配置,“StudentEntity”就能在任何地方代替“com.manager.data.model.StudentEntity”被使用。对于普通的Java类型,有许多内建的类型别名。它们都是大小写不敏感的,由于重载的名字,要注意原生类型的特殊处理。 别名映射的类型_bytebyte_longlong_shortshort_intint_integerint_doubledouble_floatfloat_booleanbooleanstringStringbyteBytelongLongshortShortintIntegerintegerIntegerdoubleDoublefloatFloatbooleanBooleandateDatedecimalBigDecimalbigdecimalBigDecimalobjectObjectmapMaphashmapHashMaplistListarraylistArrayListcollectionCollectioniteratorIterator ## 4 typeHandlers类型句柄 无论是MyBatis在预处理语句中设置一个参数,还是从结果集中取出一个值时,类型处理器被用来将获取的值以合适的方式转换成Java类型。下面这个表格描述了默认的类型处理器。 类型处理器Java类型JDBC类型BooleanTypeHandlerBoolean,boolean任何兼容的布尔值ByteTypeHandlerByte,byte任何兼容的数字或字节类型ShortTypeHandlerShort,short任何兼容的数字或短整型IntegerTypeHandlerInteger,int任何兼容的数字和整型LongTypeHandlerLong,long任何兼容的数字或长整型FloatTypeHandlerFloat,float任何兼容的数字或单精度浮点型DoubleTypeHandlerDouble,double任何兼容的数字或双精度浮点型BigDecimalTypeHandlerBigDecimal任何兼容的数字或十进制小数类型StringTypeHandlerStringCHAR和VARCHAR类型ClobTypeHandlerStringCLOB和LONGVARCHAR类型NStringTypeHandlerStringNVARCHAR和NCHAR类型NClobTypeHandlerStringNCLOB类型ByteArrayTypeHandlerbyte[]任何兼容的字节流类型BlobTypeHandlerbyte[]BLOB和LONGVARBINARY类型DateTypeHandlerDate(java.util)TIMESTAMP类型DateOnlyTypeHandlerDate(java.util)DATE类型TimeOnlyTypeHandlerDate(java.util)TIME类型SqlTimestampTypeHandlerTimestamp(java.sql)TIMESTAMP类型SqlDateTypeHandlerDate(java.sql)DATE类型SqlTimeTypeHandlerTime(java.sql)TIME类型ObjectTypeHandlerAny其他或未指定类型EnumTypeHandlerEnumeration类型VARCHAR-任何兼容的字符串类型,作为代码存储(而不是索引)。 你可以重写类型处理器或创建你自己的类型处理器来处理不支持的或非标准的类型。要这样做的话,简单实现TypeHandler接口(org.mybatis.type),然后映射新的类型处理器类到Java类型,还有可选的一个JDBC类型。然后再typeHandlers中添加这个类型处理器。 新定义的类型处理器将会覆盖已经存在的处理Java的String类型属性和VARCHAR参数及结果的类型处理器。要注意MyBatis不会审视数据库元信息来决定使用哪种类型,所以你必须在参数和结果映射中指定那是VARCHAR类型的字段,来绑定到正确的类型处理器上。这是因为MyBatis直到语句被执行都不知道数据类型的这个现实导致的。 ```java 1 public class LimingStringTypeHandler implements TypeHandler { @Override public void setParameter(PreparedStatement ps, int i, Object parameter, JdbcType jdbcType) throws SQLException { System.out.println("setParameter - parameter: " + ((String) parameter) + ", jdbcType: " + jdbcType.TYPE_CODE); ps.setString(i, ((String) parameter)); } @Override public Object getResult(ResultSet rs, String columnName) throws SQLException { System.out.println("getResult - columnName: " + columnName); return rs.getString(columnName); } @Override public Object getResult(CallableStatement cs, int columnIndex) throws SQLException { System.out.println("getResult - columnIndex: " + columnIndex); return cs.getString(columnIndex); } } ``` 在配置文件的typeHandlers中添加typeHandler标签。 ```xml ``` ## 5 ObjectFactory对象工厂 每次MyBatis 为结果对象创建一个新实例,都会用到ObjectFactory。默认的ObjectFactory 与使用目标类的构造函数创建一个实例毫无区别,如果有已经映射的参数,那也可能使用带参数的构造函数。 如果你重写ObjectFactory 的默认操作,你可以通过继承org.apache.ibatis.reflection.factory.DefaultObjectFactory创建一下你自己的。 ObjectFactory接口很简单。它包含两个创建用的方法,一个是处理默认构造方法的,另外一个是处理带参数构造方法的。最终,setProperties方法可以被用来配置ObjectFactory。在初始化你的ObjectFactory实例后,objectFactory元素体中定义的属性会被传递给setProperties方法。 ```java public class LimingObjectFactory extends DefaultObjectFactory { private static final long serialVersionUID = -399284318168302833L; @Override public Object create(Class type) { return super.create(type); } @Override public Object create(Class type, List constructorArgTypes, List constructorArgs) { System.out.println("create - type: " + type.toString()); return super.create(type, constructorArgTypes, constructorArgs); } @Override public void setProperties(Properties properties) { System.out.println("setProperties - properties: " + properties.toString() + ", someProperty: " + properties.getProperty("someProperty")); super.setProperties(properties); } } ``` 配置文件中添加objectFactory标签 ```xml ``` ## 6 plugins插件 MyBatis允许你在某一点拦截已映射语句执行的调用。默认情况下,MyBatis允许使用插件来拦截方法调用: - Executor(update, query, flushStatements, commit, rollback, getTransaction, close, isClosed) - ParameterHandler(getParameterObject, setParameters) - ResultSetHandler(handleResultSets, handleOutputParameters) - StatementHandler(prepare, parameterize, batch, update, query) 这些类中方法的详情可以通过查看每个方法的签名来发现,而且它们的源代码在MyBatis的发行包中有。你应该理解你覆盖方法的行为,假设你所做的要比监视调用要多。如果你尝试修改或覆盖一个给定的方法,你可能会打破MyBatis的核心。这是低层次的类和方法,要谨慎使用插件。 使用插件是它们提供的非常简单的力量。简单实现拦截器接口,要确定你想拦截的指定签名。 ## 7 environments环境 MyBatis 可以配置多个环境。这可以帮助你SQL 映射对应多种数据库等。 ## 8 mappers映射器 这里是告诉MyBatis 去哪寻找映射SQL 的语句。可以使用类路径中的资源引用,或者使用字符,输入确切的URL 引用。 例如: ```xml ```