image - Go 生成的动画 GIF 在 Windows 中不起作用
问题描述
我发现一个示例在 Windows 中无法正常工作。这个程序演示了 Go 标准图像包的基本用法,我们将使用它来创建一系列位图图像,然后将序列编码为 GIF 动画。
package main
import (
"image"
"image/color"
"image/gif"
"io"
"math"
"math/rand"
"os"
)
import (
"log"
"net/http"
"time"
)
//!+main
var palette = []color.Color{color.White, color.Black}
const (
whiteIndex = 0 // first color in palette
blackIndex = 1 // next color in palette
)
func main() {
//!-main
// The sequence of images is deterministic unless we seed
// the pseudo-random number generator using the current time.
// Thanks to Randall McPherson for pointing out the omission.
rand.Seed(time.Now().UTC().UnixNano())
if len(os.Args) > 1 && os.Args[1] == "web" {
//!+http
handler := func(w http.ResponseWriter, r *http.Request) {
lissajous(w)
}
http.HandleFunc("/", handler)
//!-http
log.Fatal(http.ListenAndServe("localhost:8000", nil))
return
}
//!+main
lissajous(os.Stdout)
}
func lissajous(out io.Writer) {
const (
cycles = 5 // number of complete x oscillator revolutions
res = 0.001 // angular resolution
size = 100 // image canvas covers [-size..+size]
nframes = 64 // number of animation frames
delay = 8 // delay between frames in 10ms units
)
freq := rand.Float64() * 3.0 // relative frequency of y oscillator
anim := gif.GIF{LoopCount: nframes}
phase := 0.0 // phase difference
for i := 0; i < nframes; i++ {
rect := image.Rect(0, 0, 2*size+1, 2*size+1)
img := image.NewPaletted(rect, palette)
for t := 0.0; t < cycles*2*math.Pi; t += res {
x := math.Sin(t)
y := math.Sin(t*freq + phase)
img.SetColorIndex(size+int(x*size+0.5), size+int(y*size+0.5),
blackIndex)
}
phase += 0.1
anim.Delay = append(anim.Delay, delay)
anim.Image = append(anim.Image, img)
}
gif.EncodeAll(out, &anim) // NOTE: ignoring encoding errors
}
代码在 cmd 中正常运行,但是,如果我在 Windows Power Shell 中运行它,例如:
.\lissajous.exe >out.gif
out.gif 无法打开,我不知道为什么。
解决方案
GIF 文件(gif 数据)是二进制格式,而不是文本格式。尝试将其写入标准输出并将其重定向到文件可能会遭受转换。例如,Windows PowerShell 很可能会转换一些控制字符(例如"\n"
to "\r\n"
),因此生成的二进制文件与gif.EncodeAll()
写入标准输出的二进制文件不同。显然cmd.exe
不做这样的转变。
我建议直接写入文件(您可以将一个os.File
作为输出传递),或者一个内存缓冲区,您可以使用ioutil.WriteFile()
.
以下是直接写入文件的样子:
f, err := os.Create("a.gif")
if err != nil {
panic(err)
}
defer f.Close()
lissajous(f)
以下是内存解决方案的外观:
buf := &bytes.Buffer{}
lissajous(buf)
if err := ioutil.WriteFile("a.gif", buf.Bytes(), 0666); err != nil {
panic(err)
}
请参阅相关问题:图像/gif:EncodeAll 的结果在 GNOME 之眼中不可见
某些应用程序仍有可能无法读取结果(请参见上述问题),可以通过使用以下命令在 unix 中转换输出图像来解决此问题:
convert original.gif -coalesce unoptimized.gif
推荐阅读
- android - 如何在另一个 RecyclerView 的项目中实现 RecyclerView
- python - 将 caffe 上的 net forward 从 MATLAB 转换为 Python 的问题
- excel - 如何创建允许使用 UserForm 在列中连续输入数据的数据条目
- nginx - 在 Azure 上处理 Kubernetes 集群的并发请求
- php - 如何将 2 个字符串插入数据库的 1 个列中?(代码点火器)
- sql - 如何从联合中选择一列
- android - 如何在 Android Studio 中构建 Android 包/应用程序。构建此应用程序的任何其他方式
- c++ - 通过 typedef 强制模板实例化:g++ 成功,Visual C++ 失败
- typescript - 如何在打字稿中使用 jest
- angular - 在 Angular HttpClient 中识别错误源(客户端或服务器)