golang图片处理库image基本操作

 更新时间:2022年07月26日 15:00:56   作者:alwaysrun  
这篇文章主要介绍了golang图片处理库image简介,主要包括图片的基本读取与保存及图片的修改,本文通过通过实例代码给大家介绍的非常详细,需要的朋友可以参考下

基本操作

图片的基本读取与保存。

读取

图片读取和文件读取类似,需要先获取流:

  • 注册图片的解码器(如:jpg则import _ "image/jpeg", png则import _ "image/png"
  • 通过os.open打开文件获取流;
  • 通过image.Decode解码流,获取图片;
import _ "image/jpeg"
func readPic() image.Image {
    f, err := os.Open("C:\\hatAndSunglass.jpg")
    if err != nil {
        panic(err)
    }
    defer f.Close()

    img, fmtName, err := image.Decode(f)
    if err != nil {
        panic(err)
    }
    fmt.Printf("Name: %v, Bounds: %+v, Color: %+v", fmtName, img.Bounds(), img.ColorModel())

    return img
}

解码后返回的第一个参数为Image接口:

type Image interface {
  ColorModel() color.Model // 返回图片的颜色模型
  Bounds() Rectangle       // 返回图片外框
  At(x, y int) color.Color // 返回(x,y)像素点的颜色
}

新建

新建一个图片非常简单,只需image.NewRGBA即可创建一个透明背景的图片了

img := image.NewRGBA(image.Rect(0, 0, 300, 300))

保存

保存图片也很简单,需要编码后,写入文件流即可:

  • 注册图片的解码器
  • 通过os.create创建文件;通
  • png.Encode编码图片并写入文件;
func savePic(img *image.RGBA) {
    f, err := os.Create("C:\\tmp.jpg")
    if err != nil {
        panic(err)
    }
    defer f.Close()
    b := bufio.NewWriter(f)
    err = jpeg.Encode(b, img, nil)
    if err != nil {
        panic(err)
    }
    b.Flush()
}

图片修改

很多操作都需要用到绘制图片:

func Draw(dst Image, r image.Rectangle, src image.Image, sp image.Point, op Op)
func DrawMask(dst Image, r image.Rectangle, src image.Image, sp image.Point, mask image.Image, mp image.Point, op Op)​

主要参数说明:

  • dst:绘图的背景图
  • r:背景图的绘图区域
  • src:要绘制的图
  • sp:要绘制图src的开始点
  • op:组合方式

DrawMask多了一个遮罩蒙层参数,Draw为其一种特殊形式(遮罩相关参数为nil)。

转换

读取的jpg图像不是RGBA格式的(为YCbCr格式);在操作前需要先转换格式:

  • 创建一个大小相同的RGBA图像;
  • 把jpg画到新建的图像上去;
func jpg2RGBA(img image.Image) *image.RGBA {
    tmp := image.NewRGBA(img.Bounds())

    draw.Draw(tmp, img.Bounds(), img, img.Bounds().Min, draw.Src)
    return tmp
}

裁剪

通过subImage方法可方便地裁剪图片(需要为RGBA格式的)

func subImg() {
    pic := readPic()
    fmt.Printf("Type: %T\n", pic)
    img := jpg2RCBA(pic)

    sub := img.SubImage(image.Rect(0, 0, pic.Bounds().Dx(), pic.Bounds().Dy()/2))
    savePic(sub.(*image.RGBA))
}

缩放

图片缩放分为保持比例与不保持比例的缩放;保持比例时,要确定新图片的位置(是否居中),以及如何填充空白处。
为了缩放,需要引入新的库golang.org/x/image/draw

在保持比例缩放时,需要先计算缩放后的图片大小:

  • 分别计算宽、高的缩放比例,以小者为准;
  • 若是居中(否则靠左上)需要计算填充大小,然后据此计算位置;
func calcResizedRect(width int, src image.Rectangle, height int, centerAlign bool) image.Rectangle {
    var dst image.Rectangle
    if width*src.Dy() < height*src.Dx() { // width/src.width < height/src.height
        ratio := float64(width) / float64(src.Dx())

        tH := int(float64(src.Dy()) * ratio)
        pad := 0
        if centerAlign {
            pad = (height - tH) / 2
        }
        dst = image.Rect(0, pad, width, pad+tH)
    } else {
        ratio := float64(height) / float64(src.Dy())
        tW := int(float64(src.Dx()) * ratio)
        pad := 0
        if centerAlign {
            pad = (width - tW) / 2
        }
        dst = image.Rect(pad, 0, pad+tW, height)
    }

    return dst
}

有了缩放后的大小后,即可通过双线性插值bilinear的方式进行图片的缩放

  • img为要缩放的图片width、height为缩放后的大小
  • keepRatio为是否保持比例缩放
  • fill为填充的颜色(R、G、B都为fill)
  • centerAlign:保持比例缩放时,图片是否居中存放
import (
    "image"
    "image/color"

    "golang.org/x/image/draw"
)

func resizePic(img image.Image, width int, height int, keepRatio bool, fill int, centerAlign bool) image.Image {
    outImg := image.NewRGBA(image.Rect(0, 0, width, height))
    if !keepRatio {
        draw.BiLinear.Scale(outImg, outImg.Bounds(), img, img.Bounds(), draw.Over, nil)
        return outImg
    }

    if fill != 0 {
        fillColor := color.RGBA{R: uint8(fill), G: uint8(fill), B: uint8(fill), A: 255}
        draw.Draw(outImg, outImg.Bounds(), &image.Uniform{C: fillColor}, image.Point{}, draw.Src)
    }
    dst := calcResizedRect(width, img.Bounds(), height, centerAlign)
    draw.ApproxBiLinear.Scale(outImg, dst.Bounds(), img, img.Bounds(), draw.Over, nil)
    return outImg
}

到此这篇关于golang图片处理库image简介的文章就介绍到这了,更多相关golang图片处理库image内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 使用Go goroutine实现并发的Clock服务

    使用Go goroutine实现并发的Clock服务

    这篇文章主要为大家详细介绍了如何使用Go goroutine实现并发的Clock服务,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-06-06
  • 用Go语言标准库实现Web服务之创建路由

    用Go语言标准库实现Web服务之创建路由

    在上一节中创建了项目,这篇文章主要介绍如何用Go语言标准库创建路由,文中有详细的代码示例,对大家的学习或工作有一定的帮助,感兴趣的同学可以参考下
    2023-05-05
  • Go语言defer的一些神奇规则示例详解

    Go语言defer的一些神奇规则示例详解

    这篇文章主要为大家介绍了Go语言defer的一些神奇规则示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-11-11
  • Go和Java算法详析之分数到小数

    Go和Java算法详析之分数到小数

    这篇文章主要给大家介绍了关于Go和Java算法详析之分数到小数的相关资料,文中通过实例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2022-08-08
  • golang实现LRU缓存淘汰算法的示例代码

    golang实现LRU缓存淘汰算法的示例代码

    这篇文章主要介绍了golang实现LRU缓存淘汰算法的示例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-12-12
  • Go返回int64类型字段超出javascript Number范围的解决方法

    Go返回int64类型字段超出javascript Number范围的解决方法

    这篇文章主要介绍了Go返回int64类型字段超出javascript Number范围的解决方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-07-07
  • 深入理解Go高级并发模式编写更高效可扩展的应用程序

    深入理解Go高级并发模式编写更高效可扩展的应用程序

    Go对并发提供了强大的原生支持,本文讨论Go的高级并发模式,理解这些并发模式,可以帮助我们编写高效的Go应用程序,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2024-02-02
  • Goland 生成可执行文件的操作

    Goland 生成可执行文件的操作

    这篇文章主要介绍了Goland 生成可执行文件的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • Go语言声明一个多行字符串的变量

    Go语言声明一个多行字符串的变量

    这篇文章主要介绍了Go语言声明一个多行字符串的变量的方法和示例,非常简单实用,有需要的小伙伴可以参考下。
    2015-04-04
  • 用golang实现一个定时器任务队列实例

    用golang实现一个定时器任务队列实例

    golang中提供了2种定时器timer和ticker,分别是一次性定时器和重复任务定时器。这篇文章主要介绍了用golang实现一个定时器任务队列实例,非常具有实用价值,需要的朋友可以参考下
    2018-05-05

最新评论