0

0

在 GO 中整理您的下载

心靈之曲

心靈之曲

发布时间:2024-10-30 13:18:03

|

869人浏览过

|

来源于dev.to

转载

在 go 中整理您的下载

大家好,离子又来了。

害怕学习人工智能有一天会完成的事情让我感到非常痛苦。但是,如果“解决问题”仍然是对未来人类的要求,为什么不坚持呢?

这次我又带来了一个教程。比第一个没啥用。那么我们来定义一下“问题”的结构,因为我们已经知道一件事:那些没有问题的人,是因为他们看起来不够。对于那些还没有找到它们的人来说,创建它们只是时间问题。

项目结构

程序最简单的结构是:

  • 扫描文件夹(例如下载文件夹或其他目录)
  • 识别相关目录中每个文件的类型
  • 将文件移动到与其类型对应的子文件夹(图像、视频、文档...)

启动项目

创建一个目录并导航到它:

mkdir organizador
cd organizador

创建一个 organizer.go 文件并启动其模块:

touch organizador.go
go mod init organizador.go

你应该有或多或少像这样的东西:

~/organizador
.
├── go.mod
└── organizador.go

第 1 部分:检查目录是否存在

让我们定义源目录dirorigem,我们将在其中执行组织。定义后,我们检查它是否确实存在,否则我们将返回错误:

package main

import (
    "fmt"
    "os"
)

// defina o deretório o qual você quer organizar como variável global
var dirorigem string := "/users/user/downloads" // troque o diretório 

func main() {

    // verificar se o diretório existe, caso contrário, retornar erro
    if _, err := os.stat(dirorigem); os.isnotexist(err) {
        fmt.println("bad dir :( \ndiretório não encontrado: ", dirorigem)
        return
    } else {
        // imprimir mensagem caso o diretório exista
        fmt.println("good dir :) \ndiretório encontrado: ", dirorigem)
    }
}

现在,让我们对上面的代码进行一些考虑:

  • os.stat函数返回两个类型为os.fileinfo和错误err的值。
  1. fileinfo 是一个返回详细文件信息的接口,在本例中我们不需要。因此,为了忽略这个接口,我们使用_,。但是,我们不想忽略错误: if _, err := ...
  2. 我们将错误 err 作为参数传递给 os.isnotexist() 函数,因为如果目录不存在,os.stat() 函数将返回 not null 错误,从而导致os.isnotexist() 返回 true,执行我们的消息:bad dir :(
  3. 如果 os.isnotexist() 返回 false,我们将从 else 条件中打印消息:good dir :)

第 2 部分:回调函数概念太疯狂了,伙计!

你有没有注意到,我们在这里一点一点地享受机械键盘声音的比特和字节。 _查卡查卡繁荣! _

现在我们要创建一个回调函数。这是我从未真正了解过的东西,或者从来没有足够的好奇心来质疑我是否在我之前的 golang 生活中的一些 python 代码中使用过这个概念。

回调函数是一个带有参数传递给另一个函数的函数。

如果您已经熟悉这个概念,那么恭喜您,否则,恭喜您。换句话说,恭喜你!

现在让我们创建一个 filepath.walk 回调函数,它将作为参数传递给另一个函数。

func main() { 
// restante do código
.
.
.

// percorrer e listar os arquivos no diretório dirorigem
    err := filepath.walk(dirorigem, listararquivos)
    if err != nil {
        fmt.println("erro ao percorrer o diretório: ", err)
    }
}
// função que lista os arquivos do diretório
func listararquivos(caminho string, info os.fileinfo, err error) error {
    if err != nil {
        return err
    }

    // ignorar diretórios e exibir apenas arquivos
    if !info.isdir() && !strings.hasprefix(info.name(), ".") {
        fmt.println("arquivo encontrado: ", info.name())
    }
    return nil
}

但是等等,filepath.walk 是如何调用回调函数的?

当您调用 filepath.walk(sourcedir, listfiles) 时,filepath.walk 函数会执行遍历 sourcedir 中的所有文件和子目录的繁重工作。

对于找到的每个文件或目录,它都会使用三个参数调用 listfiles 函数:

  • path:当前文件或目录的完整路径。
  • info:一个 os.fileinfo 对象,包含文件/目录的信息(例如名称、是否是目录、大小等)。
  • err:如果访问此文件或目录时出现问题,则会出现错误。

go 自动理解 listfiles 必须接收这三个参数,因为 filepath.walk 期望一个完全遵循此签名的函数:

