成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

使用Mybatis遇到的there is no getter 異常

zhichangterry / 1404人閱讀

摘要:像在本次的測試代碼的話,會直接在返回,不過這不是重點,繼續(xù)往下走,會返回到的方法的這一行此時的就是一個對象了。那么為什么加了注解之后就不會拋出異常呢此時就需要注意類的方法。

在使用mybatis的時候有時候會遇到一個問題就是明明參數(shù)是正確的,但是還是會提示There is no getter XXX這個異常,但是一般的解決辦法是在mapper里面添加@Param注解來完成是別的,那么為什么會遇到這個問題呢?

以下為舉例代碼:

Mapper層代碼
public interface Pro1_Mapper {

    Pro1_Studnet insertStu(Pro1_Studnet pro1_studnet);

}
實體類代碼
public class Pro1_Studnet {

    private String stuId;

    private String stuName;

    private String stuClass;

    private String stuTeacher;

    public String getStuId() {
        return stuId;
    }

    public void setStuId(String stuId) {
        this.stuId = stuId;
    }

    public String getStuName() {
        return stuName;
    }

    public void setStuName(String stuName) {
        this.stuName = stuName;
    }

    public String getStuClass() {
        return stuClass;
    }

    public void setStuClass(String stuClass) {
        this.stuClass = stuClass;
    }

    public String getStuTeacher() {
        return stuTeacher;
    }

    public void setStuTeacher(String stuTeacher) {
        this.stuTeacher = stuTeacher;
    }
}
Main方法
public static void main(String[] args) {
        Logger logger = null;
        logger = Logger.getLogger(Pro1_Main.class.getName());
        logger.setLevel(Level.DEBUG);
        SqlSession sqlSession = null;
        try {
            sqlSession = study.mybatis.MybatisUtil.getSqlSessionFActory().openSession();
            Pro1_Mapper pro1_Mapper = sqlSession.getMapper(Pro1_Mapper.class);
            Pro1_Studnet pro1_studnet =new Pro1_Studnet();
            pro1_studnet.setStuName("張三");
            Pro1_Studnet pro1_studnet1 =pro1_Mapper.insertStu(pro1_studnet);
            System.out.println(pro1_studnet1.getStuClass());
            sqlSession.commit();
        } finally {
            sqlSession.close();
        }
    }
XML文件



    
        
        
        
        
    
    

如果執(zhí)行上述的代碼,你會發(fā)現(xiàn)mybatis會拋出一個異常:
There is no getter for property named "pro1_studnet" in "class study.szh.demo.project1.Pro1_Studnet"
很明顯就是說pro1_studnet這個別名沒有被mybatis正確的識別,那么將這個pro1_studnet去掉呢?

嘗試將xml文件中的pro1_studnet去掉然后只保留stuName,執(zhí)行代碼:

張三

這表明程序運行的十分正常,但是在實際的寫法中,還有如果參數(shù)為String也會導(dǎo)致拋出getter異常,所以此次正好來分析下

分析 mybatis是如何解析mapper參數(shù)的

跟蹤源碼你會發(fā)現(xiàn)在MapperProxyinvoke處會進行入?yún)?

@Override
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    try {
      if (Object.class.equals(method.getDeclaringClass())) {
        return method.invoke(this, args);
      } else if (isDefaultMethod(method)) {
        return invokeDefaultMethod(proxy, method, args);
      }
    } catch (Throwable t) {
      throw ExceptionUtil.unwrapThrowable(t);
    }
    final MapperMethod mapperMethod = cachedMapperMethod(method);
    return mapperMethod.execute(sqlSession, args);
  }

注意此處的args,這個參數(shù)就是mapper的入?yún)ⅰ?br>

那么mybatis在這里接收到這個參數(shù)之后,它會將參數(shù)再一次進行傳遞,此時會進入到MapperMethodexecute方法

public Object execute(SqlSession sqlSession, Object[] args) {
    //省略無關(guān)代碼
      case SELECT:
        if (method.returnsVoid() && method.hasResultHandler()) {
          executeWithResultHandler(sqlSession, args);
          result = null;
        } else if (method.returnsMany()) {
          result = executeForMany(sqlSession, args);
        } else if (method.returnsMap()) {
          result = executeForMap(sqlSession, args);
        } else if (method.returnsCursor()) {
          result = executeForCursor(sqlSession, args);
        } else {
          Object param = method.convertArgsToSqlCommandParam(args);
          result = sqlSession.selectOne(command.getName(), param);
        }
        break;
      case FLUSH:
        result = sqlSession.flushStatements();
        break;
      default:
        throw new BindingException("Unknown execution method for: " + command.getName());
    }
    if (result == null && method.getReturnType().isPrimitive() && !method.returnsVoid()) {
      throw new BindingException("Mapper method "" + command.getName() 
          + " attempted to return null from a method with a primitive return type (" + method.getReturnType() + ").");
    }
    return result;
  }

