go学习之文件操作与命令行参数

文章目录

    • 一、文件操作
      • 1.基本介绍
      • 2.常用文件操作函数和方法
      • 3.关于文件操作应用实例
      • 4.写文件操作应用实例(创建文件并写入文件)
        • 1)基本介绍
        • 2)基本应用实例-方式一
      • 5.判断文件是否存在
      • 6.统计英文、数字、空格和其他字符数量
    • 二、命令行参数
      • 1.使用os.Args对参数进行解析
      • 2.flag包解析命令行参数

一、文件操作

1.基本介绍

文件对于我们并不陌生,文件是数据源(保存数据的地方)一种,比如大家经常使用的word文件,txt文件,excel文件…都是文件。文件最主要的作用就是保存数据,它既可以保存一张图片,也可以保存视频声音

文件在程序中是以流的形式来操作的

在这里插入图片描述

  • :数据在数据源(文件)和程序(内存)之间经历的路径

  • 输入流:数据从数据源(文件)到程序(内存)的路径

  • 输出流:数据从程序(内存)到数据源(文件)的路径

os.File结构体封装所有文件相关操作

2.常用文件操作函数和方法

1)打开一个文件进行操作

os.Open(name string)(*File,error)

2)关闭一个文件:

File.Close()

3)其他的函数和方法在案例提

案例演示

package main
import (
	"fmt"
	"os"
)
func main(){
	//打开文件
	//概念说明: file的叫法
	//1. file 叫file对象
	//1. file 叫file指针
	//1. file 叫file文件句柄
	file , err := os.Open("D:/test/test01/test.txt")
	if err !=nil {
		fmt.Println("open file err=",err)
	}
	//输出下文件,看看文件是什么,看出file/就是一个指针
	fmt.Printf("file=%v",file) //file=&{0xc0420705a0}

   //关闭文件
  err = file.Close()
  if err != nil {
	fmt.Println("close file err=",err)
  }

}

3.关于文件操作应用实例

1)读取文件的内容并显示在终端(带缓冲区的方式),使用os.Open,file.Close, bufio.NewReader(),reader.ReadString函数和方法

package main
import (
	"fmt"
	"os"
	"io"
	"bufio"
)
func main(){
	//打开文件
	//概念说明: file的叫法
	//1. file 叫file对象
	//1. file 叫file指针
	//1. file 叫file文件句柄
	file , err := os.Open("D:/test/test01/test.txt")
	if err !=nil {
		fmt.Println("open file err=",err)
	}

	//当函数退出时,要及时的关闭file
	/*
    const (
		defaultBufSize  =4096 //默认缓冲区4096个字节
	)
	*/
	defer file.Close()  //要及时关闭file句柄,否则会有内存泄露

	//创建一个*Read ,带缓冲
	reader :=bufio.NewReader(file)
	//循环的读取文件的内容
	for {
		str,err := reader.ReadString('\n') //读到一个换行符就结束一次
		if err == io.EOF { //io.EOF表示文件的末尾
			break
		}
		//输出内容
		fmt.Printf(str)
	}
	fmt.Println("文件读取结束")
}
   

2)读取文件的内容并显示在终端(使用ioutil一次将整个文件读入到内存中),这种方式适用于文件不大的情况。相关方法和函数(ioutil.ReadFile)

package main
import (
	"fmt"
	"io/ioutil"
	
)
func main(){
	//使用ioutil.ReaderFile一次性将文件读取到位
	file := "D:/test/test01/test.txt"
	content, err := ioutil.ReadFile(file)
	if err !=nil {
		fmt.Println("read file err=%v",err)
	}

	//把读取到的内容显示到终端
    fmt.Printf("%v",string(content)) //[]byte
    //因为我们没有显示的open文件,因此也不需要显示的close文件
	//因为。文件Open和Close背封装到ReadFile函数内部
	
}

注意:这种只适合文件不太大的情况使用

4.写文件操作应用实例(创建文件并写入文件)

1)基本介绍
func OpenFile(name string,flag int,perm FileMode)(file *File,err error)

