3

说你给定的一个实例List[Class[_ <: Base]],并说许多类扩展Base

class A extends Base
class B extends Base
class C extends A

等等'...
现在,给定的列表可能只包含一些类。例如:val classes = classOf[A] :: Nil好吧,当我实例化val它的类是否属于列表中的类型,或者它是否是列表中某个类的子类时,我该如何测试?即您将如何实施:
def testClass(class : List[Class[_ <: Base]], instance : Base) : Boolean
何时:

val classes = classOf[A] :: Nil
testClass(classes, new A) // should return true
testClass(classes, new B) // should return false
testClass(classes, new C) // should return true

用例:
我正在尝试编写一个通用的重试模式,即获取一些非常容易出错的代码,并且我想重试执行它的最大尝试次数,当每次失败时,它应该执行一些“等待“ 方法。
例如

retryRequest({
    //Some code that throws exceptions
}, classOf[SomeException] :: classOf[SomeOtherException] :: Nil,
100, {Thread.sleep(5000)})

好吧,这可以正常工作,但它不会测试给定异常的子类:

def retryRequest(req : => Unit, validExceptions : List[Class[_ <: java.lang.Throwable]], tries : Int, waitMethod : => Unit) {
    var keepTrying = false
    var tryCount = 0
    do{
        try{
            logger.debug("retryRequest, try #" + tryCount)
            keepTrying = false
            req
        }catch{
            case ex if(tryCount >= tries && validExceptions.contains(ex.getClass)) => {
                throw new MaxTriesReachedException("tried for " + tryCount + "times, but no luck. " +
                                                   "you may want to try ommitting generic exceptions types from the given list.") 
            }
            case ex if (validExceptions.contains(ex.getClass)) => {
                logger.debug("intercepted " + ex.toString)
                tryCount += 1
                keepTrying = true
                waitMethod
            }
        }
    }while(keepTrying)
}

我真的想替换:

validExceptions.contains(ex.getClass)

有类似的东西:

validExceptions.exists(exClass => ex.isInstanceOf[exClass]) //won't compile

是否可以?如何?

4

2 回答 2

9

一种更简单的方法是使用util.control.Exception

  def retryRequest(req: => Unit, 
                   validExceptions: List[Class[_ <: Throwable]], 
                   tries: Int, 
                   waitMethod: => Unit): Unit =
   (Exception.catching(validExceptions:_*) withApply { e => 
      waitMethod
      if (tries > 1) retryRequest(req, validExceptions, tries - 1, waitMethod)
   }) { req }

  retryRequest( { println("a"); throw new Exception }, 
                List(classOf[Exception]), 
                3,
                Thread.sleep(100))

简而言之:withApply采用一个闭包来处理传递给的异常之一catching被抛出时的情况。在我们的例子中,我们只是简单地递归调用自己(鉴于重试次数很少,我认为这不是问题)。

于 2012-09-27T12:18:48.683 回答
7

我认为这应该有效:

validExceptions.exists(exClass => exClass.isAssignableFrom(ex.getClass))

http://docs.oracle.com/javase/6/docs/api/java/lang/Class.html#isAssignableFrom%28java.lang.Class%29

于 2012-09-27T10:35:49.637 回答