因為在xml文件里面使用的是select標簽,所以會進入case的select,然后此時會進入到Object param = method.convertArgsToSqlCommandParam(args); 在這里args還是Stu的實體類,并未發(fā)生變化

隨后進入convertArgsToSqlCommandParam方法,然后經(jīng)過一個方法的跳轉(zhuǎn),最后會進入到ParamNameResolvergetNamedParams方法,

public Object getNamedParams(Object[] args) {
    final int paramCount = names.size();
    if (args == null || paramCount == 0) {
      return null;
    } else if (!hasParamAnnotation && paramCount == 1) {
      return args[names.firstKey()];
    } else {
      final Map param = new ParamMap();
      int i = 0;
      for (Map.Entry entry : names.entrySet()) {
        param.put(entry.getValue(), args[entry.getKey()]);
        // add generic param names (param1, param2, ...)
        final String genericParamName = GENERIC_NAME_PREFIX + String.valueOf(i + 1);
        // ensure not to overwrite parameter named with @Param
        if (!names.containsValue(genericParamName)) {
          param.put(genericParamName, args[entry.getKey()]);
        }
        i++;
      }
      return param;
    }
  }

此時注意hasParamAnnotation這個判斷,這個判斷表示該參數(shù)是否含有標簽,有的話在這里會在Map里面添加一個參數(shù),其鍵就是GENERIC_NAME_PREFIX(param) + i 的值。像在本次的測試代碼的話,會直接在return args[names.firstKey()];返回,不過這不是重點,繼續(xù)往下走,會返回到MapperMethodexecute方法的這一行result = sqlSession.selectOne(command.getName(), param);

此時的param就是一個Stu對象了。

繼續(xù)走下去...由于mybatis的調(diào)用鏈太多,此處只會寫出需要注意的點,可以在自己debug的時候稍微注意下。

BaseExecutorcreateCacheKey的方法
@Override
  public CacheKey createCacheKey(MappedStatement ms, Object parameterObject, RowBounds rowBounds, BoundSql boundSql) {
    if (closed) {
      throw new ExecutorException("Executor was closed.");
    }
    CacheKey cacheKey = new CacheKey();
    cacheKey.update(ms.getId());
    cacheKey.update(rowBounds.getOffset());
    cacheKey.update(rowBounds.getLimit());
    cacheKey.update(boundSql.getSql());
    List parameterMappings = boundSql.getParameterMappings();
    TypeHandlerRegistry typeHandlerRegistry = ms.getConfiguration().getTypeHandlerRegistry();
    // mimic DefaultParameterHandler logic
    for (ParameterMapping parameterMapping : parameterMappings) {
      if (parameterMapping.getMode() != ParameterMode.OUT) {
        Object value;
        String propertyName = parameterMapping.getProperty();
        if (boundSql.hasAdditionalParameter(propertyName)) {
          value = boundSql.getAdditionalParameter(propertyName);
        } else if (parameterObject == null) {
          value = null;
        } else if (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) {
          value = parameterObject;
        } else {
          MetaObject metaObject = configuration.newMetaObject(parameterObject);
          value = metaObject.getValue(propertyName);
        }
        cacheKey.update(value);
      }
    }
    if (configuration.getEnvironment() != null) {
      // issue #176
      cacheKey.update(configuration.getEnvironment().getId());
    }
    return cacheKey;
  }

當(dāng)進行到這一步的時候,由于mybatis的類太多了,所以這里選擇性的跳過,當(dāng)然重要的代碼還是會介紹的。

DefaultReflectorFactory的findForClass方法
@Override
  public Reflector findForClass(Class type) {
    if (classCacheEnabled) {
            // synchronized (type) removed see issue #461
      Reflector cached = reflectorMap.get(type);
      if (cached == null) {
        cached = new Reflector(type);
        reflectorMap.put(type, cached);
      }
      return cached;
    } else {
      return new Reflector(type);
    }
  }

