- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我在 this page 上找到了一个使用 Haskell 编写 PNG 文件的小型库。 .我只是重新排列它,使其支持所有单色、灰度和 RGB 输出。
但是,似乎我在编写大型单色图像时总是出现堆栈溢出,但如果我使用灰度或 RGB 则不会。在此示例中,大小阈值大约为 2000:如果我将 width
设置为小于该值,则会生成图像,否则会发生堆栈溢出。
import Png
import qualified Data.ByteString.Lazy as B
width = 2000 :: Int
main = do
let setG = [ [ (r + c) `mod` 256 | c <- [0..width]] | r <- [0..width]]
let outputG = pngGrayscale setG
putStrLn "Writing grayscale image..."
B.writeFile "grayscale.png" outputG
putStrLn "Done"
let setR = [ [ (r `mod` 256, c `mod` 256, (r+c) `mod` 256) | c <- [0..width]] | r <- [0..width]]
let outputR = pngRGB setR
putStrLn "Writing RGB image..."
B.writeFile "rgb.png" outputR
putStrLn "Done"
let setM = [ [ even (r + c) | c <- [0..width]] | r <- [0..width]]
let outputM = pngMonochrome setM
putStrLn "Writing monochrome image..."
B.writeFile "monochrome.png" outputM
putStrLn "done"
因为这三个函数 png*
之间唯一的显着差异似乎是对 bitpack*
的调用,我想这是罪魁祸首,但我不知道如何解决这个问题。
这是图书馆(原文可以找到here):
{-
A small library for creating monochrome PNG files.
This file is placed into the public domain.
Dependencies: Zlib.
-}
module Png (pngRGB, pngGrayscale, pngMonochrome) where
import Data.Array
import Data.Bits
import Data.List
import Data.Word
import qualified Codec.Compression.Zlib as Z
import qualified Data.ByteString.Lazy as B
import Control.DeepSeq (deepseq)
be8 :: Word8 -> B.ByteString
be8 x = B.singleton x
be32 :: Word32 -> B.ByteString
be32 x = B.pack [fromIntegral (x `shiftR` sh) | sh <- [24,16,8,0]]
pack :: String -> B.ByteString
pack xs = B.pack $ map (fromIntegral.fromEnum) xs
unpack :: B.ByteString -> String
unpack xs = map (toEnum.fromIntegral) (B.unpack xs)
hdr, iHDR, iDAT, iEND :: B.ByteString
hdr = pack "\137\80\78\71\13\10\26\10"
iHDR = pack "IHDR"
iDAT = pack "IDAT"
iEND = pack "IEND"
chunk :: B.ByteString -> B.ByteString -> [B.ByteString]
chunk tag xs = [be32 (fromIntegral $ B.length xs), dat, be32 (crc dat)]
where dat = B.append tag xs
-- | Return a monochrome PNG file from a two dimensional bitmap
-- stored in a list of lines represented as a list of booleans.
pngMonochrome :: [[Bool]] -> B.ByteString
pngMonochrome dat = B.concat $ hdr : concat [ihdr, imgdat, iend]
where height = fromIntegral $ length dat
width = fromIntegral $ length (head dat)
ihdr = chunk iHDR (B.concat [
be32 width, be32 height, be8 1, be8 0, be8 0, be8 0, be8 0])
imgdat = chunk iDAT (Z.compress imgbits)
imgbits = B.concat $ map scanlineMonochrome dat
iend = chunk iEND B.empty
scanlineMonochrome :: [Bool] -> B.ByteString
scanlineMonochrome dat = 0 `B.cons` bitpackMonochrome dat
bitpackMonochrome' :: [Bool] -> Word8 -> Word8 -> B.ByteString
bitpackMonochrome' [] n b = if b /= 0x80 then B.singleton n else B.empty
bitpackMonochrome' (x:xs) n b =
if b == 1
then v `B.cons` bitpackMonochrome' xs 0 0x80
else bitpackMonochrome' xs v (b `shiftR` 1)
where v = if x then n else n .|. b
bitpackMonochrome :: [Bool] -> B.ByteString
bitpackMonochrome xs = bitpackMonochrome' xs 0 0x80
crc :: B.ByteString -> Word32
crc xs = updateCrc 0xffffffff xs `xor` 0xffffffff
updateCrc :: Word32 -> B.ByteString -> Word32
updateCrc = B.foldl' crcStep
crcStep :: Word32 -> Word8 -> Word32
crcStep crc ch = (crcTab ! n) `xor` (crc `shiftR` 8)
where n = fromIntegral (crc `xor` fromIntegral ch)
crcTab :: Array Word8 Word32
crcTab = listArray (0,255) $ flip map [0..255] (\n ->
foldl' (\c k -> if c .&. 1 == 1
then 0xedb88320 `xor` (c `shiftR` 1)
else c `shiftR` 1) n [0..7])
white, black :: Int
white = 255
black = 0
-- | Produces a single grayscale bit given a percent black
gray :: Int -> Int
gray percent = 255 - floor (fromIntegral percent * 2.55)
-- | Return a grayscale PNG file from a two dimensional bitmap stored in a list
-- of lines represented as a list of 0-255 integer values.
pngGrayscale :: [[Int]] -> B.ByteString
pngGrayscale dat = B.concat $ hdr : concat [ihdr, imgdat, iend]
where height = fromIntegral $ length dat
width = fromIntegral $ length (head dat)
ihdr = chunk iHDR $ B.concat
[ be32 width
, be32 height
, be8 8 -- bits per pixel
, be8 0 -- color type
, be8 0 -- compression method
, be8 0 -- filter method
, be8 0 ] -- interlace method
imgdat = chunk iDAT (Z.compress imgbits)
imgbits = B.concat $ map scanlineGrayscale dat
iend = chunk iEND B.empty
scanlineGrayscale :: [Int] -> B.ByteString
scanlineGrayscale dat = B.pack (0 : map fromIntegral dat)
-- | Return a RGB PNG file from a two dimensional bitmap stored in a list
-- of lines represented as a list of triples of 0-255 integer values.
pngRGB :: [[(Int,Int,Int)]] -> B.ByteString
pngRGB dat = B.concat $ hdr : concat [ihdr, imgdat ,iend]
where height = fromIntegral $ length dat
width = fromIntegral $ length (head dat)
ihdr = chunk iHDR $ B.concat
[ be32 height
, be32 width
, be8 8 -- bits per sample (8 for r, 8 for g, 8 for b)
, be8 2 -- color type (2=rgb)
, be8 0 -- compression method
, be8 0 -- filter method
, be8 0 ] -- interlace method
imgdat = chunk iDAT (Z.compress imagedata)
imagedata = B.concat $ map scanlineRGB dat
iend = chunk iEND B.empty
scanlineRGB :: [(Int,Int,Int)] -> B.ByteString
scanlineRGB dat = B.pack (0 : (map fromIntegral $ concatMap (\(r,g,b) -> [r,g,b]) dat))
最佳答案
罪魁祸首是
bitpackMonochrome' :: [Bool] -> Word8 -> Word8 -> B.ByteString
bitpackMonochrome' [] n b = if b /= 0x80 then B.singleton n else B.empty
bitpackMonochrome' (x:xs) n b =
if b == 1
then v `B.cons` bitpackMonochrome' xs 0 0x80
else bitpackMonochrome' xs v (b `shiftR` 1)
where v = if x then n else n .|. b
它使用 B.cons
连接 ByteString
。这无论如何都是相当低效的,而且 B.cons
在它的第二个参数中是严格的。
因此,你会得到一个巨大的(对于 2000×2000
位图来说大约有 50 万深)thunk 的形式
v1 `B.cons` (v2 `B.cons` (v3 ...)))
溢出堆栈。
一个简单的解决方法 - 仍然相当低效 - 是在 bitpackMonochrome'
中使用列表,因为 (:)
在其第二个参数中是惰性的,
bitpackMonochrome :: [Bool] -> B.ByteString
bitpackMonochrome xs = B.pack $ bitpackMonochrome' xs 0 0x80
bitpackMonochrome' :: [Bool] -> Word8 -> Word8 -> [Word8]
bitpackMonochrome' [] n b = if b /= 0x80 then [n] else []
bitpackMonochrome' (x:xs) n b =
if b == 1
then v : bitpackMonochrome' xs 0 0x80
else bitpackMonochrome' xs v (b `shiftR` 1)
where v = if x then n else n .|. b
和 B.pack
在 bitpackMonochrome
中。
这样,你就不会得到巨大的 thunk,因为 (:)
可以在它的第二个参数之前被评估。
一个更高效的版本会根据维度计算所需的大小并使用
create :: Int -> (Ptr Word8 -> IO ()) -> IO ByteString
甚至
unsafeCreate :: Int -> (Ptr Word8 -> IO ()) -> ByteString
从 Data.ByteString.Internal
直接填充正确大小的分配缓冲区。
关于haskell - 创建单色 PNG 图像时堆栈溢出,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15731002/
我正在寻找一种经济合理的解决方案来长时间存储图片。我读到 PNG 文件格式,它与 JPEG 相比具有优越的特性,即在这些类别中: 无专利、无许可、无版税 无质量损失 尚未压缩 我有很多来自 Photo
关闭。这个问题是off-topic .它目前不接受答案。 想改善这个问题吗? Update the question所以它是 on-topic对于堆栈溢出。 9年前关闭。 Improve this q
我怎么能用 FFMEG 做到这一点? 输入 : 背景图片:background.png 图片帧列表:image_001.png,image_002.png ...(每张图片为一帧) 输出:所有帧都有背
$ cat png.ll ./packages/apps/Mms/res/mipmap-hdpi/ic_launcher_smsmms.png ./packages/apps/Mms/res/draw
这个问题在这里已经有了答案: Natural Sort Order in C# (18 个答案) 关闭 7 年前。 这是我的代码: private void Method1() { int
我一直在考虑用 Webp 图像替换我的 Android 应用程序中的 png 文件以减小 APK 大小。 虽然结果不错,但我想知道我是否使用了一些 png 转换器/压缩器,并且能够将尺寸减小到比我为
在 gnuplot-4.2.6 中,我可以使用 set term png medium x000000 xffffff set output 'file.png' plot x 这将生成一个带有黑色背
背景: 我正在努力使一堆 PNG 尽可能小。我正在使用诸如 PngOut、PngCrush 和 OptiPng 之类的工具。 问题: 我遇到了一个大小为 1434 KB 但只有 230 x 230 像
我正在使用 ImageMagick 调整图像大小。如果我传递 -resize WxH 选项,它会按预期运行。但是如果我通过 -resize WxH! (在调整大小时忽略纵横比),一些图像,尤其是 PN
如何访问/删除 PNG 元数据? 我正在寻找 Mac 应用程序或 PHP 代码段。 最佳答案 抱歉发布了一个 Windows 软件,但如果你没有找到任何对 MAC 有用的东西,那就是 TweakPNG
到目前为止似乎没有任何效果。我看到了 pnginfo以下消息: concept_Sjet_dream6.png... Image Width: 200 Image Length: 240 Bi
我有一个带有 Alpha channel (即透明度)的 PNG 图像,我需要创建将图像层合成到白色背景上的版本。我想使用可编写脚本的命令,使用 CLI 工具(例如 Image Magick)将 PN
我是初学者。我昨天问了一个类似的问题,但不知何故被否决了。所以这次我尽量简化问题。 带有 alpha png 的 24 位与 32 位 png 相同吗? 非常感谢您的一些提示。 最佳答案 没有“24
我有这个带点的荷兰 pdf 图像: pdf image of the netherlands with dots 当我尝试将此 pdf 转换为 png 图像时,使用 pdftools和 png像这样:
我在我的启动图像通用项目中添加了“Default.png,Default-568h@2x.png,Default@2x.png”这三个文件,我有三个不同的图像,分辨率与苹果中提到的完全相同文档,适用于
我在 Python 中使用 google app engine 并有几个静态 .png 图像文件,但它们都以“image/x-png”内容类型提供。这是一个问题,当我使用像 chrome 这样的浏览器
我做了一个 python 脚本,该脚本设法根据特定模式解散乱序(png)图像,该 python 脚本使用 ffmpeg 并进行 12 次编码来解乱它(通过裁剪特定部分并将其粘贴到现有图片上)。 因此,
我有一个 PNG 图像文件。我想将其转换为 GeoTiff。我安装了 QGIS 软件,但无法使用它,也不知道如何对图像进行地理配准。请帮我。有没有在线软件? 最佳答案 这是一个非常好的教程,其中包含有
我有一堆使用我编写的 Java 图表工具创建的图表 - 它们主要是黑白图表,带有浅绿色的块,偶尔还有其他颜色。它们当前被保存为 JPG 文件,我想将它们插入到我准备按需打印的书中。 这本书是一个 Op
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 7年前关闭。 Improve thi
我是一名优秀的程序员,十分优秀!