今天整理 Pocket 中待看的文章,看到这篇《Creating ASCII art in functional Swift》,讲解如何用 Swift 将图片转成 ASCII 字符。具体原理文中讲解的很详细,不再赘述,但是标题中的 in functional Swift 让我很感兴趣,想知道 functional 到底体现在哪里,于是下载 swift-ascii-art 源码一探究竟。

Pixel

图片是由各个像素点组成的,在代码中像素通过 Pixel 这个 struct 实现。每个像素分配了4个字节,这4个字节 (2^8 = 256) 分别用来存储 RBGA 的值。

createPixelMatrix

可以通过 createPixelMatrix 这个静态方法创建一个 width * height 像素矩阵:

static func createPixelMatrix(width: Int, _ height: Int) -> [[Pixel]] {
       return map(0..<height) {="" row="" in<br="">           map(0..<width) {="" col="" in<br="">
               let offset = (width * row + col) * Pixel.bytesPerPixel
               return Pixel(offset)
           }
       }
   }
和传统方法中使用 for 循环来创建多维数组有所不同的是,这里是通过 map 函数实现的。在 Swift 2.0 中, map 函数已经被干掉了,只能作为方法调用。

intensityFromPixelPointer

intensityFromPixelPointer 方法计算并返回像素点的亮度值,代码如下:

  1. func intensityFromPixelPointer(pointer: PixelPointer) -> Double {
  2.     let
  3.     red   = pointer[offset + 0],
  4.     green = pointer[offset + 1],
  5.     blue  = pointer[offset + 2]
  6.     return Pixel.calculateIntensity(red, green, blue)
  7. }
  8. private static func calculateIntensity(r: UInt8, _ g: UInt8, _ b: UInt8) -> Double {
  9.     let
  10.     redWeight   = 0.229,
  11.     greenWeight = 0.587,
  12.     blueWeight  = 0.114,
  13.     weightedMax = 255.0 * redWeight   +
  14.                   255.0 * greenWeight +
  15.                   255.0 * blueWeight,
  16.     weightedSum = Double(r) * redWeight   +
  17.                   Double(g) * greenWeight +
  18.                   Double(b) * blueWeight
  19.     return weightedSum / weightedMax
  20. }

calculateIntensity 方法基于 Y’UV 编码获取某个像素的亮度 (intensity) :

  1. Y = 0.299 R + 0.587 G + 0.114 B

YUV 是一种颜色编码方法,Y 表示亮度, UV 用来表示色差, U 和 V 是构成彩色的两个分量。它的优点是可以利用人眼的特性来降低数字彩色图像所需要的存储容量。我们通过这个公式获取到的 Y 就是亮度的值。

Offset

Pixel 中其实只存了一个值: offset 。 Pixel.createPixelMatrix 创建出来的矩阵是这样的:

  1. [[0, 4, 8, ...], ...]

并没有像想象中那样存储了每个像素相关数据,而更像是一个转换工具,计算 PixelPointer 的灰度值。

AsciiArtist

AsciiArtist 里封装了一些生成字符画的方法。

createAsciiArt

createAsciiArt 方法就是创建字符画:

  1. func createAsciiArt() -> String {
  2.     let
  3.     // 加载图片数据,获取指针对象
  4.     dataProvider = CGImageGetDataProvider(image.CGImage),
  5.     pixelData    = CGDataProviderCopyData(dataProvider),
  6.     pixelPointer = CFDataGetBytePtr(pixelData),
  7.     // 将图片转成亮度值矩阵
  8.     intensities  = intensityMatrixFromPixelPointer(pixelPointer),
  9.     // 将亮度值转成对应字符
  10.     symbolMatrix = symbolMatrixFromIntensityMatrix(intensities)
  11.     return join("\n", symbolMatrix)
  12. }

其中 CFDataGetBytePtr 函数返回了图像的字节数组指针,数组里每个元素都是一个字节,即 0~255 的整数。每4个字节组成了一个 Pixel ,分别对应着 RGBA 的值。

intensityMatrixFromPixelPointer

intensityMatrixFromPixelPointer 这个方法是通过 PixelPointer 生成对应的亮度值矩阵:

  1. private func intensityMatrixFromPixelPointer(pointer: PixelPointer) -> [[Double]]
  2. {
  3.     let
  4.     width  = Int(image.size.width),
  5.     height = Int(image.size.height),
  6.     matrix = Pixel.createPixelMatrix(width, height)
  7.     return matrix.map { pixelRow in
  8.         pixelRow.map { pixel in
  9.             pixel.intensityFromPixelPointer(pointer)
  10.         }
  11.     }
  12. }

