Goldbach's Conjecture (Haskell)

From LiteratePrograms

Jump to: navigation, search

Introduction

The Goldbach conjecture says that any even integer greater than 2 can be expressed as the sum of two prime numbers. Note that most numbers will have more than one unique pair--this implementation tries to find all of these. Note also that in his work, Goldbach considered 1 a prime number. This conjecture has been shown to be true for numbers in the 10^18 range, but has yet to be proven for all such cases.

For more information, visit wikipedia's entry.

Implementation

This program is a code dump.
Code dumps are articles with little or no documentation or rearrangement of code. Please help to turn it into a literate program. Also make sure that the source of this code does consent to release it under the MIT or public domain license.

<<goldbach.hs>>=
module Main (goldbach, main) where
import Prelude
import System.Environment
import System.IO (hFlush, stdout)
import Primes (primes)             -- Updated 10/19/09, Added module (see below)
{- The Goldbach Conjecture: An Implementation
   James Brannan (irregularexpression@gmail.com)
-}
-- For convenience
type Pair = (Integer, Integer)
{- showPList: Prints a list of Pairs and formats them as a
              set into columns between braces. -}
showPList :: [Pair] -> IO ()
showPList [] = putStr "{ }\n"
showPList plist = do
    putStr "{ " 
    fmt 1 plist
    putStr " }\n"  where
        showPair (x, y) = "(" ++ show x ++ "+" ++ show y ++ ")"
        fmt n (y:ys) = do
            putStr (showPair y)
            -- Every 4 elements, start a new column.
            if n `mod` 4 == 0 then 
                if ys == [] then return () else putStr "\n  "
             else putStr "   "
            fmt (n+1) ys
        fmt _ [] = return ()
{- findsum: Finds a sum x by zipping a list (of primes) less than
            x with itself (excluding repeats) and checking for 
            pairs that add up to x. I remove a lot of the garbage
            by throwing away values
            of z greater than y (to remove [x,y] == [y,x]). -}
findsum :: Integer -> [Integer] -> [Pair]
findsum x plist = [(y,z) | y <- plist, z <- plist, y+z == x, y<=z]
{- goldbach: Takes a positive even integer x and exhaustively checks
             pairs of primes less than that integer that add up to
             precisely x. Note we chop off half the list -- each 
             pair has a non-unique equivalent => (a,b) = (b,a) -}
goldbach :: Integer -> [Pair]
goldbach x | odd x || x < 3 = error "Must be a positive even integer greater than 2"
           | otherwise = findsum x plist where
                -- Find list of primes smaller than x.
                plist = takeWhile (< x) primes
{- main: Driver interface for the defined functions. On first run,
         an introduction message will be displayed, otherwise only
         the prompt will be shown. -}
main :: IO ()
main = do
    args <- getArgs
    let args_num = [(read x::Integer) | x <- args]
    case args of
        [] -> helper True                 -- No arguments, run interactive
        _  -> mapM_ showvalue args_num    -- Arguments given, run through
    where
    -- I *hate* nested paren's (too much lisp...) Use $!
    showvalue x = do putStr $ resultMsg x $ length l
                     showPList l
        where l = goldbach x
    helper b = do
        intro b    -- Show intro message depending on b
        {- I have to import some IO functions here
           because by default the output is buffered
           and won't print until an \n is found, meaning
           user input can't be on the same line as the
           prompt. Without flushing the output first, 
           the program executes out of order (when compiled).
        -}
        putStr "[Number or 0] -> "
        hFlush stdout
        x <- readLn :: IO Integer -- Get input and calculate or exit
        exec x
    -- Intro message to be shown when the program is first run
    intro True = putStr ("The Goldbach Conjecture states that any positive, " ++
                         "even integer (>2) can be expressed as the sum of " ++
                         "two prime numbers. This application is an " ++
                         "algorithmic solver for the Goldbach Conjecture. " ++
                         "Begin by entering values, and the appropriate sets " ++
                         "of prime numbers will be returned. You may quit at " ++
                         "any time by typing the number 0.\n\n")
    intro False = return ()
    resultMsg int res = "\nGoldbach(" ++ show int ++ ") = " ++ show res ++ " unique sets:\n"
    -- Allow user to quit by typing 0
    exec 0 = putStr "\tQuitting...\n"
    exec y = do showvalue y
             putStr "\n"
             helper False -- Call without intro msg until user quits.
-- James Brannan (irregularexpression@gmail.com) 9/18/09
<<primes.hs>>=
module Primes (primes) where
primes :: [Integer]
primes = 1:2:3:primes'
  where
	{- Throw away first value (always 1), capture prime, and candidates
	   (using that fact that all *candidate* primes but 2 and 3 are of
	   the form [6k+1] and [6k+5] for any integer k (but not all) -}
    1:p:candidates = [6*k+r | k <- [0..], r <- [1,5]]
	-- Create list of primes by adding the found prime p to the list of 
	-- candidates after they have been filtered for primes
    primes' = p : filter isPrime candidates
	-- Finds if n is prime by testing all elements p where (p^2) is not
	-- greater than n from the list of generated primes
    isPrime n = all (not . divides n) $ takeWhile (\p -> p*p <= n) primes'
	-- Divides is true if the remainder between n and p is 0
    divides n p = n `mod` p == 0	
Download code
Views
Personal tools