snippetMinor
Haskell requires more memory than Python when I read map from file. Why?
Viewed 0 times
mapfilewhyreadthanrequiresmorepythonmemoryhaskell
Problem
I have this simple code in Python:
It requires 300MB for work. "baseforms.txt" size is 123M.
I've written the same code in Haskell:
`{-# LANGUAGE OverloadedStrings #-}
import qualified Data.Map as M
import qualified Data.ByteString.Lazy.Char8 as B
import Data.Text.Lazy.Encoding(decodeUtf8)
import qualified Data.Text.Lazy as T
import qualified Data.Text.Lazy.IO as I
import Control.Monad(liftM)
main = do
text
It requires 544 MB and it's slower than Python version. Why? Is it possible to optimise Haskell version?
Data file can be downloaded here
I've tried non-lazy version of Data.Text and Data.Text.IO - memory usage is near 650MB
I've also tried hashtables, but memory usage growth up to 870 MB
input = open("baseforms.txt","r",encoding='utf8')
S = {}
for i in input:
words = i.split()
S.update( {j:words[0] for j in words} )
print(S.get("sometext","not found"))
print(len(S))
It requires 300MB for work. "baseforms.txt" size is 123M.
I've written the same code in Haskell:
`{-# LANGUAGE OverloadedStrings #-}
import qualified Data.Map as M
import qualified Data.ByteString.Lazy.Char8 as B
import Data.Text.Lazy.Encoding(decodeUtf8)
import qualified Data.Text.Lazy as T
import qualified Data.Text.Lazy.IO as I
import Control.Monad(liftM)
main = do
text
It requires 544 MB and it's slower than Python version. Why? Is it possible to optimise Haskell version?
Data file can be downloaded here
I've tried non-lazy version of Data.Text and Data.Text.IO - memory usage is near 650MB
I've also tried hashtables, but memory usage growth up to 870 MB
Solution
I can't seem to download your data set. Could you re-upload a smaller dataset (say, ~5mb? and with your testing statistics for it) to Pastebin or somewhere?
The only obvious performance difference with your Python version is that
(There also seems to be a functional difference between your two versions if my understanding of Python hasn't rusted off after years of neglect.
On issues of style, see the below rewrite. Note particularly the reordering of definitions so as to eliminate the
The only obvious performance difference with your Python version is that
Data.Map is based on a binary tree, whereas Python's dict is a hashtable. This makes fromList an O(n * log n) operation in Haskell (unless you luck into the fact that your list of (form, base) pairs happen to all be distinct and sorted in ascending order) but constructing the Python dict is O(n). This should make a significant time difference, but not necessarily space, so at this point I'm out of ideas. Make sure you're compiling with -O2 I guess.(There also seems to be a functional difference between your two versions if my understanding of Python hasn't rusted off after years of neglect.
{j:words[0] for j in words} will pair words[0] with itself, so maybe you want parseLine to be defined as let forms@(base:_) = ....)On issues of style, see the below rewrite. Note particularly the reordering of definitions so as to eliminate the
where-clause. In your version it was easy to miss due to being inline with the do-block, you can do as I did, or try decreasing the indent.{-# LANGUAGE OverloadedStrings #-}
import qualified Data.Map as Map
import qualified Data.ByteString.Lazy as ByteString
import qualified Data.Text.Lazy as Text
import Data.Text.Lazy.Encoding (decodeUtf8)
main :: IO ()
main = do
text <- fmap decodeUtf8 $ ByteString.readFile "baseforms.txt"
let parseLine line = let (base:forms) = Text.words line
in zip forms (repeat base)
m = Map.fromList . concatMap parseLine . Text.lines $ text
print $ Map.lookup "sometext" m
print $ Map.size mCode Snippets
{-# LANGUAGE OverloadedStrings #-}
import qualified Data.Map as Map
import qualified Data.ByteString.Lazy as ByteString
import qualified Data.Text.Lazy as Text
import Data.Text.Lazy.Encoding (decodeUtf8)
main :: IO ()
main = do
text <- fmap decodeUtf8 $ ByteString.readFile "baseforms.txt"
let parseLine line = let (base:forms) = Text.words line
in zip forms (repeat base)
m = Map.fromList . concatMap parseLine . Text.lines $ text
print $ Map.lookup "sometext" m
print $ Map.size mContext
StackExchange Code Review Q#84984, answer score: 2
Revisions (0)
No revisions yet.