内容优化

内容优化

Products

当前位置:首页 > 内容优化 >

Go语言实现动态图片生成方案

96SEO 2025-07-04 15:53 5


要将PHP动态图片生成功能迁移到Go语言站点(如AnqiCMS),以下是完整的解决方案:

1. Go语言实现代码

`go package main

import ( "crypto/sha1" "fmt" "image" "image/draw" "image/jpeg" "image/png" "io/ioutil" "log" "net/http" "os" "path/filepath" "strings" "sync"

"github.com/golang/freetype"
"github.com/golang/freetype/truetype"
"golang.org/x/image/font"
"golang.org/x/image/webp"

)

var ( fontData *truetype.Font bgImages []string allowedExts = []string{".jpg", ".jpeg", ".png", ".webp"} imageCache = sync.Map{} fontLoadError error )

func init() { // 初始化加载字体和背景图片 loadResources() }

func loadResources() { // 加载字体 fontBytes, err := ioutil.ReadFile("../images/font.ttf") if err != nil { fontLoadError = fmt.Errorf("加载字体失败: %v", err) return } fontData, err = freetype.ParseFont(fontBytes) if err != nil { fontLoadError = fmt.Errorf("解析字体失败: %v", err) return }

// 加载背景图片
dir := "../images/"
files, err := ioutil.ReadDir(dir)
if err != nil {
	fontLoadError = fmt.Errorf("读取图片目录失败: %v", err)
	return
}

for _, file := range files {
	if file.IsDir() || strings.HasPrefix(file.Name(), ".") {
		continue
	}
	ext := strings.ToLower(filepath.Ext(file.Name()))
	for _, allowed := range allowedExts {
		if ext == allowed && file.Name() != "font.ttf" {
			bgImages = append(bgImages, filepath.Join(dir, file.Name()))
			break
		}
	}
}

if len(bgImages) == 0 {
	fontLoadError = fmt.Errorf("没有找到可用的背景图片")
}

}

func generateImage(text string) (image.Image, error) { if fontLoadError != nil { return nil, fontLoadError }

// 选择背景图片
h := sha1.New()
h.Write([]byte(text))
hash := h.Sum(nil)
idx := int(hash[0]) % len(bgImages)
bgPath := bgImages[idx]

// 检查缓存
if img, ok := imageCache.Load(bgPath); ok {
	// 复制一份避免修改缓存
	bgImg := img.(image.Image)
	rgba := image.NewRGBA(bgImg.Bounds())
	draw.Draw(rgba, rgba.Bounds(), bgImg, image.Point{}, draw.Src)
	return rgba, nil
}

// 加载图片
file, err := os.Open(bgPath)
if err != nil {
	return nil, fmt.Errorf("打开图片失败: %v", err)
}
defer file.Close()

var bgImg image.Image
ext := strings.ToLower(filepath.Ext(bgPath))
switch ext {
case ".png":
	bgImg, err = png.Decode(file)
case ".webp":
	bgImg, err = webp.Decode(file)
default: // jpg/jpeg
	bgImg, err = jpeg.Decode(file)
}

if err != nil {
	return nil, fmt.Errorf("解码图片失败: %v", err)
}

// 缓存图片
rgba := image.NewRGBA(bgImg.Bounds())
draw.Draw(rgba, rgba.Bounds(), bgImg, image.Point{}, draw.Src)
imageCache.Store(bgPath, rgba)

return rgba, nil

}

func addTextToImage(img image.Image, text string) (image.Image, error) { // 创建绘图上下文 c := freetype.NewContext() c.SetDPI(72) c.SetFont(fontData) c.SetFontSize(36) c.SetClip(img.Bounds()) c.SetDst(img.(draw.Image)) c.SetSrc(image.White) c.SetHinting(font.HintingFull)

// 计算文字位置
pt := freetype.Pt(
	(img.Bounds().Dx()-len(text)*20)/2, // 简单估算宽度
	100,
)

_, err := c.DrawString(text, pt)
if err != nil {
	return nil, fmt.Errorf("绘制文字失败: %v", err)
}

return img, nil

}

func imageHandler(w http.ResponseWriter, r *http.Request) { // 获取参数 text := r.URL.Query().Get("text") if text == "" { text = "默认文字" }

// 生成图片
img, err := generateImage(text)
if err != nil {
	http.Error(w, err.Error(), http.StatusInternalServerError)
	return
}

// 添加文字
img, err = addTextToImage(img, text)
if err != nil {
	http.Error(w, err.Error(), http.StatusInternalServerError)
	return
}

// 设置响应头
w.Header().Set("Content-Type", "image/jpeg")
w.Header().Set("Cache-Control", "public, max-age=31536000")

// 输出图片
err = jpeg.Encode(w, img, &jpeg.Options{Quality: 90})
if err != nil {
	http.Error(w, "图片编码失败", http.StatusInternalServerError)
}

}

func main() { http.HandleFunc("/gen_img", imageHandler) log.Fatal(http.ListenAndServe(":8080", nil)) } `

2. 适用于AnqiCMS的配置

对于AnqiCMS,你需要:

  1. 将上面的Go代码编译为可执行文件
  2. 在AnqiCMS中配置路由规则

AnqiCMS路由配置

在AnqiCMS的配置文件中添加以下路由规则:

yaml routes: - pattern: "/images/*path" handler: "ImageHandler"

然后实现对应的ImageHandler,或者直接使用反向代理到上面的Go服务。

3. Nginx配置

对于Go站点,Nginx配置可以简化为:

`nginx location /images/ { # 代理到Go服务 proxypass http://localhost:8080/genimg?text=$1; proxysetheader Host $host;

# 缓存控制
expires max;
add_header Cache-Control "public";

# 错误处理
proxy_intercept_errors on;
error_page 404 = @img_rewrite;

}

location @imgrewrite { rewrite ^/images/(.+)$ /genimg?text=$1 last; } `

4. 部署步骤

  1. 将上面的Go代码保存为img_gen.go
  2. 安装依赖:go get github.com/golang/freetype golang.org/x/image
  3. 编译:go build img_gen.go
  4. 运行:./img_gen
  5. 配置Nginx反向代理到Go服务

5. 功能优化建议

  1. 缓存优化:Go版本已经实现了内存缓存,可以进一步添加磁盘缓存
  2. 字体大小自适应:根据文字长度自动调整字体大小
  3. 多行文字支持:处理长文本自动换行
  4. 颜色自定义:通过URL参数指定文字颜色
  5. 图片格式选择:支持输出PNG、WebP等格式

6. 访问方式

  • 直接访问:http://yoursite.com/gen_img?text=示例文字
  • 通过伪静态访问:http://yoursite.com/images/示例文字.jpg

这个Go实现方案比PHP版本更高效,且更适合集成到AnqiCMS等Go语言开发的站点中。


标签:

提交需求或反馈

Demand feedback