在mybatis中返回HashMap,并将其用作春季MVC中的ModelAttribute

2022-09-01 12:59:52

我想使用弹簧mvc@modelAttribute在我的Jsp页面中显示类别列表。

在我的映射器中.xml文件是

<select id="selectAllCategories" resultMap="BaseResultMap">
  select id, name from categories  
</select>

在我的映射器.java类中,我有方法

List<Map<String, String>> selectAllCategories();

我想要一个这样的方法:

Map<Integer, String>`selectAllCategories();

而不是 ,这可能吗?List<Map<>>


答案 1

您希望获得 一个,其中 Integer 是 integer,String 是 .如果表中有 200 个类别,则地图中需要 200 个条目,而不是包含 200 张地图的列表。Map<Integer,String>idname

MyBatis不能开箱即用地做到这一点,但你可以使用它的设施来做到这一点。我看到两种选择。

选项 1:

第一个不是你所要求的,但值得展示。它为您提供了一个 Category 是类别表的域对象,该表具有 id、名称(可能还有类别表中的其他字段)。创建类别域对象后,在 MyBatis 中使用注释可以很容易地执行此操作:Map<Integer,Category>@MapKey

@Select("SELECT id, name FROM categories")
@MapKey("id")
Map<Integer,Category> getAllCategories();

在你的代码中,你会做:

MyMapper mapper = session.getMapper(MyMapper.class);
Map<Integer,Category> m = mapper.getAllCategories();

这可能适用于您的用例,也可能不起作用,具体取决于您是否可以将名称提取为Category对象的属性。


选项 2:

为了得到你所要求的,我知道的最简单方法是创建一个实现MyBatis ResultHandler接口的类。Map<Integer,String>

您的 ResultHandler 将使用默认的列名 = > MyBatis 创建的列值哈希映射并创建单个主映射。代码如下:

public class CategoryResultHandler implements ResultHandler {

  Map<Integer,String> inMap = new HashMap<Integer,String>(); 

  public Map<Integer, String> getIdNameMap() {
    return inMap;
  }

  @Override
  public void handleResult(ResultContext rc) {
    @SuppressWarnings("unchecked")
    Map<String,Object> m = (Map<String,Object>)rc.getResultObject();
    inMap.put((Integer)getFromMap(m, "id"), 
              (String)getFromMap(m, "name"));
  }

  // see note at bottom of answer as to why I include this method
  private Object getFromMap(Map<String, Object> map, String key) {
    if (map.containsKey(key.toLowerCase())) {
      return map.get(key.toLowerCase());
    } else {
      return map.get(key.toUpperCase());
    }
  }
}

handleResult 方法在类别表中每行调用一次。您告诉 MyBatis 使用 ResultHandler,然后像这样提取您的主地图:

CategoryResultHandler rh = new CategoryResultHandler();
session.select("getAllCategories", rh);
Map<Integer,String> m = rh.getIdNameMap();

这两个中的一个应该适合你。

最后要注意的几点:

  1. 为什么我包含帮助程序方法?因为您无法始终控制 MyBatis 返回的哈希映射中列名的大小写。更多细节在这里: mybatis- 3.1.1. 如何覆盖从 mybatis 返回的结果映射getFromMap()

  2. 我在Mybatis-koans的Koan26(我根据您的问题添加)中有这些解决方案的工作示例:https://github.com/midpeter444/mybatis-koans


答案 2

推荐