从小案例学习Go语言-将Excel各部分内容分发到不同的电子邮箱

关键点:

  • Go语言读取Excel
  • Go语言正则表达式
  • Go语言发送电子邮件

案例场景

今天公司行政部小妹妹跑来问,有什么办法可以把工资条自动发送到每个员工的企业邮箱里?公司每个员工的工资条以Excel的形式放在同一个文档里,之前用OA发送,复制粘贴,操作相当简单,但是公司要求改用电子邮件发送工资条后,给行政部的同事增加了较大的工作量,而且每个月都需要做一次,这很浪费时间,于是爽快的答应帮忙解决。

情况梳理

公司工资条大概这个样子的



为了方便,行政部门会把所有人的工资条按顺序排列在同一个Excel文件里的同一个sheet里。全貌会是以下这个样子


Paste_Image.png

行政部的妹妹希望能够自动的把每个人自己的工资条发送到各自的邮箱里,所以至少得有个地方填写邮箱号吧,于是我在每个人的工资条上增加了一行,标记每个人的邮箱,于是文档成了这样

Paste_Image.png

好了,Excel格式确定下来,对于程序员来说,他就只是个二维数组了。接下来,就开始写代码吧。

用Go读取Excel

Go语言自己有个CSV库,不过在这个场景里,还是用github.com/tealeg/xlsx库来处理xlsx文件更合适。
创建go文件,将工资表与go文件放在同一个目录,本文假设讲工资表Excel 命名为 list.xlsx

package main

import (
    "bufio"
    "fmt"

    "github.com/tealeg/xlsx"
    "log"
)

func main() {
    excelFileName := "./list.xlsx"
    xlFile, err := xlsx.OpenFile(excelFileName)
    if err != nil {
        log.Fatalln("err:", err.Error())
    }
}

xlsx打开Excel文件成功,会返回一个xlsx.File对象,这个对象里除有一些基础的文件操作方法,还包含一个Sheets的对象,这个对象是Excel文件中Sheet的map集合,可以通过遍历获得所有Sheet。
Sheet中包含一个名叫Rows的对象,这个对象是Sheet中所有行的集合。
Rows中包含一个名叫Cells的对象,这个对象是行中所有格子的集合。

所以,一个xlsx.File对象不考虑其包含的方法的话就相当于一个三维数组。

我们只需要做三次嵌套的循环就可以获得其中的所有单元格数据,像这样

func main() {
    excelFileName := "./list.xlsx"
    xlFile, err := xlsx.OpenFile(excelFileName)
    if err != nil {
        log.Fatalln("err:", err.Error())
    }
    for _, sheet := range xlFile.Sheets {
        for _, row := range sheet.Rows {
            for _, cell := range row.Cells {
                fmt.Printf("%s\n", cell.Value)
            }
        }
    }
}

接下来,要进入关键点,把数据读取出来后,要分隔没个人的工资条,从之前的图片上可以看出,当表格中出现一次电子邮件内容的单元格的时候,就是新的一个人的工资条了。所以,需要通过正则表达式判断有没有读取到电子邮件的单元格,如果读取到,就要用新的存储空间保存工资条的内容。

用正则表达式找到含有Email地址的单元格

正则表达式判断很简单,创建一个函数,读取整行的数据,如果其中出现了电子邮件,就返回真,以及电子邮件字符串(这个地方可以不用穿反isEmail这个参数,只需要判断email是不是零值就可以了)

func isEmailRow(r []string) (isEmail bool, email string) {
    reg := regexp.MustCompile(`^[a-zA-Z_0-9.-]{1,64}@([a-zA-Z0-9-]{1,200}.){1,5}[a-zA-Z]{1,6}$`)
    for _, v := range r {
        if reg.MatchString(v) {
            return true, v
        }
    }
    return false, ""
}

为了后面操作方便,我用getCellValues函数将行的Cells直接读取成字符串数组,并且过滤了空格和换行。

func getCellValues(r *xlsx.Row) (cells []string) {
    for _, cell := range r.Cells {
        txt := strings.Replace(strings.Replace(cell.Value, "\n", "", -1), " ", "", -1)
        cells = append(cells, txt)
    }
    return
}