说明:os.OpenFile是一个更一般性的文件打开函数,它会使用指定的选项(如O_RDONLY)、指定的模式(如0666等)打开指定名称的文件。如果操作成功,返回的文件对象可用于i/o如果出错,错误底层类型是*PathError.

第二个参数:文件打开模式(可以组合),第三个参数:权限控制(linux)
在这里插入图片描述

2)基本应用实例-方式一

(1)创建一个新文件,写入内容 5句"hello Gardon"

package main
import (
	"fmt"
	"os"
	"bufio"
)
func main(){
    //创建一个新文件,写入内容 5句"hello Gardon"
	//1.打开一个文件 "D:/test/test01/test.txt"
	filePath := "D:/test/test01/abc.txt"
	file, err := os.OpenFile(filePath,os.O_WRONLY | os.O_CREATE,0666)
	if err != nil {
		fmt.Printf("open file err=%v",err)
		return
	}
	//及时关闭file句柄,防止内存泄露
	defer file.Close()

	//准备写入6句话
	str := "hello Gardon\r\n"  // \r\n表示换行
	//写入时,使用带缓存的*writer
	writer := bufio.NewWriter(file)
	for i := 0; i < 5; i++ {
		writer.WriteString(str)
	}

	//因为writer是带缓存的,因此在调用WriterString方法时,其实内存是先写入缓存的
	//所以需要调用Flush()方法,将缓存的数据
	//真正写入到文件中,否则文件中会没有数据
	writer.Flush()

(2)打开一个存在的文件中,将原来的内容覆盖成新的内容10句 “你好,爸爸"

package main
import (
	"fmt"
	"os"
	"bufio"
)
func main(){
   //打开一个存在的文件中,将原来的内容覆盖成新的内容10句 “你好,爸爸"
	//1.打开一个文件 "D:/test/test01/test.txt"
	filePath := "D:/test/test01/abc.txt"
	file, err := os.OpenFile(filePath,os.O_WRONLY | os.O_TRUNC,0666)
	if err != nil {
		fmt.Printf("open file err=%v",err)
		return
	}
	//及时关闭file句柄,防止内存泄露
	defer file.Close()

	//准备写入10句话:你好,爸爸
	str := "你好,爸爸!\r\n"  // \n 表示换行
	//写入时,使用带缓存的*writer
	writer := bufio.NewWriter(file)
	for i := 0; i < 10; i++ {
		writer.WriteString(str)
	}

	//因为writer是带缓存的,因此在调用WriterString方法时,其实内存是先写入缓存的
	//所以需要调用Flush()方法,将缓存的数据
	//真正写入到文件中,否则文件中会没有数据
	writer.Flush()
}

(3)打开一个存在的文件,在原来的内容追加内容“ABCI ENGLISH!”

package main
import (
	"fmt"
	"os"
	"bufio"
)
func main(){
	//打开一个存在的文件,在原来的内容追加内容“ABCI ENGLISH!”
	//1.打开一个文件 "D:/test/test01/test.txt"
	filePath := "D:/test/test01/abc.txt"
	file, err := os.OpenFile(filePath,os.O_WRONLY | os.O_APPEND,0666)
	if err != nil {
		fmt.Printf("open file err=%v",err)
		return
	}
	//及时关闭file句柄,防止内存泄露
	defer file.Close()

	str := "ABCI ENGLISH!\r\n"  // \n 表示换行
	//写入时,使用带缓存的*writer
	writer := bufio.NewWriter(file)
	for i := 0; i < 10; i++ {
		writer.WriteString(str)
	}

	//因为writer是带缓存的,因此在调用WriterString方法时,其实内存是先写入缓存的
	//所以需要调用Flush()方法,将缓存的数据
	//真正写入到文件中,否则文件中会没有数据
	writer.Flush()
}

(4)打开一个存在的文件,将原来的内容读出显示在终端,并且追加"hello 北京"

package main
import (
	"fmt"
	"os"
	"bufio"
	"io"
)
func main(){
	//打开一个存在的文件,将原来的内容读出显示在终端,并且追加"hello 北京"
	//1.打开一个文件 "D:/test/test01/abc.txt"
	//这是一个既要读又要写的操作
	filePath := "D:/test/test01/abc.txt"
	file, err := os.OpenFile(filePath,os.O_RDWR | os.O_APPEND,0666)
	if err != nil {
		fmt.Printf("open file err=%v",err)
		return
	}
	//及时关闭file句柄,防止内存泄露
	defer file.Close()

	//先读取原来文件的内容,并显示在终端
	reader := bufio.NewReader(file)
	for {
		str,err := reader.ReadString('\n')
		if err == io.EOF { //如果读到文件末尾
            break
		}
		//显示到终端
		fmt.Print(str)
	}
//写到文件中
	str := "hello 北京\r\n"  // \n 表示换行
	//写入时,使用带缓存的*writer
	writer := bufio.NewWriter(file)
	for i := 0; i < 5; i++ {
		writer.WriteString(str)
	}

	//因为writer是带缓存的,因此在调用WriterString方法时,其实内存是先写入缓存的
	//所以需要调用Flush()方法,将缓存的数据
	//真正写入到文件中,否则文件中会没有数据
	writer.Flush()
}

使用os.OpenFile(),bufio.NewWriter(),“Writer的方法WriteString完成上面的任务

3)基本应用实例-方式二