注意MetaObjectgetValue方法:

 public Object getValue(String name) {
    PropertyTokenizer prop = new PropertyTokenizer(name);
    if (prop.hasNext()) {
      MetaObject metaValue = metaObjectForProperty(prop.getIndexedName());
      if (metaValue == SystemMetaObject.NULL_META_OBJECT) {
        return null;
      } else {
        return metaValue.getValue(prop.getChildren());
      }
    } else {
      return objectWrapper.get(prop);
    }
  }

這里的name的值是pro1_stu.stuName,而prop的屬性是這樣的:

這里的hasNext函數(shù)會判斷這個prop的children是不是為空,如果不是空的話就會進入 get 方法,然后進入到如下的方法通過返回獲取get方法。
所以當(dāng)遍歷到stuName的時候會直接return,

然后就需要注意ReflectorgetGetInvoker方法,

public Invoker getGetInvoker(String propertyName) {
    Invoker method = getMethods.get(propertyName);
    if (method == null) {
      throw new ReflectionException("There is no getter for property named "" + propertyName + "" in "" + type + """);
    }
    return method;
  }

這個propertyName就是pro1_studnet,而getMethods.get(propertyName);就是要通過反射獲取pro1_studnet方法,但是很明顯,這里是獲取不到的,所以此時就會拋出這個異常。

那么為什么加了@param注解之后就不會拋出異常呢

此時就需要注意MapWrapper類的get方法。

 @Override
  public Object get(PropertyTokenizer prop) {
    if (prop.getIndex() != null) {
      Object collection = resolveCollection(prop, map);
      return getCollectionValue(prop, collection);
    } else {
      return map.get(prop.getName());
    }
  }

在之前就說過,如果加了注解的話,map的結(jié)構(gòu)是{"param1","pro1_studnet","pro1_studnet",XXX對象},此時由于prop的index是null,所以會直接返回map的鍵值為pro1_studnet的對象。
而在DefaultReflectorFactoryfindForClass里面,由于所加載的實體類已經(jīng)包含了Pro1_Student,隨后在metaValue.getValue(prop.getChildren());的將stu_name傳入過去,就可以了獲取到了屬性的值了。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/77091.html

相關(guān)文章

  • 基于 SpringBoot2.0+優(yōu)雅整合 SpringBoot+Mybatis

    摘要:基于最新的,是你學(xué)習(xí)的最佳指南。驅(qū)動程序通過自動注冊,手動加載類通常是不必要。由于加上了注解,如果轉(zhuǎn)賬中途出了意外和的錢都不會改變。三的方式項目結(jié)構(gòu)相比于注解的方式主要有以下幾點改變,非常容易實現(xiàn)。公眾號多篇文章被各大技術(shù)社區(qū)轉(zhuǎn)載。 Github 地址:https://github.com/Snailclimb/springboot-integration-examples(Sprin...

    gghyoo 評論0 收藏0
  • mybatis 為什么每次插入時候總會創(chuàng)建一個SqlSession?

    摘要:問題記錄工作環(huán)境是使用使用用的,在一次調(diào)試中。發(fā)現(xiàn)每一次插入一條數(shù)據(jù)都會創(chuàng)建一個。如圖圖問題可能的原因原因分析沒有使用緩存因為這個是插入,不是查詢,所以這里不存在什么緩存的問題。但是在在關(guān)閉的時候也是做了判斷。 問題記錄: 工作環(huán)境是使用spring boot,使用用的mybatis,在一次調(diào)試中。發(fā)現(xiàn)每一次插入一條 數(shù)據(jù)都會創(chuàng)建一個SqlSession。如圖: 圖1:...

    sixgo 評論0 收藏0
  • SSM框架整合

    摘要:環(huán)境工具數(shù)據(jù)庫連接工具準備工作步驟新建項目建好基本目錄,按住配置目錄,如下圖新建幾個包,如下圖包名名稱作用數(shù)據(jù)訪問層接口與數(shù)據(jù)操作有關(guān)的都放在這里實體類一般與數(shù)據(jù)庫的表相對應(yīng),封裝層取出來的數(shù)據(jù)為一個對象,也就是我們常說的,一般只在層與層之 環(huán)境 Windows10 JDK1.8 MySql5.5 工具 Maven3.5 IDEA2017 Navicat(數(shù)據(jù)庫連接工具) 準備...

    lanffy 評論0 收藏0
  • JavaScript Getters and Setters

    For the most part, in JavaScript, what you see is what you get. A value’s a value; there are no tricks. Sometimes however, you want a value that’s based on some other values: someone’s full name, for ...

    warnerwu 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<