Golang实例001

2024-08-30 14:58
文章标签 golang 实例 001

本文主要是介绍Golang实例001,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Part 01 目标

    对日志文件进行过滤,拆分;

Part 02 了解与思路

    这是实习公司派给我的任务,在我快要淡出鸟的时候。有几份日志文件,很大的,以G为单位的。里面很多不必要的信息,需要过滤掉。另外,日志是记录在一起的,分析起来的时候,没有条理性,所以,要求根据日期,进行拆分,同一天的日志为一个文件,以日期命名,文件。
编程的重点是正则表达式的使用和文件操作

Part 03 来几行瞧瞧

    因为需要构造正则,所以得知道文件的格式。直接打开文件是个不明智的选择,对我而言,准确点,对我的电脑来说,打开大文件,我的小本本承受不了这样的负担。在Linux环境中,其实一条命令就可以了:

head -n 100 # 显示前面100行;
tail -n 100 # 显示最后100行;
sed -n '5,10p' filename # 这样可以只查看文件的第5行到第10行;

    是不是很方便。但是,我想练习一下golang,所以花了点时间,写了个小程序:


// Auth:Reluxer
// Time:2015-07-23
// Name:preview.go
// Description:闲得慌,写了两个函数
package main
import ("bufio""fmt""io""os"
)
const (ViewLineNum = 40
)
func ReadFileByLine002(filename string) error {f, err := os.OpenFile(filename, os.O_RDONLY, 0660)if err != nil {return err}defer f.Close()sc := bufio.NewScanner(f)for i := 0; i < ViewLineNum; i++ {sc.Scan()fmt.Println(sc.Text())}return sc.Err()
}
func ReadFileByLine001(filename string) error {f, err := os.Open(filename)if err != nil {return err}defer f.Close()buf := bufio.NewReader(f)for i := 0; i < ViewLineNum; i++ {line, err := buf.ReadString('\n')if err != nil {if err == io.EOF {fmt.Println("EOF")break}return err}fmt.Print(line)}return nil
}
func main() {filename := os.Args[1]err := ReadFileByLine002(filename)if err != nil {fmt.Println(err)}
}

    这样之后,我看到了文件的格式是这样的:

10.88.111.101 - - [18/Jul/2015:00:00:02 +0800] "GET /topics/2014-11-18/2269791.html HTTP/1.1" 200 15571
10.88.111.101 - - [18/Jul/2015:00:00:02 +0800] "GET /topics/2014-12-26/2419017.html HTTP/1.1" 200 15176
10.88.111.101 - - [18/Jul/2015:00:00:03 +0800] "GET /topics/2012-10-18/3535945.html HTTP/1.1" 200 15250
10.88.111.101 - - [18/Jul/2015:00:00:03 +0800] "GET /topics/2014-11-10/2245185.html HTTP/1.1" 200 15250

Part 04 过滤吧

    知道了文件格式,接下来就是过滤掉冗杂的日志,主要是.css,.js,.jpg,.png,.gif。这部分的代码如下:

package main
import ("bufio""fmt""io""os""regexp"
)
const (RegRuler = `(\.js HTTP)|(\.jpg HTTP)|(\.png HTTP)|(\.css HTTP)|(\.gif HTTP)`
)
func Process(filesrc, fielestore string) error {var fsrc, fdes *os.Filevar err errorfsrc, err = os.OpenFile(filesrc, os.O_RDONLY, 0660)if err != nil {return err}defer fsrc.Close()if fdes, err = os.OpenFile(fielestore, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0666); err != nil {return err}defer fdes.Close()sc := bufio.NewScanner(fsrc)reg := regexp.MustCompile(RegRuler)for sc.Scan() {line := sc.Text()match := reg.FindString(line)if match == "" {if _, err = io.WriteString(fdes, line+"\n"); err != nil {return err}}}return sc.Err()return nil
}
func main() {filename := os.Args[1]filestorename := "no-css-jpg-png-gif-js_log"if err := Process(filename, filestorename); err != nil {fmt.Println(err)return}fmt.Println("ok")
}

    让我没想到的是,程序的运行时间超级长,长的我看了一集花千骨,它还没结束。
    另外,这一步在Linux底下,也是敲一敲命令就完工的事情。Linux真是强大到没边了。

Part 05 拆分,时间是标准

    光是这样,文件还是太大,而且没有条理,所以啊,要按照时间来拆分文件。那好吧,你说拆分就拆分喽,我没意见。
    代码如下:

package main
import ("bufio""fmt""io""os"// "path""regexp"
)
const (RegRuler = `\[(?P<day>\d{2})/(?P<month>[A-Za-z]{3})/(?P<year>\d{4}):\d{2}:\d{2}:\d{2}\s\+\d{4}\]`
)
func Process(filesrc string) error {var fsrc, fdes *os.Filevar err errorpremonth := ""preday := ""preyear := ""curmonth := ""curday := ""curyear := ""filedes := ""fsrc, err = os.OpenFile(filesrc, os.O_RDONLY, 0660)if err != nil {return err}defer fsrc.Close()sc := bufio.NewScanner(fsrc)reg := regexp.MustCompile(RegRuler)for sc.Scan() {line := sc.Text()// fmt.Println(line)match := reg.FindStringSubmatch(line)// fmt.Println(match)// if match == nil {//  if curday == "" && curmonth == "" && curyear == "" {//      if fdes, err = os.OpenFile("unkonow.txt", os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0666); err != nil {//          return err//      }//  }// fmt.Println("here")// } else {if match != nil {for i, name := range reg.SubexpNames() {//Ignore the whole regexp match and unnamed groupsif i == 0 || name == "" {continue}switch name {case "month":curmonth = match[i]case "day":curday = match[i]case "year":curyear = match[i]}}// fmt.Println(filedes)if curday != preday || curmonth != premonth || curyear != preyear {fdes.Close()filedes = curyear + "-" + curmonth + "-" + curdayif fdes, err = os.OpenFile(filedes, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0666); err != nil {return err}}}_, err = io.WriteString(fdes, line+"\n")if err != nil {// fmt.Println("here")fdes.Close()return err}preyear = curyearpreday = curdaypremonth = curmonth// match :=//fmt.Fprintf(os.Stdout,"%s\n",scanner.Text())}fdes.Close()return sc.Err()return nil
}
func main() {filename := os.Args[1]// fmt.Println(filename)// err := ReadFileByLine001(filename)err := Process(filename)if err != nil {fmt.Println(err)}fmt.Println("ok")
}

Part 06 结束吧

    到这儿呢,基本就差不多了。该结束了。说一下,感慨。

  • 之前的代码少些了几行,然后,我的小本本就承受不住,主板过热,自动关机了,真的是吓死爹了。
  • 编程的过程都是一边百度,一边码代码,也算是小有收获。
  • 学语言就得实践,在实践中学习。

这篇关于Golang实例001的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/1121142

相关文章

golang中slice扩容的具体实现

《golang中slice扩容的具体实现》Go语言中的切片扩容机制是Go运行时的一个关键部分,它确保切片在动态增加元素时能够高效地管理内存,本文主要介绍了golang中slice扩容的具体实现,感兴趣... 目录1. 切片扩容的触发append 函数的实现2. runtime.growslice 函数gro

golang实现动态路由的项目实践

《golang实现动态路由的项目实践》本文主要介绍了golang实现动态路由项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习... 目录一、动态路由1.结构体(数据库的定义)2.预加载preload3.添加关联的方法一、动态路由1

Golang interface{}的具体使用

《Golanginterface{}的具体使用》interface{}是Go中可以表示任意类型的空接口,本文主要介绍了Golanginterface{}的具体使用,具有一定的参考价值,感兴趣的可以了... 目录一、什么是 interface{}?定义形China编程式:二、interface{} 有什么特别的?✅

Java List排序实例代码详解

《JavaList排序实例代码详解》:本文主要介绍JavaList排序的相关资料,Java排序方法包括自然排序、自定义排序、Lambda简化及多条件排序,实现灵活且代码简洁,文中通过代码介绍的... 目录一、自然排序二、自定义排序规则三、使用 Lambda 表达式简化 Comparator四、多条件排序五、

Java实例化对象的​7种方式详解

《Java实例化对象的​7种方式详解》在Java中,实例化对象的方式有多种,具体取决于场景需求和设计模式,本文整理了7种常用的方法,文中的示例代码讲解详细,有需要的可以了解下... 目录1. ​new 关键字(直接构造)​2. ​反射(Reflection)​​3. ​克隆(Clone)​​4. ​反序列化

Python解决雅努斯问题实例方案详解

《Python解决雅努斯问题实例方案详解》:本文主要介绍Python解决雅努斯问题实例方案,雅努斯问题是指AI生成的3D对象在不同视角下出现不一致性的问题,即从不同角度看物体时,物体的形状会出现不... 目录一、雅努斯简介二、雅努斯问题三、示例代码四、解决方案五、完整解决方案一、雅努斯简介雅努斯(Janu

Python开发文字版随机事件游戏的项目实例

《Python开发文字版随机事件游戏的项目实例》随机事件游戏是一种通过生成不可预测的事件来增强游戏体验的类型,在这篇博文中,我们将使用Python开发一款文字版随机事件游戏,通过这个项目,读者不仅能够... 目录项目概述2.1 游戏概念2.2 游戏特色2.3 目标玩家群体技术选择与环境准备3.1 开发环境3

Golang HashMap实现原理解析

《GolangHashMap实现原理解析》HashMap是一种基于哈希表实现的键值对存储结构,它通过哈希函数将键映射到数组的索引位置,支持高效的插入、查找和删除操作,:本文主要介绍GolangH... 目录HashMap是一种基于哈希表实现的键值对存储结构,它通过哈希函数将键映射到数组的索引位置,支持

Vue3组件中getCurrentInstance()获取App实例,但是返回null的解决方案

《Vue3组件中getCurrentInstance()获取App实例,但是返回null的解决方案》:本文主要介绍Vue3组件中getCurrentInstance()获取App实例,但是返回nu... 目录vue3组件中getCurrentInstajavascriptnce()获取App实例,但是返回n

SQL表间关联查询实例详解

《SQL表间关联查询实例详解》本文主要讲解SQL语句中常用的表间关联查询方式,包括:左连接(leftjoin)、右连接(rightjoin)、全连接(fulljoin)、内连接(innerjoin)、... 目录简介样例准备左外连接右外连接全外连接内连接交叉连接自然连接简介本文主要讲解SQL语句中常用的表