编写一个程序,将一个文件的内容,写入到另一个文件,注意:这两个文件已经存在了

说明:

1)使用ioutil.ReadFile /ioutil.WriteFile 完成文件的任务

package main
import (
	"fmt"
	"io/ioutil"
)
func main(){
	//将D:/test/test01/abc.txt文件的内容导入到D:/test/test01/kkk.txt中

	//1.首先将D:/test/test01/abc.txt内容读取到内存
	
	filePath :="D:/test/test01/abc.txt"
	filePath2 :="D:/test/test01/kkk.txt"
	data, err := ioutil.ReadFile(filePath)
	if err != nil {
		//说明读取文件有错误
		fmt.Printf("read file err=%v\n",err)
		return 
	}
//2.将读取到的内容写入:D:/test/test01/kkk.txt中
	err = ioutil.WriteFile(filePath2,data,0666)
	if err != nil {
		fmt.Printf("write file err=%v\n",err)
	}
	//会覆盖掉写入之后的文件中的所有内容

}

5.判断文件是否存在

golang判断文件或文件夹是否存在的方法是使用os.Stat()函数返回的错误进行判断:

1)如果返回的错误为nil,说明文件或文件夹存在

2)如果返回的错误类型使用so.IsNotExist()判断为true,说明文件或文件夹不存在

3)如果返回的错误为其他类型,则不确定是否存在

//自己写了一个函数
func PathExists(path string)(bool,error){
	_,err :=os.Stat(path)
	if err == nil { //文件或目录存在
		return true,nil
	}
	if os.IsNotExist(err){
		return false,nil
	}
	return false,err
}
              

6.文件编程应用实例

拷贝文件

说明:将一张图片拷贝到另外一个目录下io包

func Copy (dst Writer,src Reader)(written int64,err error)
package main
import (
	"fmt"
	"io"
	"os"
	"bufio"
)

//自己写一个函数,接收两个文件路径 srcFileName dstFileName
func CopyFile(dstFileName string,srcFileName string)(written int64,err error){
	srcFile, err := os.Open(srcFileName)
	if err != nil {
		fmt.Println("open file err=%v\n",err)
	}
	//用完了需要关闭
	defer srcFile.Close()
	//通过srcFile,获取到Reader
	reader := bufio.NewReader(srcFile)

	//打开dstFileName :不能单纯地打开因为你不确定是否存在
	dstFile, err := os.OpenFile(dstFileName,os.O_WRONLY | os.O_CREATE,0666)
    if err != nil {
		fmt.Printf("open file err=%v\n",err)
		return
	}
	
	//通过dstFile,获取到writer
	writer := bufio.NewWriter(dstFile)//用完了需要关闭
	defer dstFile.Close()

	return io.Copy(writer,reader)

}
func main() {
	//将D:/test/dog.jpg拷贝到D:/test/test01/dog1.jpg
	//调用CopyFile完成文件的拷贝
	srcFile := "D:/test/dog.jpg"
	dstFile := "D:/test/test01/dog1.jpg"
	_, err :=CopyFile(dstFile,srcFile)
	if err == nil {
		fmt.Println("拷贝完成")
	} else {
		fmt.Printf("拷贝错误err=%v\n",err)
	}
}

