How to generate the power set of a set in Scala

前端 未结 8 1947
佛祖请我去吃肉
佛祖请我去吃肉 2020-12-13 14:41

I have a Set of items of some type and want to generate its power set.

I searched the web and couldn\'t find any Scala code that adresses this specific task.

相关标签:
8条回答
  • 2020-12-13 15:16

    Can be as simple as:

    def powerSet[A](xs: Seq[A]): Seq[Seq[A]] = 
      xs.foldLeft(Seq(Seq[A]())) {(sets, set) => sets ++ sets.map(_ :+ set)}
    

    Recursive implementation:

    def powerSet[A](xs: Seq[A]): Seq[Seq[A]] = {
      def go(xsRemaining: Seq[A], sets: Seq[Seq[A]]): Seq[Seq[A]] = xsRemaining match {
        case Nil => sets
        case y :: ys => go(ys, sets ++ sets.map(_ :+ y))
      }
    
      go(xs, Seq[Seq[A]](Seq[A]()))
    }
    
    0 讨论(0)
  • 2020-12-13 15:17

    Notice that if you have a set S and another set T where T = S ∪ {x} (i.e. T is S with one element added) then the powerset of T - P(T) - can be expressed in terms of P(S) and x as follows:

    P(T) = P(S) ∪ { p ∪ {x} | p ∈ P(S) }
    

    That is, you can define the powerset recursively (notice how this gives you the size of the powerset for free - i.e. adding 1-element doubles the size of the powerset). So, you can do this tail-recursively in scala as follows:

    scala> def power[A](t: Set[A]): Set[Set[A]] = {
       |     @annotation.tailrec 
       |     def pwr(t: Set[A], ps: Set[Set[A]]): Set[Set[A]] =
       |       if (t.isEmpty) ps
       |       else pwr(t.tail, ps ++ (ps map (_ + t.head)))
       |
       |     pwr(t, Set(Set.empty[A])) //Powerset of ∅ is {∅}
       |   }
    power: [A](t: Set[A])Set[Set[A]]
    

    Then:

    scala> power(Set(1, 2, 3))
    res2: Set[Set[Int]] = Set(Set(1, 2, 3), Set(2, 3), Set(), Set(3), Set(2), Set(1), Set(1, 3), Set(1, 2))
    

    It actually looks much nicer doing the same with a List (i.e. a recursive ADT):

    scala> def power[A](s: List[A]): List[List[A]] = {
       |     @annotation.tailrec 
       |     def pwr(s: List[A], acc: List[List[A]]): List[List[A]] = s match {
       |       case Nil     => acc 
       |       case a :: as => pwr(as, acc ::: (acc map (a :: _)))
       |     }
       |     pwr(s, Nil :: Nil)
       |   }
    power: [A](s: List[A])List[List[A]]
    
    0 讨论(0)
  • 2020-12-13 15:23

    Use the built-in combinations function:

    val xs = Seq(1,2,3)
    (0 to xs.size) flatMap xs.combinations
    
    // Vector(List(), List(1), List(2), List(3), List(1, 2), List(1, 3), List(2, 3),
    // List(1, 2, 3))
    

    Note, I cheated and used a Seq, because for reasons unknown, combinations is defined on SeqLike. So with a set, you need to convert to/from a Seq:

    val xs = Set(1,2,3)
    (0 to xs.size).flatMap(xs.toSeq.combinations).map(_.toSet).toSet
    
    //Set(Set(1, 2, 3), Set(2, 3), Set(), Set(3), Set(2), Set(1), Set(1, 3), 
    //Set(1, 2))
    
    0 讨论(0)
  • 2020-12-13 15:27

    All the other answers seemed a bit complicated, here is a simple function:

        def powerSet (l:List[_]) : List[List[Any]] =
          l match {
           case Nil => List(List())
           case x::xs =>
             var a = powerSet(xs)
             a.map(n => n:::List(x)):::a
          }
    

    so

        powerSet(List('a','b','c'))
    

    will produce the following result

        res0: List[List[Any]] = List(List(c, b, a), List(b, a), List(c, a), List(a), List(c, b), List(b), List(c), List())
    
    0 讨论(0)
  • 2020-12-13 15:28

    Here's one of the more interesting ways to write it:

    import scalaz._, Scalaz._
    
    def powerSet[A](xs: List[A]) = xs filterM (_ => true :: false :: Nil)
    

    Which works as expected:

    scala> powerSet(List(1, 2, 3)) foreach println
    List(1, 2, 3)
    List(1, 2)
    List(1, 3)
    List(1)
    List(2, 3)
    List(2)
    List(3)
    List()
    

    See for example this discussion thread for an explanation of how it works.

    (And as debilski notes in the comments, ListW also pimps powerset onto List, but that's no fun.)

    0 讨论(0)
  • 2020-12-13 15:31

    Here's another (lazy) version... since we're collecting ways of computing the power set, I thought I'd add it:

    def powerset[A](s: Seq[A]) =
      Iterator.range(0, 1 << s.length).map(i =>
        Iterator.range(0, s.length).withFilter(j =>
          (i >> j) % 2 == 1
        ).map(s)
      )
    
    0 讨论(0)
提交回复
热议问题