2010-05-23 7 views

Trả lời

17

Tôi nghĩ rằng tôi đã tìm ra cách để làm điều đó.

def combine(acc:Set[String], set:Set[String]) = for (a <- acc; s <- set) yield { 
    a + "&" + s 
} 

val expanded = sets.reduceLeft(combine) 

expanded: scala.collection.immutable.Set[java.lang.String] = Set(b&2&T, a&1&S, 
    a&1&T, b&1&S, b&1&T, c&1&T, a&2&T, c&1&S, c&2&T, a&2&S, c&2&S, b&2&S) 
+0

Đồng bộ hóa! Trong giải pháp của tôi, xây dựng hoãn lại các chuỗi cho đến bước cuối cùng, nhưng về cơ bản là giống nhau. – retronym

12

Câu hỏi hay. Dưới đây là một cách:

scala> val seqs = Seq(Seq("a","b","c"), Seq("1","2"), Seq("S","T"))     
seqs: Seq[Seq[java.lang.String]] = List(List(a, b, c), List(1, 2), List(S, T)) 

scala> val seqs2 = seqs.map(_.map(Seq(_))) 
seqs2: Seq[Seq[Seq[java.lang.String]]] = List(List(List(a), List(b), List(c)), List(List(1), List(2)), List(List(S), List(T))) 

scala> val combined = seqs2.reduceLeft((xs, ys) => for {x <- xs; y <- ys} yield x ++ y) 
combined: Seq[Seq[java.lang.String]] = List(List(a, 1, S), List(a, 1, T), List(a, 2, S), List(a, 2, T), List(b, 1, S), List(b, 1, T), List(b, 2, S), List(b, 2, T), List(c, 1, S), List(c, 1, T), List(c, 2, S), List(c, 2, T)) 

scala> combined.map(_.mkString("&"))    
res11: Seq[String] = List(a&1&S, a&1&T, a&2&S, a&2&T, b&1&S, b&1&T, b&2&S, b&2&T, c&1&S, c&1&T, c&2&S, c&2&T) 
+0

Cảm ơn. Lần đầu tiên tôi đã thử với foldLeft và cuối cùng đã tìm ra tôi cần sử dụng reduceLeft thay thế (giữ lại kết quả trống). Việc chuyển đổi sang Seq chỉ cần thiết để duy trì thứ tự? – huynhjl

+0

Kết hợp vào cuối sẽ thực sự hữu ích bởi vì các bộ thực tế trong cùng một không gian và tôi cần phải kết hợp "b & a & a" vào "a & b" (loại bỏ dups và đặt hàng kết hợp). – huynhjl

+0

@huynhjl Việc sử dụng seq (để bắt đầu với) có lẽ để từ viết tắt có thể tránh nhập 'scala.collection.immutable.Set', và có thể cho thấy rằng điều này có thể được thực hiện với giao diện tổng quát hơn. –

6

đưa ra sau khi batle;) nhưng có một khác:

sets.reduceLeft((s0,s1)=>s0.flatMap(a=>s1.map(a+"&"+_))) 
3

Mở rộng trên @Patrick's answer. Bây giờ là nói chung và lazier hơn:

def combine[A](f:(A, A) => A)(xs:Iterable[Iterable[A]]) = 
    xs.reduceLeft { (x, y) => x.view.flatMap {a => y.map(f(a, _)) } } 

Có nó được lười biếng cho phép bạn tiết kiệm không gian, vì bạn không lưu trữ các mặt hàng theo cấp số nhân nhiều trong tập mở rộng; thay vào đó, bạn tạo ra chúng trên bay. Tuy nhiên, nếu bạn thực sự muốn tập hợp đầy đủ, bạn có thể vẫn nhận được nó như vậy:

val expanded = combine{(x:String, y:String) => x + "&" + y}(sets).toSet 
+0

Đây là giải pháp tổng quát hơn cho phép áp dụng bản đồ trước khi lấy sản phẩm Descartes: http://stackoverflow.com/a/4515050/244526 – dsg

3

Mở rộng trên dsg's answer, bạn có thể viết nó rõ ràng hơn (tôi nghĩ) Bằng cách này, nếu bạn không nhớ các cà ri chức năng:

def combine[A](f: A => A => A)(xs:Iterable[Iterable[A]]) = 
    xs reduceLeft { (x, y) => x.view flatMap { y map f(_) } } 

Một thay thế (hơi dài, nhưng nhiều dễ đọc hơn):

def combine[A](f: (A, A) => A)(xs:Iterable[Iterable[A]]) = 
    xs reduceLeft { (x, y) => for (a <- x.view; b <- y) yield f(a, b) } 

Cách sử dụng:

combine[String](a => b => a + "&" + b)(sets) // curried version 

combine[String](_ + "&" + _)(sets)    // uncurried version 
+0

tuyệt vời, cảm ơn! – dsg