6.统计英文、数字、空格和其他字符数量

说明:统计一个文件中含有的英文、数字-、空格及其它字符数量

package main
import (
	"fmt"
	"os"
	"io"
	"bufio"
)
//定义个结构体,用于保存统计结构1
type CharCount struct {
	CharCount int //记录英文个数
	NumCount int //记录数字的个数
	SpaceCount int //记录空格的个数
	OtherCount int //记录其他字符的个数
}


func main(){
	//思路:打开一个文件。创建一个Reader
	//每读取一行,就去统计该行有多少个 英文、数字、空格和其他字符
	//然后将它们保存到一个结构体当中
fileName := "D:/test/abc.txt"
	file, err := os.Open(fileName)
	if err != nil {
		fmt.Printf("open file err =%v\n",err)
		return
	}
     defer file.Close()
	//定义一个结构体实例
	var count CharCount
	//创建一个Reder
	reader := bufio.NewReader(file)

	//开始循环读取fileName的内容
	for {
		str,err :=reader.ReadString('\n')
		if err == io.EOF {  //读到文件末尾
			break
		}
		//遍历str进行统计
		for _, v := range str {
			
		   switch {
			case v >= 'a' && v <= 'z' :
					fallthrough //穿透处理
			case v >= 'A' && v <= 'A' :
				count.CharCount ++ 	
			case v == ' ' || v =='\t' :
				count.SpaceCount ++	
			case v >= '0' && v <='9' :
				count.NumCount ++ 	
			default :
				count.OtherCount ++				 
			}
		}

	}
	//输出统计的结构
	fmt.Printf("字符的个数=%v,数字的个数=%v,空格的个数=%v,其他字符的个数=%v",count.CharCount,count.NumCount,count.SpaceCount,count.OtherCount)
}

二、命令行参数

有一个需求

我们希望能够获取到命令行输入的各种参数,该如何处理?如下,我们执行一个可执行文件并附带一个参数

D:\myfile\GO\project\src\go_code\exec\count\main> test.exe tom c:/aa/bb/config.init 88

1.使用os.Args对参数进行解析

基本介绍

os.Args是一个string的切片,用来存储所有的命令行参数

应用案例

请编写一段代码,可以获取命令行的各种参数

package main
import (
	"fmt"
	"os"
)

func main() {
	fmt.Println("命令行的参数值",len(os.Args))
	//遍历os.Args切片,就可以得到所有命令行输入的参数值
	for i,v :=range os.Args {
		fmt.Printf("Args[%v]=%v\n",i,v)
	}
}

使用go build去编译一个可执行文件test.exe进行测试

在这里插入图片描述

2.flag包解析命令行参数

flag包用来解析命令行参数

说明:前面的方式是比较原生的方式,对解析参数不是特别的方便,特别是带有指定参数形式的命令行

比如:cmd>main.exe -f C:/aaa.txt -p 200 -u root 这样的命令行,go设计者给我们提供了flag包,,可以方便的解析命令行参数,而且参数顺序可以随意

请编写一段代码,可以获取命令行的各个参数

package main
import (
	"fmt"
	"flag"
)
func main() {

	 //定义几个变量,用于接收命令行的参数值
	 var user string
	 var pwd string
	 var host string
	 var port int
	 //&user 就是接收用户命令行中输入的-u后面的参数
	 //"u",就是-u 指定参数
	 //"",默认值
	 //"用户名,默认为空" 说明
	 flag.StringVar(&user,"u","","用户名,默认为空")
	 flag.StringVar(&pwd,"pwd","","密码,默认为空")
	 flag.StringVar(&host,"h","localhost","主机名,默认为localhost")
     flag.IntVar(&port,"port",3306,"端口号默认为3306")

	 //这里有一个非常重要的操作,转换,必须调用该方法
	 flag.Parse()

	 //输出结果
	 fmt.Printf("user=%v pwd=%v host=%v port=%v",
	user,pwd,host,port)

}

