Problem 165

http://projecteuler.net/index.php?section=problems&id=165
この問題は結構苦労した。
線分交差列挙問題
といったら平面走査法とおもい、実装しようと試みた(前から実装してみたいとは思っていた)が…
無理
例外処理が予想以上に大変だ。

処理しなくてはいけないと思われる例外

  • 走査線と平行な線分
  • 端点での交差
  • 3線分以上の交差

線分を蓄えるデータ構造(平衡木など)が必要になるが、
この要素(線分)の順序関係が動的に変化していくことと例外の存在が非常に厄介。

どこからか「そんな時は記号摂動だよ」という声が聞こえてくるがそんな気力はない。
というかそもそも平面走査法の計算量は交差数をkとすると O ((n+k)log n) (実は O(nlog n + k)があるらしいが複雑らしい)。
つまり、線分が密なとき(k が n^2 程度 )は大して効率よくないよ。
この問題は [0,499]x[0,499]のなかに5000本の線分があるんだから密に決まってるよ。
というわけで O (n^2) のナイーブな解法を採用。

  • 計算終わらない -> 有理数方から実数型
  • 答えあわない -> 共有点の個数ではなく 異なる 共有点の個数だった。
  • なんか遅い -> 重複要素を取り除くのに quick sort みたいな nub を使った (できたリストの順序は非保存、quick sort は もらってきた)
import Data.Maybe (mapMaybe)
import Data.Function (on)
import Control.Arrow ((***))

type Point = (Integer,Integer)
type Segment = (Point,Point,Integer,Integer,Integer)
type Intersection = (Double,Double)
blumBlumShub :: [Integer]
blumBlumShub = map (flip mod 500).tail.iterate sqMod $ 290797 
    where sqMod s = mod (s*s) 50515093

takeSegs :: Int -> [Integer] -> [Segment]
takeSegs 0 _ = []
takeSegs (n+1) (a:b:c:d:es) = segment (a,b) (c,d) : takeSegs n es

segment :: Point -> Point -> Segment
segment p@(x1,y1) q@(x2,y2) = (p,q,a,b,a*x1+b*y1)
    where a = y1-y2
          b = x2-x1

innerSeg (p1,q1,_,_,_) (x,y) = closure (fst p1,fst q1) x && closure (snd p1,snd q1) y && not endpoint
    where closure (a,b) c = fromIntegral (min a b) <= c && c <= fromIntegral (max a b)
          endpoint = same p1 (x,y) || same q1 (x,y)
          same (a,b) (c,d) = fromIntegral a == c && fromIntegral b == d

overlapp (a,b) (c,d) = min a b < max c d || max a b > min b c

intersection :: Segment -> Segment -> Maybe Intersection
intersection l1@(p1,q1,a1,b1,d1) l2@(p2,q2,a2,b2,d2)
    | overlappX && overlappY &&  det /= 0 &&
      innerSeg l1 (x,y)  && innerSeg l2 (x,y) = Just (x,y)
    | otherwise = Nothing 
    where overlappX = on overlapp (fst***fst) (p1,q1) (p2,q2)
          overlappY = on overlapp (snd***snd) (p1,q1) (p2,q2)
          det = fromIntegral $ a1*b2-a2*b1
          x = fromIntegral (b2*d1-b1*d2) / det
          y = fromIntegral (a1*d2-a2*d1) / det


choose2 :: [a] -> [(a, a)]
choose2 [] = []
choose2 (x:xs) = map ((,) x) xs ++ choose2 xs

main = print.length.quicknub.intersections $ 5000
intersections n = mapMaybe (uncurry intersection).choose2 $ takeSegs n blumBlumShub

-- quick sort like nub
quicknub :: Ord a => [a] -> [a]
quicknub xs = qnub compare xs []

qnub :: (a -> a -> Ordering) -> [a] -> [a] -> [a]
qnub _   []     r = r
qnub _   [x]    r = x:r
qnub cmp (x:xs) r = qpart cmp x xs [] [] r

qpart :: (a -> a -> Ordering) -> a -> [a] -> [a] -> [a] -> [a] -> [a]
qpart cmp x [] rlt rge r =
    qnub cmp rlt (x:qnub cmp rge r)
qpart cmp x (y:ys) rlt rge r =
    case cmp x y of
        LT -> qpart cmp x ys rlt (y:rge) r
        EQ -> qpart cmp x ys rlt rge r
        GT -> qpart cmp x ys (y:rlt) rge r

メモリ足りなくなったり、苦労したんだけど、遅いし、きれいなコードじゃない。