首先通过 Pixel.createPixelMatrix 方法创建了一个空的二维数组,用来存放数值。然后用两个 map 嵌套遍历里面的所有元素,将像素 (pixel) 转换成亮度 (intensity) 的值。

symbolMatrixFromIntensityMatrix

symbolMatrixFromIntensityMatrix 函数将亮度值数组转换成字符画数组:

  1. private func symbolMatrixFromIntensityMatrix(matrix: [[Double]]) -> [String]
  2. {
  3.     return matrix.map { intensityRow in
  4.         intensityRow.reduce("") {
  5.             $0 + self.symbolFromIntensity($1)
  6.         }
  7.     }
  8. }

map + reduce 成功实现了字符串的累加,每次 reduce 都是通过 symbolFromIntensity 方法获取到亮度值对应的字符。 symbolFromIntensity 方法如下:

  1. private func symbolFromIntensity(intensity: Double) -> String
  2. {
  3.     assert(0.0 <= intensity && intensity <= 1.0)
  4.     let
  5.     factor = palette.symbols.count - 1,
  6.     value  = round(intensity * Double(factor)),
  7.     index  = Int(value)
  8.     return palette.symbols[index]
  9. }

传入 intensity ,在确保了值的范围是 0 ~ 1 之后,通过 AsciiPalette 将它转换成对应的字符,输出 sumbol 。

AsciiPalette

AsciiPalette 是用来将数值转换成字符的工具,像是一个字符画里的调色板一样,根据不同的颜色生成字符。

loadSymbols

loadSymbols 加载了所有的字符:

  1. private func loadSymbols() -> [String]
  2. {
  3.     return symbolsSortedByIntensityForAsciiCodes(32...126) // from ' ' to '~'
  4. }

可以看到,我们选用的字符范围是 32 ~ 126 的字符,接下来就是通过 symbolsSortedByIntensityForAsciiCodes 方法将这些字符按照亮度进行排序。比如 & 符号肯定代表着比 . 暗的区域,那么它是如何比较的呢?请看排序方法。

symbolsSortedByIntensityForAsciiCodes

symbolsSortedByIntensityForAsciiCodes 方法实现了字符串的生成和排序:

  1. private func symbolsSortedByIntensityForAsciiCodes(codes: Range) -> [String]
  2. {
  3.     let
  4.     // 通过 Ascii 码生成字符数组备用
  5.     symbols          = codes.map { self.symbolFromAsciiCode($0) },
  6.     // 将字符绘制出来,把字符数组转换成图片数组,用于比较亮度
  7.     symbolImages     = symbols.map { UIImage.imageOfSymbol($0, self.font) },
  8.     // 将图片数组转换成亮度值数组,亮度值的表现形式是图片中白色像素的个数
  9.     whitePixelCounts = symbolImages.map { self.countWhitePixelsInImage($0) },
  10.     // 将字符数组通过亮度值就行排序
  11.     sortedSymbols    = sortByIntensity(symbols, whitePixelCounts)
  12.     return sortedSymbols
  13. }

其中, sortByIntensity 这个排序方法如下:

  1. private func sortByIntensity(symbols: [String], _ whitePixelCounts: [Int]) -> [String]
  2. {
  3.     let
  4.     // 用字典建立 白色像素数目 和 字符 之间的关系
  5.     mappings      = NSDictionary(objects: symbols, forKeys: whitePixelCounts),
  6.     // 白色像素数目数组去重
  7.     uniqueCounts  = Set(whitePixelCounts),
  8.     // 白色像素数目数组排序
  9.     sortedCounts  = sorted(uniqueCounts),
  10.     // 利用前面的字典映射,将排序后的白色像素数目转换成对应的字符,从而输出有序数组
  11.     sortedSymbols = sortedCounts.map { mappings[$0] as! String }
  12.     return sortedSymbols
  13. }

小结

简单了过了一下项目,可以隐约感觉到一些函数式风格的气息,主要体现在一下几个方面:

  • map reduce 等函数的应用恰到好处,自如处理数组的转换和拼接。

  • 通过 input 和 output 进行数据处理,比如 sortByIntensity 方法和 symbolFromIntensity 方法。

  • 很少有状态和属性,更多的是直接的函数转换,函数逻辑不依赖外部变量,只依赖于传入的参数

代码感觉简单轻快。通过这个简单的小例子,验证了前面在 函数式的特性 中学习到的东西。

感觉很赞!

参考文献:

用函数式的 Swift 实现图片转字符画的功能的更多相关文章

  1. Swift 实现图片转字符画的功能

    本文介绍一个IOS APP , 将图片转换成ASCII字符画,使用Swift语言编写. 举个例子,我们使用著名的蕾娜照片作为原图片 经APP转换后形成的字符画,我们打印出来,效果如下: 放大她的脸部可 ...

  2. Python 图片转字符画

    Python 图片转字符画 一.课程介绍 1. 课程来源 原创 2. 内容简介 本课程讲述怎样使用 Python 将图片转为字符画 3. 前置课程 Python编程语言 Linux 基础入门(新版) ...

  3. [笔记] Python 图片转字符画

    一.介绍 用Python 代码完成图片转字符画 二.python 环境 Python 3.6.6 pillow 5.1.0  Python 图像处理库, 需要另外安装 三.原理 gray = 0.21 ...

  4. Python 图片转字符画 学习笔记

    Python 图片转字符画 学习笔记 标签(空格分隔): Python 声明:此文章和所有代码是学习笔记,非原创,原文教程地址:https://www.shiyanlou.com/courses/37 ...

  5. 图片转字符画 【学习ing】

    1.创建ascii.py 2. 下面是 ascii.py 的完整代码: from PIL import Image import argparse #命令行输入参数处理 parser = argpar ...

  6. python3----练习题(图片转字符画)

    import argparse from PIL import Image def parse_param(): parser = argparse.ArgumentParser() # 命令行输入参 ...

  7. 图片处理拓展篇 : 图片转字符画(ascii)

    首先要明确思路, 图片是由像素组成的, 不同的像素有不同的颜色(rgb), 那么既然我们要转化为字符画, 最直接的办法就是利用字符串来替代像素, 也就是用不同的字符串来代表不同的像素. 另外图片一般来 ...

  8. python学习---50行代码实现图片转字符画2

    from PIL import Image codeLib = '''@B%8&WM#*oahkbdpqwmZO0QLCJUYXzcvunxrjft/\|()1{}[]?-_+~<> ...

  9. python学习---50行代码实现图片转字符画1

    转自:https://blog.csdn.net/mm1030533738/article/details/78447714 项目链接: https://www.shiyanlou.com/cours ...

随机推荐

  1. 如何实现带照片缩略图的Listview

    ackage com.demo; import java.util.ArrayList; import android.app.Activity; import android.content.Con ...

  2. oracle(天猫处方药留言sql)

    " ?> .dtd" > <sqlMap namespace="TmallTcMessage"> <typeAlias alias ...

  3. OkHttp 上手

    OkHttp 上手 优点 快.节省带宽. 支持 HTTP/2 和 SPDY. HTTP/2 和 SPDY 允许对同一个主机的所有请求,使用一个 socket. 如果不支持 SPDY 的话,可以用连接池 ...

  4. java socket报文通信(二)报文的封装

    昨天我们谈了怎么建立socket通信的服务端和客户端,今天我们就来谈一谈怎么封装报文. 什么是报文这里我就不在阐述了,不清楚的朋友可以自己去查资料.我们今天要谈的报文主要友以下几个部分组成: 3位同步 ...

  5. 服务器安装VMware ESXI5.5

    一.VMware ESXI5.5 vSphere5.5 VIMSetup下载http://blog.sina.com.cn/s/blog_61c07ac50101gy64.html(1)VMware安 ...

  6. IIS报500.0错误

    IIS安全里面配置:Everyone.IUSR.IIS_IUSRS 参考地址:http://blog.chinaunix.net/uid-21375345-id-3213631.html

  7. JS常用验证方法

    1.验证必须为数字(可有小数点) if(isNaN(value))execCommand('undo') 说明:1.isNaN()方法用于验证value值是否为非法数字,返回值true或者false. ...

  8. 【Nutch2.2.1基础教程之6】Nutch2.2.1抓取流程

    一.抓取流程概述 1.nutch抓取流程 当使用crawl命令进行抓取任务时,其基本流程步骤如下: (1)InjectorJob 开始第一个迭代 (2)GeneratorJob (3)FetcherJ ...

  9. 设置ubuntu Android sdk环境变量

    cd /etc/ sudo gedit profile 在后面把tools和platform-tools的路径追加进去即可 PATH=$PATH:/home/android_sdk/tools 然后再 ...

  10. kafka相关应用

    一.kafka官网地址 http://kafka.apache.org 下载地址: http://kafka.apache.org/downloads.html 二.版本 0.9.0.1 is the ...