测试结果

在这里插入图片描述

将顺序打乱或者不传参数再次进行测验

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Kubernetes

Kubernetes Docker的安装Docker安装&#xff1a;安装docker依赖环境配置国内docker-ce的yum源&#xff08;这里采用的是阿里云&#xff09;安装docker。插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants 创建一个自…

Sui主网升级至V1.14.2版本

Sui主网现已升级至V1.14.2版本&#xff0c;同时Sui协议升级至31版本。其他升级要点如下所示&#xff1a; #14875: [修复] 为所有权限设置共识度量值。 #14811: [Narwhal] 改进每个权限的共识信息度量的可用性。 完整变更日志&#xff1a;Release mainnet-v1.14.2 MystenL…

考虑极端天气线路脆弱性的配电网分布式电源配置优化模型_IEEE33节点(附带Matlab代码)

随着新能源技术及智能电网的发展&#xff0c;越来越多的分布式电源加入配电网中&#xff0c;不仅改变了配电网结构及供电方式&#xff0c;而且提升了配电网的供电质量。但是在全球气候变暖的背景下&#xff0c;极端天气发生的频率也越来越高&#xff0c;一旦发生必将对配电网系…

HashMap的死循环及数据覆盖问题

目录 一&#xff0c;HashMap 线程不安全的原因 二&#xff0c;HashMap 死循环问题 死循环发生的条件 死循环的具体过程 死循环执行步骤1 死循环执行步骤2 死循环执行步骤3 三&#xff0c;HashMap 数据覆盖问题 数据覆盖执行流程1 数据覆盖执行流程2 数据覆盖执行流…

8、CobaltStrike使用

文章目录 一、实验拓扑图二、实验步骤 一、实验拓扑图 二、实验步骤 1、登录"Kali"操作机&#xff0c;在终端中进入CS文件夹&#xff0c;然后使用命令chmod x teamserver给teamserver文件赋予执行权限&#xff0c;然后查看当前主机的本地IP地址。 2、启动服务端服务…

SAS9.2软件“OLE:对象的类没有在注册数据库中注册“问题的解决. 2023-11-25

操作系统测试平台: Win7 sp1 32bit (6.1.7601.26321 (Win7 RTM)) ; Win 11 64bit(具体版本不详) 其它win平台理论上也可以,可自行测试 1.安装依赖库(必要步骤) 下载地址: Microsoft Visual C 2005 Redistributable 下载 Microsoft Visual C 2008 Redistributable 官方vc库总…

信号类型(通信)——最小频移键控(MSK)

系列文章目录 《信号类型&#xff08;通信&#xff09;——仿真》 《信号类型&#xff08;通信&#xff09;——QAM调制信号》 《信号类型&#xff08;通信&#xff09;——QPSK、OQPSK、IJF_OQPSK调制信号》 目录 前言 一、MSK信号特点 1.1、最小频移 1.2、相位连续 二…

【Vulnhub 靶场】【Coffee Addicts: 1】【简单-中等】【20210520】

1、环境介绍 靶场介绍&#xff1a;https://www.vulnhub.com/entry/coffee-addicts-1,699/ 靶场下载&#xff1a;https://download.vulnhub.com/coffeeaddicts/coffeeaddicts.ova 靶场难度&#xff1a;简单 - 中等 发布日期&#xff1a;2021年5月20日 文件大小&#xff1a;1.3 …

Spring——感谢尚硅谷官方文档

Spring——尚硅谷学习笔记 1 Spring简介&#x1f47e;1.1 Spring概述1.2 Spring Framework1.2.1 Spring Framework特性1.2.2 Spring Framework五大功能模块 2 IOC-IOC容器思想&#x1f47e;IOC容器思想IOC在Spring中的实现 3 IOC-基于XML文件管理Bean&#x1f47e;3.1 准备工作…

拼多多商品详情API接口,详情页接口,宝贝详情页接口,商品属性接口,商品信息查询,商品详细信息接口,获取拼多多已拼数量,实时准确数据调用案例

