gpt4 book ai didi

function - 自定义 scala 递归预防机制的改进

转载 作者:行者123 更新时间:2023-12-02 09:12:31 25 4
gpt4 key购买 nike

我想创建一个智能的递归预防机制。我希望能够以某种方式注释一段代码,以标记它不应该在递归中执行,如果它确实在递归中执行,那么我想抛出一个自定义错误(可以捕获该错误以允许执行)发生这种情况时自定义代码)

这是我在此之前的尝试:

import scala.collection.mutable.{Set => MutableSet, HashSet => MutableHashSet }

case class RecursionException(uniqueID:Any) extends Exception("Double recursion on " + uniqueID)

object Locking {
var locks:MutableSet[Any] = new MutableHashSet[Any]

def acquireLock (uniqueID:Any) : Unit = {
if (! (locks add uniqueID))
throw new RecursionException(uniqueID)
}

def releaseLock (uniqueID:Any) : Unit = {
locks remove uniqueID
}

def lock1 (uniqueID:Any, f:() => Unit) : Unit = {
acquireLock (uniqueID)
try {
f()
} finally {
releaseLock (uniqueID)
}
}

def lock2[T] (uniqueID:Any, f:() => T) : T = {
acquireLock (uniqueID)
try {
return f()
} finally {
releaseLock (uniqueID)
}
}
}

现在要锁定代码段:

import Locking._

lock1 ("someID", () => {

// Custom code here

})

我的问题是:

  1. 是否有任何明显的方法可以消除对唯一标识符进行硬编码的需要?我需要一个唯一标识符,该标识符实际上将在包含锁定部分的函数的所有调用之间共享(因此我不能有类似计数器的东西来生成唯一值,除非 scala 有静态函数变量)。我想了想
  2. 有什么方法可以美化匿名函数的语法吗?具体来说,就是让我的代码看起来像 lock1 ("id") {/* 代码放在这里 */} 或任何其他更漂亮的外观。
  3. 在这个阶段问有点傻,但我还是会问 - 我是在重新发明轮子吗?(即是否存在这样的事情?)

疯狂的最终想法:我知道滥用synchronized关键字(至少在java中)可以保证代码只会执行一次(在某种意义上)没有多个线程可以同时输入该部分代码)。我不认为它会阻止同一个线程执行代码两次(尽管我在这里可能是错的)。不管怎样,如果它确实阻止了它,我仍然不想要它(即使我的程序是单线程的),因为我很确定它会导致死锁并且不会报告异常。

编辑:为了更清楚地说明,该项目用于错误调试目的和学习 scala。除了在运行时轻松查找代码错误(用于检测不应该发生的递归)之外,它没有真正的用途。请参阅这篇文章的评论。

最佳答案

不太确定您的目标是什么,但有几点说明:

首先,你不需要做lock1和lock2来区分Unit和其他类型。 Unit 是一个合适的值类型,通用方法也适用于它。另外,您可能应该使用按名称调用 argument => T,而不是函数 () => T,并使用两个参数列表:

def lock[T] (uniqueID:Any)(f: => T) : T = {
acquireLock (uniqueID)
try {
f
} finally {
releaseLock (uniqueID)
}
}

然后你可以用lock(id){block}调用,它看起来像常见的指令,例如if或synchronized。

第二,为什么需要uniqueId,为什么要把Lock做成单例?相反,让 Lock 成为一个类,并拥有与拥有 id 一样多的实例。

class Lock {
def lock[T](f: => T): T = {acquireLock() ...}
}

(您甚至可以将锁定方法命名为 apply,这样您就可以执行 myLock{....} 而不是 myLock.lock{.. .})

除了多线程之外,您现在只需要一个 bool 变量来acquire/releaseLock

最后,如果需要支持多线程,则必须决定是否有多个线程可以进入锁(这样就不会是递归)。如果可以的话, bool 值应该替换为 DynamicVariable[Boolean] (或者可能是 java ThreadLocal,因为 DynamicVariable 是一个 InheritableThreadLocal,您可能想要也可能不想要)。如果不能,您只需在 acquire/releaseLock 中同步访问即可。

关于function - 自定义 scala 递归预防机制的改进,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7881194/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com