Скажем, у меня есть файл dictionary.txt
, мне нужно его прочитать на карте и использовать в моей программе, как я могу сделать этот файл dictionary.txt, содержащийся в скомпилированном exe файле?
Как скомпилировать ресурс в двоичный файл в Haskell?
Ответ 1
Вы можете хранить сериализованные типы данных в виде литералов. Вот пример:
- http://code.haskell.org/~dons/code/compiled-constants/
- http://haskell.org/haskellwiki/Compiling_in_constants
Тем не менее, file-embed автоматизирует этот процесс, упрощая работу с нетривиальными вложениями.
Все подходы, по существу, сводятся к представлению статических данных как литералу bytestring,
{-# LANGUAGE OverloadedStrings #-}
import Data.Binary
import qualified Data.Map as M
import qualified Data.ByteString.Char8 as S
import Data.ByteString.Lazy
import Codec.Compression.GZip
--
-- this is a gzip compressed literal bytestring, storing a binary-encoded Data.Map
--
mytable =
"\US\139\b\NUL\NUL\NUL\NUL\NUL\NUL\ETXEN\
\\219\SO\194 \f\197\224\188\196\CAN\227\US\
\\224\171~\NAKc\GS4ce\161`\178\191\215(\176\
\\190\180\167\231\210\n\241\171\203\191\ti\
\\157\217\149\249< \ENQ\214\&9>\202\162\179a\
\\132X\233\ESC=\231\215\164\SYN\157\DC2D\226*\
\\146\174o\t\167\DLE\209\"i_\240\193\129\199<W\
\\250nC\CAN\212\CAN\162J\160\141C\178\133\216;\
\\\@4\144-W\203\209x\205\140\166\RS\163\237]9f\
\\170\143\ACK\163g\223\STX\184\&7\rH\222\FSW\
\\130\&7D\197\NUL\164\&0U\193\186\t\186o\
\\228\180~\NUL\a6\249\137#\SOH\NUL\NUL"
main = print =<< M.lookup "ghc" m
where
-- build the table from the bytestring:
m :: M.Map String (Maybe String)
m = decode . decompress . fromChunks . return $ mytable
Ответ 2
Вам придется придумать свой способ, чтобы получить его на карте, но http://hackage.haskell.org/package/file-embed получит его в ваш скомпилированный двоичный файл. Мы используем его для встраивания шаблонов в некоторые из наших веб-приложений.
Ответ 3
Вы можете использовать собственный Makefile
/Setup.hs
крючок и вызывать windres
(если вы в Windows) или objcopy
/elfrc
(если вы работаете в Linux) для компиляции ресурсов в COFF/ELF, которые затем можно объединить с вашими объектными файлами Haskell, чтобы сформировать окончательный исполняемый файл. Затем вы можете получить доступ к ресурсам, используя Haskell FFI, как это (не тестировалось):
-- We have an image resource called "_imgdata"
foreign import ccall "&" _imgdata :: CString
-- Size of _imgdata is 405585 bytes.
imgdata :: CStringLen
imgdata = (_imgdata, 405585)
Это решение будет более эффективным, чем использование file-embed (нет CString
→ ByteString
), но также и более активное участие.
В стороне я также нуждался в поддержке файлов ресурсов во время моей работы над cabal-install, поэтому он, вероятно, будет интегрирован в некоторые будущую версию Cabal (если я ее реализую).
Ответ 4
Я настоятельно рекомендую использовать Template Haskell здесь, чтобы загрузить файл и превратить его в тип данных Haskell. Он будет не только скомпилирован в финальный двоичный файл, но и полностью оптимизирован.