接入拼多多API接口的操作流程一般包括以下步骤&#xff1a; 了解API接口&#xff1a;首先&#xff0c;你需要了解你要接入的API接口的文档和规范。这些信息通常可以在API提供商的官方文档或开发者门户网站上找到。文档通常会包含API的请求和响应格式、参数、权限等信息。获取A…

【LeetCode刷题】--77.组合

77.组合 class Solution {public List<List<Integer>> combine(int n, int k) {List<List<Integer>> ans new ArrayList<>();if( k < 0 || n < k){return ans;}Deque<Integer> list new ArrayDeque<>();dfs(ans,list,n,k,1)…

C语言——一个数如果恰好等于它的因子之和,这个数就称为“完全数”。

一个数如果恰好等于它的因子之和,这个数就称为“完全数”。例如,6的因子是 1、2、3,而6123。因此6是一个完全数。编程找出 1000 之内的所有完全数。 #include <stdio.h> int main() {int i, j, sum;for (i 1; i < 1000; i) {sum 0; //这一步很重要&#xff0c;每…

图片伪装,将RAR文件隐藏到图片里

下载链接 效果图&#xff1a; 代码&#xff1a; ECHO OFF TITLE PtoR MODE con COLS55 LINES25 color 0A:main cls echo.当前时间&#xff1a;%date% %time% echo.欢迎使用图片伪装&#xff0c;本脚本可以将RAR文件隐藏到图片里. echo.set /p "imagefile①请拖入图像文件…

hugegraph-server安装部署(docker)

1、安装docker不说了&#xff0c;可以直接看我文章一键安装docker https://blog.csdn.net/qq_41060647/article/details/131568289?spm1001.2014.3001.5502 2、一个docker-compose文件解决。 如果不使用mysql&#xff0c;可以将docker-compose.yml文件中的mysql配置修改为其他…

基于BP神经网络的手写体数字识别matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 2.算法运行软件版本 matlab2022a 3.部分核心程序 filename dir(images\*.bmp); %图像文件格式 load BP.matfilename dir(test\*.bmp); …

Ubuntu16.04.4系统本地提权实验

目录 1.介绍&#xff1a; 2.实验&#xff1a; 3.总结&#xff1a; 1.介绍&#xff1a; 1.1&#xff1a;eBPF简介&#xff1a;eBPF(extendedBerkeleyPacketFilter)是内核源自于BPF的一套包过滤机制&#xff0c;BPF可以理解成用户与内核之间的一条通道&#xff0c;有非常强大的…

token认证机制,基于JWT的Token认证机制实现,安全性的问题

文章目录 token认证机制几种常用的认证机制HTTP Basic AuthOAuthCookie AuthToken AuthToken Auth的优点 基于JWT的Token认证机制实现JWT的组成认证过程登录请求认证 对Token认证的五点认识JWT的JAVA实现 基于JWT的Token认证的安全问题确保验证过程的安全性如何防范XSS Attacks…

(一)基于高尔夫优化算法GOA求解无人机三维路径规划研究(MATLAB)

一、无人机模型简介&#xff1a; 单个无人机三维路径规划问题及其建模_IT猿手的博客-CSDN博客 参考文献&#xff1a; [1]胡观凯,钟建华,李永正,黎万洪.基于IPSO-GA算法的无人机三维路径规划[J].现代电子技术,2023,46(07):115-120 二、高尔夫优化算法GOA简介 高尔夫优化算法…

uniapp ios 授权弹窗 uniapp弹出框怎么实现

新版本的信息弹窗组件 可以弹出很多条信息&#xff0c;并单独控制消失时间、点击消失。 用循环来生成很多个弹窗&#xff0c;用this.$refs来传值&#xff0c;并添加数组。 1.布局 2.js 具体流程。需要一个弹窗&#xff0c;基本信息传入组件&#xff0c;处理后添加入数组&am…

什么是半监督学习

1 概述 1.1 定义 半监督学习&#xff08;Semi-Supervised Learning&#xff09;是机器学习中的一个重要分支&#xff0c;它介于监督学习和无监督学习之间。半监督学习利用少量标注数据和大量未标注数据共同训练模型&#xff0c;旨在充分挖掘未标注数据中潜在的信息和模式&…