我用了一个map来统一存放不同的人的工资条数据,并且用电子邮件作为键值,然后将数据组装成一个HTML的表格行代码(因为需要发送HTML格式的电子邮件才能以表格的形式展现)。于是,main里的循环代码就变成了这样

for _, sheet := range xlFile.Sheets {
        curMail := ""
        for _, row := range sheet.Rows {
            cells := getCellValues(row)
            //如果行包含电子邮件,创建一个新字典项
            if isEmail, emailStr := isEmailRow(cells); isEmail {
                curMail = emailStr
            } 
            sendList[curMail] += fmt.Sprintf("<tr><td>%s</td></tr>", strings.Join(cells, "</td><td>"))
            
        }
    }

用Go语言发送电子邮件(SMTP)

Go语言发送电子邮件很简单,用标准包 net/smtp就足够了。

先封装一个发送邮件的函数,用官方的例子改造一下。

func sendToMail(user, password, host, to, subject, body, mailtype string) error {
   auth := smtp.PlainAuth("", user, password, strings.Split(host, ":")[0])
   msg := []byte("To: " + to + "\r\nFrom: " + user + "\r\nSubject: " + subject + "\r\n" + "Content-Type: text/" + mailtype + "; charset=UTF-8" + "\r\n\r\n" + body)
   sendto := strings.Split(to, ";")
   err := smtp.SendMail(host, auth, user, sendto, msg)
   return err
}

再创建一个函数,遍历所有内容并调用发送邮件函数发送出去

func sendMail(sendList map[string]string) {

    fmt.Printf("共需要发送%d封邮件\n", len(sendList))
    index := 1
    for mail, content := range sendList {
        fmt.Printf("发送第%d封", index)
        if err := sendToMail("xxx@mybigcompany.com",
            "thisismypassword",
            "smtp.mybigcompany.com:25",
            mail,
            "工资条",
            fmt.Sprintf("<table border='2'>%s</table>", content),
            "html"); err != nil {
            fmt.Printf(" ... 发送错误(X) %s %s \n", mail, err.Error())

        } else {
            fmt.Printf(" ... 发送成功(V) %s \n", mail)
        }
        index++
        fmt.Printf("<table border='2'>%s</table> \n", content)
    }
}

最后,将sendMail放在main函数中,for迭代读取出所有数据之后,就完成了。
行政的同事使用的是Windows,使用终端程序往往会让他们摸不着头脑,完全不知道发生什么事情,然而我也不可能花太多时间为这样的小程序开发界面,所以即便在终端运行,也尽量提供友善的用户体验,代码中关键的信息都尽量输出友好提示。程序结束后,做一个终端输入等待,让用户看到运行的结果。

    fmt.Print("按下回车结束")
    bufio.NewReader(os.Stdin).ReadLine()

完整代码

package main

import (
    "bufio"
    "fmt"
    "net/smtp"
    "os"
    "regexp"

    "strings"

    "log"

    "github.com/tealeg/xlsx"
)

func main() {
    excelFileName := "./list.xlsx"
    xlFile, err := xlsx.OpenFile(excelFileName)
    if err != nil {
        log.Fatalln("err:", err.Error())
    }

    sendList := make(map[string]string)

    for _, sheet := range xlFile.Sheets {
        curMail := ""
        for _, row := range sheet.Rows {
            cells := getCellValues(row)
            //如果行包含电子邮件,创建一个新字典项
            if isEmail, emailStr := isEmailRow(cells); isEmail {
                curMail = emailStr
            } else {
                count := 0
                for _, c := range cells {
                    if len(c) > 0 {
                        count++
                    }
                }

                if count > 1 {
                    sendList[curMail] += fmt.Sprintf("<tr><td>%s</td></tr>", strings.Join(cells, "</td><td>"))
                } else {
                    sendList[curMail] += fmt.Sprintf("<tr><td colspan='%d'>%s</td></tr>", len(cells), strings.Join(cells, ""))
                }

            }

        }
    }

    sendMail(sendList)
    fmt.Print("按下回车结束")
    bufio.NewReader(os.Stdin).ReadLine()

}

