
本文旨在详细解析 golang Tour Exercise 36 (tour.golang.org#36),即 `pic.Show` 函数的功能实现。我们将深入探讨该函数如何利用传入的图像数据生成图像,并将其编码为 Base64 字符串,最终通过特殊的方式在 Go Playground 中显示出来。本文将提供代码示例和详细解释,帮助读者理解图像生成和显示的原理。
在 golang Tour 的 Exercise 36 中,我们遇到了 pic.Show 函数,它接受一个函数作为参数,该函数负责生成图像数据。初学者可能会困惑,为什么运行示例代码后,控制台只输出了字符串,而不是预期的图像。本文将深入解析 pic.Show 的内部实现,揭示图像生成的全过程,并解释为何在 Go Playground 中能看到图像。
pic.Show 函数的内部机制
pic.Show 函数的核心功能是将图像数据转换为 Base64 编码的字符串,然后打印到标准输出。Go Playground 会解析标准输出中的特定格式的字符串,并将其渲染成图像。
立即学习“go语言免费学习笔记(深入)”;
让我们分解 pic.Show 函数的实现:
-
定义图像尺寸:
const ( dx = 256 dy = 256 )
pic.Show 函数内部定义了图像的宽度 dx 和高度 dy,均为 256 像素。
-
获取图像数据:
data := f(dx, dy)
pic.Show 接收一个函数 f 作为参数,该函数接受宽度和高度作为输入,并返回一个二维的 uint8 切片,表示图像的像素数据。data 变量存储了由 f 函数生成的图像数据。
-
创建 image.NRGBA 对象:
m := image.NewNRGBA(image.Rect(0, 0, dx, dy))
image.NewNRGBA 函数创建了一个新的 image.NRGBA 对象,用于存储图像数据。image.Rect(0, 0, dx, dy) 定义了图像的矩形区域,左上角坐标为 (0, 0),右下角坐标为 (dx, dy)。
-
填充像素数据:
for y := 0; y < dy; y++ { for x := 0; x < dx; x++ { v := data[y][x] i := y*m.Stride + x*4 m.Pix[i] = v m.Pix[i+1] = v m.Pix[i+2] = 255 m.Pix[i+3] = 255 } }这段代码遍历图像的每一个像素,并将图像数据 data 中的值赋给 image.NRGBA 对象的 Pix 字段。m.Stride 表示每行像素所占的字节数。这里将红色和绿色通道设置为相同的值 v,蓝色通道设置为 255 (最大值),透明度通道也设置为 255 (不透明)。
-
将图像编码为 PNG 格式:
var buf bytes.Buffer err := png.Encode(&buf, m) if err != nil { panic(err) }png.Encode 函数将 image.NRGBA 对象 m 编码为 PNG 格式,并将编码后的数据存储在 bytes.Buffer 对象 buf 中。
-
将 PNG 数据编码为 Base64 字符串:
enc := base64.StdEncoding.EncodeToString(buf.Bytes())
base64.StdEncoding.EncodeToString 函数将 PNG 格式的图像数据编码为 Base64 字符串。
-
打印 Base64 字符串:
fmt.Println("IMAGE:" + enc)最后,pic.Show 函数将字符串 “IMAGE:” 与 Base64 编码的图像数据拼接在一起,并打印到标准输出。
Go Playground 的特殊处理
Go Playground 会扫描标准输出,查找以 “IMAGE:” 开头的字符串。如果找到这样的字符串,它会将后面的 Base64 字符串解码为图像数据,并在网页上显示图像。
示例代码分析
让我们回顾一下 Exercise 36 中的示例代码:
package main import "code.google.com/p/go-tour/pic" func Pic(dx, dy int) [][]uint8 { var p = make([]([]uint8), dy) for i := 0; i < len(p); i++ { p[i] = make([]uint8, dx) for j := 0; j < len(p[i]); j++ { p[i][j] = uint8((i + j) / 2) } } return p } func main() { pic.Show(Pic) }
Pic 函数根据传入的宽度和高度,创建一个二维的 uint8 切片,并根据像素的坐标计算像素值。main 函数调用 pic.Show 函数,并将 Pic 函数作为参数传递给它。
总结
pic.Show 函数通过将图像数据编码为 Base64 字符串,并利用 Go Playground 的特殊处理,实现了在网页上显示图像的功能。理解 pic.Show 函数的内部机制,有助于我们更好地理解图像处理和数据编码的原理。
注意事项
- pic.Show 函数只能在 Go Playground 中正常显示图像,因为只有 Go Playground 才会解析 “IMAGE:” 开头的字符串。
- pic.Show 函数默认生成 256×256 像素的图像。
- 可以修改 Pic 函数中的像素计算逻辑,生成不同的图像。


