13.11 Scala混用Java的集合类调用scala的foreach遍历问题
问题描述
[ERROR] /Users/jack/book/lightsword/src/main/scala/com/springboot/in/action/service/LightSwordUserDetailService.scala:31: error: value foreach is not a member of java.util.List[com.springboot.in.action.entity.UserRole]
[INFO] for (userRole <- userRoles) {
[INFO] ^
[ERROR] one error found
原因分析
未添加Scala与Java集合库的隐式转换。如下:
import scala.collection.JavaConversions._
由于都运行在JVM上,Java与Scala之间基本能做到无缝的集成,区别主要在于各自的API各有不同。由于Scala为集合提供了更多便捷的函数,因此,Java与Scala在集合之间的互操作,或许是在这种多语言平台下使用最为频繁的。
Scala中操作Java集合
两种情况需要在Scala中操作Java集合。一种是Scala调用了其他的Java库,针对Java集合需要转换为Scala集合,如此才能享受Scala集合提供的福利;另一种是编写了Scala程序,但需要提供给Java库,为了更好地无缝集成,要让Java库体会不到Scala的存在。
Scala调用Java库
为了享用Scala提供的集合特性,在Scala程序中若要调用Java库,通常需要将其转换。
我们的UserRoleDao代码如下:
package com.springboot.in.action.dao
import java.util.List
import com.springboot.in.action.entity.UserRole
import org.springframework.data.jpa.repository.Query
import org.springframework.data.repository.CrudRepository
import scala.language.implicitConversions
trait UserRoleDao extends CrudRepository[UserRole, Integer] {
def findAll(): List[UserRole] // JavaConversions
def save(u: UserRole): UserRole
def findOne(id: Integer): UserRole
@Query(value = "SELECT * FROM user_role where user_id = ?1", nativeQuery = true)
def listByUserId(userId: Integer): List[UserRole]
}
其中,def listByUserId(userId: Integer): List[UserRole],为了使用Jpa的CrudRepository,我们返回的是java.util.List类型。
然后我们在service代码中,应该这样调用:
val userRoles = userRoleDao.listByUserId(user.id)
// Scala中调用java的collection类,使用scala的foreach,编译器会提示无法找到result的foreach方法。因为这里的userRoles的类型为java.util.List。若要将其转换为Scala的集合,就需要增加如下语句:
import scala.collection.JavaConversions._
for (userRole <- userRoles) {
val roleId = userRole.roleId
val roleName = roleDao.findOne(roleId).role
authorities.add(new SimpleGrantedAuthority(roleName))
System.err.println("username is " + username + ", " + roleName)
}
如果没有加上这一句:
import scala.collection.JavaConversions._
就会报开头的错误。
解决方案
加上这一句:
import scala.collection.JavaConversions._