func getCellValues(r *xlsx.Row) (cells []string) {
    for _, cell := range r.Cells {
        txt := strings.Replace(strings.Replace(cell.Value, "\n", "", -1), " ", "", -1)
        cells = append(cells, txt)
    }
    return
}

func isEmailRow(r []string) (isEmail bool, email string) {
    reg := regexp.MustCompile(`^[a-zA-Z_0-9.-]{1,64}@([a-zA-Z0-9-]{1,200}.){1,5}[a-zA-Z]{1,6}$`)
    for _, v := range r {
        if reg.MatchString(v) {
            return true, v
        }
    }
    return false, ""
}

func sendMail(sendList map[string]string) {

    fmt.Printf("共需要发送%d封邮件\n", len(sendList))
    index := 1
    for mail, content := range sendList {
        fmt.Printf("发送第%d封", index)
        if err := sendToMail("xxx@mybigcompany.com",
            "thesismypassword",
            "smtp.mybigcompany.com:25",
            mail,
            "工资条",
            fmt.Sprintf("<table border='2'>%s</table>", content),
            "html"); err != nil {
            fmt.Printf(" ... 发送错误(X) %s %s \n", mail, err.Error())

        } else {
            fmt.Printf(" ... 发送成功(V) %s \n", mail)
        }
        index++
        //fmt.Printf("<table border='2'>%s</table> \n", content)
    }

}

func sendToMail(user, password, host, to, subject, body, mailtype string) error {
    auth := smtp.PlainAuth("", user, password, strings.Split(host, ":")[0])
    msg := []byte("To: " + to + "\r\nFrom: " + user + "\r\nSubject: " + subject + "\r\n" + "Content-Type: text/" + mailtype + "; charset=UTF-8" + "\r\n\r\n" + body)
    sendto := strings.Split(to, ";")
    err := smtp.SendMail(host, auth, user, sendto, msg)
    return err
}

Go语言交叉编译,运行在不同的操作系统

我用的Mac 64位,需要编译一个Windows 32位的可执行程序,一句搞定

CGO_ENABLED=0 GOOS=windows GOARCH=386 go build

GOOS设置目标系统,可以是 windows, linux,darwin
GOARCH设置目标系统是32位还是64位,分别对应 386和amd64
CGO_ENABLED设置是否需要使用CGO,本例子不需要,设置为0,如果需要使用CGO编译,设置为1

OK,任务完成,只要编辑一份如文中第三张图那样格式的文档,保存为list.xlsx,与编译好的可执行文件放在同一目录,双击执行,文档中的内容就会根据电子邮件单元格作为分割点分别发送到该电子邮箱里。

知识点总结

  • 使用github.com/tealeg/xlsx包读取xlsx文件
  • 使用regexp包实现正则表达式判断
  • 使用net/smtp包发送电子邮件
  • 使用交叉编译命令生成不同系统上的可执行文件

欢迎大家简书或我的个人博客与我交流

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 157,298评论 4 360
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,701评论 1 290
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 107,078评论 0 237
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,687评论 0 202
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,018评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,410评论 1 211
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,729评论 2 310
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,412评论 0 194
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,124评论 1 239
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,379评论 2 242
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,903评论 1 257
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,268评论 2 251
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,894评论 3 233
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,014评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,770评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,435评论 2 269
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,312评论 2 260

推荐阅读更多精彩内容

  • Simple Excel Export 简单的Excel导出推荐http://www.cnblogs.com/hy...
    地狱咆哮Zzzzz阅读 15,486评论 0 6
  • 使用首先需要了解他的工作原理 1.POI结构与常用类 (1)创建Workbook和Sheet (2)创建单元格 (...
    长城ol阅读 8,257评论 2 25
  • 创建新工程 打开Eclipse新建一个工程 点下一步 输入名称 点完成 新建一个目录用来存在第三方库文件 选择目录...
    长新阅读 2,137评论 3 1
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,087评论 18 139
  • 小小动过,结果是把自己伤了,留了个丑陋的疤,那个曾经的男友现在跟另一半过得幸福的很。 吵架的时候没人能动脑子,但是...
    印大钞阅读 704评论 2 1