// função walk()
func walk(root string, walkfn walkfunc) error

// tipo walkfunc
type walkfunc func(path string, info os.fileinfo, err error) error

请注意,walk 函数返回一个错误!这是相关的!

这就是为什么我们将函数 filepath.walk(dirorigem,listararquivos) 等同于错误:

err := filepath.walk(dirorgiem, listararquivos)

毕竟,因为它返回一个错误,所以它是一个错误xd

实际例子

以下更详细地介绍了每个步骤发生的情况:

//percorrer e listar os arquivos no diretório
err := filepath.walk(dirorigem, listararquivos)

对于 dirorigem 中的每个文件或目录,filepath.walk 将调用 listfiles,就好像它是这样的:

listararquivos("/users/user/downloads/file1.txt", infosobrefile1, nil)
listararquivos("/users/user/downloads/file2.txt", infosobrefile2, nil)
listararquivos("/users/user/downloads/file3.mp3", infosobrefile3, nil)

在此示例中,对于每个调用:

  • path:接收文件或目录的路径。
  • 信息:包含有关此项目的信息(例如名称和类型)。
  • err:用于捕获访问文件/目录时的任何特定错误。

回调函数

listfiles 是 filepath.walk 使用这些值自动调用的回调函数。这样,我们就不需要担心设置path、info和err值; filepath.walk 已经为我们做到了这一点。

phe!

聚彩手机网店系统 免费版
聚彩手机网店系统 免费版

聚彩手机商城系统,是一款专业于手机销售的独立手机网店系统,他拥有众多的手机参数选项,以及傻瓜式的设置选项,让您可以在5分钟内建立起专业而强大的手机销售网站。他拥有多套模版可以实时切换,前台拥有新闻中心、手机中心、配件中心、软件下载、手机报价、发货查询、保修查询、分店查询、产品的对比功能,代理与加盟的申请等功能,他拥有完善的会员中心,会员等级设置等,集成在线支付接口,超强SEO,可以设置所有页面的t

下载

现在在你的终端中进行顽皮的测试:

go run organizador.go

你可以得到结果:

good dir :)
diretório encontrado:  /users/user/downloads
arquivo encontrado:  suic_ sos · 4.32pm · 10-25.jpeg
arquivo encontrado:  suic_ sos.jpeg
arquivo encontrado:  quantum-distortion-cyberpunk-music-234058.mp3
arquivo encontrado:  slow-mode-enabled-cyberpunk-music-230623.mp3
arquivo encontrado:  teaching-lp-20152-seminario-go.pdf

或者:

go run organizador.go
bad dir :(
diretório não encontrado:  /users/user/downloadss

在这种情况下,我只是在“下载”中添加了一个额外的“s”,这样原始目录就会不正确。

现在删除 listfiles 函数,因为我们不会使用它。

开个玩笑,我们只是要更改她的名字并添加其他逻辑。

第 3 部分:组织!=组织

组织得很好,组织得很棒。

在我的精彩观察之后,让我们继续我们真正感兴趣的部分:组织整个事情。

人生的一个讽刺是,在整理文件之前,我们必须先整理一下下一步的想法。

我们的下一个功能基本上需要:

  • 根据 dirorigem 目录中每个文件的扩展名创建子文件夹(如果不存在)。
  • 根据扩展名将文件移动到各自的文件夹中。
  • 但是如果文件已经在组织子文件夹中,则不应再次创建它们。

让我们了解一下这段代码的每个部分的作用:

// função que organiza os arquivos do diretório
func organizararquivos(caminho string, info os.fileinfo, err error) error {
    if err != nil {
        return err
    }

    // ignorar diretórios e exibir apenas arquivos
    if !info.isdir() && !strings.hasprefix(info.name(), ".") {
        fmt.println("arquivo encontrado: ", info.name())

        // obter extensões em letras minúsculas
        extensao := strings.tolower(filepath.ext(info.name()))

        // criar nome das subpastas diretamente em dirorigem
        subpasta := filepath.join(dirorigem, extensao[1:])

        // verificar se a subpasta já existe; caso contrário, criar a pasta
        if _, err := os.stat(subpasta); os.isnotexist(err) {
            err := os.mkdir(subpasta, os.modeperm)
            if err != nil {
                fmt.println("erro ao criar subpasta: ", err)
                return err
            }
        }

        // caminho de destino para mover arquivos
        caminhodestino := filepath.join(subpasta, info.name())

        // verificar se o arquivo já está na subpasta
        if caminhodestino == caminho {
            fmt.printf("o arquivo %s já está na subpasta %s. ignorando...\n", info.name(), subpasta)
        } else {
            // mover o arquivo para a subpasta
            err := os.rename(caminho, caminhodestino)
            if err != nil {
                fmt.println("erro ao mover arquivo: ", err)
                return err
            }
            fmt.printf("arquivo %s movido para %s\n", info.name(), subpasta)
        }
    }
    return nil
}

组织文件函数的结构

为目录结构中找到的每个文件或文件夹调用organizefiles函数。它检查根据扩展名组织每个文件的条件,创建子文件夹并在必要时移动文件。

func organizararquivos(caminho string, info os.fileinfo, err error) error {
    if err != nil {
        return err
    }

这里,organizefiles 函数采用三个参数:

  • path:当前文件或目录的完整路径。
  • info:文件或目录信息,从类型 os.fileinfo.
  • 获取
  • err:尝试访问该项目时可能发生的错误。

首先检查访问文件/目录时是否出现错误。如果是,则立即返回。

过滤文件并忽略隐藏目录

if !info.isdir() && !strings.hasprefix(info.name(), ".") {
        fmt.println("arquivo encontrado: ", info.name())

此片段进行两项检查:

  • !info.isdir():检查该项目是否不是目录(即,它是文件)。
  • !strings.hasprefix(info.name(), "."):检查文件名是否不以“.”开头,忽略基于 unix 的系统上的隐藏文件。

如果两个条件都满足,则文件将显示为 fmt.println。

识别文件扩展名并创建子文件夹名称

extensao := strings.tolower(filepath.ext(info.name()))
subpasta := filepath.join(dirorigem, extensao[1:])

这里:

  • strings.tolower(filepath.ext(info.name())):提取文件扩展名(例如.txt)并将其转换为小写以确保一致性。
  • subfolder := filepath.join(sourcedir, extension[1:]):创建文件将移动到的子文件夹的完整路径。扩展名[1:]去掉扩展名中的起始点(.),形成子文件夹的名称,如txt。

如果子文件夹尚不存在,则创建它

if _, err := os.stat(subpasta); os.isnotexist(err) {
            err := os.mkdir(subpasta, os.modeperm)
            if err != nil {
                fmt.println("erro ao criar subpasta: ", err)
                return err
            }
        }

这里,函数:

  • 使用 os.stat 检查子文件夹是否已存在。
  • 如果子文件夹不存在 (os.isnotexist(err)),则使用 os.mkdir(subfolder, os.modeperm) 创建它。
  • os.modeperm 设置新文件夹的默认权限。如果创建文件夹时出错,则会显示并返回。

设置文件目标路径

caminhodestino := filepath.join(subpasta, info.name())

此时,destinationpath 代表文件将被移动的最终路径。它是使用 filepath.join 构造的,用于将子文件夹路径连接到文件名。

检查文件是否已位于目标文件夹中

if caminhoDestino == caminho {
            fmt.Printf("O arquivo %s já está na subpasta %s. Ignorando...\n", info.Name(), subpasta)
        } else {
            // Mover o arquivo para a subpasta
            err := os.Rename(caminho, caminhoDestino)
            if err != nil {
                fmt.Println("Erro ao mover arquivo: ", err)
                return err
            }
            fmt.Printf("Arquivo %s movido para %s\n", info.Name(), subpasta)
        }
    }
    return nil
}
  • 此代码片段将目标路径与当前文件路径进行比较。如果它们相同,则意味着该文件已位于正确的子文件夹中,因此会被忽略并显示一条消息 (fmt.printf)。
  • 否则,os.rename(path, destinationpath) 会将文件移动到子文件夹。如果移动过程中出现错误,则返回。

最终总结

功能:

  1. 滚动目录,检查每个项目。
  2. 忽略隐藏的目录和文件。
  3. 确定文件扩展名以及目标子文件夹。
  4. 创建子文件夹(如果尚不存在)。
  5. 将文件移至子文件夹,除非该文件已存在。

使用 filepath.walk(dirorigem, organizefiles) 将此函数传递给目录中的每个文件,使它们全部自动组织。

这段代码非常适合作为文件组织函数,因为它在单个函数中处理创建和移动逻辑 - 一种高效且有组织的结构形式。

回购:https://github.com/ionnss/organizador


***地球上的又一天,
离子

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

182

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

229

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

343

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

210

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

396

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

240

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

194

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

458

2025.06.17

C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

8

2026.01.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.4万人学习

Django 教程
Django 教程

共28课时 | 3.7万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号