Spring 中的 MyBatis 具有 Sql 生成器和注解

MyBatis in Spring with Sql Builder and annotation

我正在 Spring 中使用 MyBatis 进行一些测试,我决定测试两种使用注解进行 sql 查询的方法,但其中一种方法不起作用,我需要一些帮助明白问题了。

正在运行:

@Select("<script> "
    + "SELECT * FROM user "
    + "<where>"
    + "<if test=\"username != null\"> OR username LIKE '%' #{username} '%' </if>"
    + "<if test=\"name != null\"> OR name LIKE '%' #{name} '%' </if>"
    + "<if test=\"email != null\"> OR email LIKE '%' #{email} '%' </if>"
    + "</where>"
    + " </script>")
public List<User> findByDynamicFilter(@Param("username") String username, 
        @Param("name") String name, @Param("email") String email);

不行的那个:

映射器

@SelectProvider(type = UserSqlBuilder.class, method = "buildFindByDynamicFilter")
public List<User> findByDynamicFilter(@Param("username") String username, 
        @Param("name") String name, @Param("email") String email);

Class SqlBuilder

public class UserSqlBuilder {

    public static String buildFindByDynamicFilter(final String username, final String name, final String email) {
        return new SQL() {
            {
                SELECT("*");
                FROM("user");
                if (name != null) {
                    OR().WHERE("name like #{name} || '%'");
                }
                if (username != null) {
                    OR().WHERE("username like #{username}");
                }
                if (email != null) {
                    OR().WHERE("email like #{email} || '%'");
                }
            }
        }.toString();
    }

}

错误:

2018-12-12 19:47:12.935 ERROR 7454 --- [nio-8080-exec-1] o.a.c.c.C.[.[.[/].[dispatcherServlet] : Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception [Request processing failed; nested exception is org.mybatis.spring.MyBatisSystemException: nested exception is org.apache.ibatis.builder.BuilderException: Error invoking SqlProvider method (br.com.andre.springmvcuser.mapper.sqlbuilder.UserSqlBuilder.buildFindByDynamicFilter). Cause: org.apache.ibatis.binding.BindingException: Parameter 'arg0' not found. Available parameters are [name, param3, param1, email, username, param2]] with root cause

org.apache.ibatis.binding.BindingException: Parameter 'arg0' not found. Available parameters are [name, param3, param1, email, username, param2] at org.apache.ibatis.binding.MapperMethod$ParamMap.get(MapperMethod.java:204) ~[mybatis-3.4.6.jar:3.4.6] at org.apache.ibatis.builder.annotation.ProviderSqlSource.extractProviderMethodArguments(ProviderSqlSource.java:156) ~[mybatis-3.4.6.jar:3.4.6] at org.apache.ibatis.builder.annotation.ProviderSqlSource.createSqlSource(ProviderSqlSource.java:120) ~[mybatis-3.4.6.jar:3.4.6] at org.apache.ibatis.builder.annotation.ProviderSqlSource.getBoundSql(ProviderSqlSource.java:102) ~[mybatis-3.4.6.jar:3.4.6] at org.apache.ibatis.mapping.MappedStatement.getBoundSql(MappedStatement.java:292) ~[mybatis-3.4.6.jar:3.4.6] at org.apache.ibatis.executor.CachingExecutor.query(CachingExecutor.java:81) ~[mybatis-3.4.6.jar:3.4.6] at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:148) ~[mybatis-3.4.6.jar:3.4.6] at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:141) ~[mybatis-3.4.6.jar:3.4.6] at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) ~[na:1.8.0_191] at sun.reflect .....

这些参数是什么 [name, param3, param1, email, username, param2]]?

我做SqlBuilder的方式还需要别的吗?

出现这个问题是因为mybatis无法将mapper方法的参数传递给builder。

您要么需要在构建器中注释参数:

public static String buildFindByDynamicFilter(
   final @Param("username") String username,
   final @Param("name") String name,
   final @Param("email") String email) {

或从构建器方法签名中删除未使用的参数(在您的情况下,所有参数都可以删除,因为构建器中使用了 none):

public static String buildFindByDynamicFilter() {
    ...
} 

include method parameter names 编译成 类 并确保构建器方法中的参数与映射器方法中的参数命名方式相同。

Mybatis java API构建查询(SELECTFROMWHERE等函数)本身不需要参数。构建器的目的是生成 sql 查询文本。当 mybatis 从构建器(或 @Select 注释或 xml 映射器)检索查询文本并且使用 JDBC 准备好的语句执行查询时,参数绑定稍后发生。

[name, param3, param1, email, username, param2]是mapper方法的参数。每个参数在两个键下可用,第一个是映射器方法参数上 @Param 注释的值,第二个只是参数的序号,前缀为 param (索引从零开始并且第一个被跳过,因为 mapper 方法的第一个参数是隐式的 this)。 Mybatis 试图通过可用参数列表显示合理的错误消息,但如果没有可用的方法参数信息,则会令人困惑。

在构建器中,如果没有 @Param 注释存在,则使用反射来获取方法参数。如果方法参数信息未包含在 -parameters javac 选项中,则参数名称为 arg0arg1 等。