Introduction
Dans ce chapitre, nous allons voir comment vous pouvez efficacement lire et écrire sur des fichiers à l'aide du langage de programmation go.
Il existe de multiples façons pour lire et écrire dans un fichier. Dans les exemples qui suivent je vais vous montrer les techniques que j'utilise selon les différents cas d'utilisation que je juge le plus simple.
Je vais manipuler un fichier nommé test.txt avec le contenu suivant :
je suis un fichier de test
Lire seulement un fichier
Dans le cas ou vous ne faites que lire un fichier, le mieux reste d'utiliser la fonction ReadFile() de la bibliothèque io/ioutil. Cette fonction retourne un type byte (bit), il faut donc penser à caster (convertir le type) le résultat obtenu en string
package main
import (
"fmt"
"io/ioutil"
)
func main() {
data, err := ioutil.ReadFile("test.txt") // lire le fichier text.txt
if err != nil {
fmt.Println(err)
}
fmt.Println(string(data)) // conversion de byte en string
}
Résultat :
je suis un fichier de test
Dans le cas ou le fichier n'existe pas, vous aurez l'erreur suivante :
Le fichier spécifié est introuvable.
Écrire sur un fichier
il y a deux manières pour écrire dans un fichier. Soit vous décidez d'écraser un fichier après écriture, soit d'écrire à la suite du contenu du fichier.
Pour écrire dans un fichier on va utiliser la bibliothèque os.
package main
import (
"fmt"
"io/ioutil"
"os"
)
func main() {
file, err := os.OpenFile("test.txt", os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0600)
defer file.Close() // on ferme automatiquement à la fin de notre programme
if err != nil {
panic(err)
}
_, err = file.WriteString("test\n") // écrire dans le fichier
if err != nil {
panic(err)
}
_, err = file.WriteString("i love test\n")
if err != nil {
panic(err)
}
data, err := ioutil.ReadFile("test.txt") // lire le fichier
if err != nil {
fmt.Println(err)
}
fmt.Print(string(data))
}
Information
J'utilise le "\n" pour faire un saut de la ligne dans mon fichier.
Résultat :
je suis un fichier de test
test
i love test
Je reviens ici sur ces lignes de code :
file, err := os.OpenFile("test.txt", os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0600)
defer file.Close() // on ferme automatiquement le fichier après l'avoir manipulé
La fonction os.OpenFile(), propose vraiment beaucoup d'options, que je vous explique ci-dessous :
- Premier paramètre : correspond au nom du fichier à ouvrir
-
Deuxième paramètre : ici ce sont des options spécialement dédiées au fichier que
vous allez manipuler :
- os.O_CREATE : Permet de créer le fichier si il n'existe pas.
- os.O_WRONLY : Permet de rendre le fichier (dans votre programme) accessible en écriture seulement.
- os.O_APPEND : Permet de ne pas écraser le fichier quand vous écrivez dessus (supprimez cette option si vous souhaitez écraser le fichier).
- Troisième paramètre : les droits d'accès de votre fichier (Plus d'information sur les permissions ici ici)
Il est absolument important de fermer le fichier à la fin de votre programme. D'où l'utilisation de la fonction close(), j'ai rajouté le mot clé defer, ce mot-clé permet d'exécuter la ligne de code en question jusqu'à la fin d'exécution d'une fonction.
Bonus
On se retrouve avec beaucoup de lignes de code répétables, il serait temps d'utiliser les super pouvoirs des fonctions pour mieux structurer notre code :
package main
import (
"fmt"
"io/ioutil"
"os"
)
func check(e error) {
if e != nil {
panic(e)
}
}
func write(text string, file *os.File) {
if _, err := file.WriteString(text); err != nil {
panic(err)
}
}
func read(filename string) string {
data, err := ioutil.ReadFile(filename)
check(err)
return string(data)
}
func main() {
file, err := os.OpenFile("test.txt", os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0600)
defer file.Close()
check(err)
write("Test\n", file)
data := read(file.Name())
fmt.Print(data)
}
Résultat :
je suis un fichier de test
test
i love test
Test
Exercice.
Voilà maintenant vous savez comment lire et écrire dans un fichier. Si vous souhaitez aller encore plus loin, vous pouvez reprendre le tp du morpions pour rajouter un système qui vous permet de sauvegarder la partie en cours (tour du joueur et le damier).
Espace commentaire
Écrire un commentaire
Rejoignez la discussion
Vous devez être connecté pour poster un message.
30 commentaires
Oui, appelle
file.Sync()après tes écritures. C'est plus lent mais plus sûr en cas de crash brutal.Je peux forcer l'écriture sur disque sans attendre le flush du cache OS ?
Ta RAM. Si le fichier fait 10Go, tu vas faire un OOM. Utilise des streams pour les gros fichiers.
C'est quoi la limite de taille pour
os.ReadFile?Top. Si tu veux pousser, regarde le format JSON pour sauvegarder l'état complexe de ta grille plutôt que du texte brut.
J'ai réussi à implémenter la sauvegarde du morpion, merci pour le tuto !
file.Write([]byte). ÉviteWriteStringqui attend du texte encodé.Comment je fais pour écrire des données binaires et pas juste du texte ?
Vérifie ton
Working Directoryau lancement. Utiliseos.Getwd()pour voir où le binaire cherche réellement.Mon programme Go ne trouve pas le fichier alors qu'il est bien dans le dossier. Une idée ?
Carrément. Ça évite de répéter le
if err != nilpartout, surtout quand tu as beaucoup d'opérations IO.Le
check(e error)dans le bonus, c'est pas mal pour aérer le code.Ça dépend de ton
umasksystème. Le0600est filtré par le masque de ton shell.J'ai un souci avec
os.O_CREATE, il me crée le fichier mais avec des droits bizarres.ioutil.ReadFilecharge tout. Utilisebufio.Scannerpour lire ton fichier par chunks.Comment je peux lire un gros fichier ligne par ligne sans tout charger en mémoire ?
En prod, oui. C'est pour l'exemple. Remplace par un retour d'erreur classique et gère ton flow proprement.
Le
panicc'est pas un peu violent pour gérer des erreurs de fichier ?N'oublie pas le
dans ta chaîne de caractères quand tu fais tonfile.WriteString.J'essaie de concaténer des logs mais ça me fout tout sur la même ligne.
Bien vu, depuis Go 1.16, utilise le package
osdirectement pouros.ReadFile. C'est devenu le standard.J'ai une erreur sur
ioutil, il me dit que c'est déprécié sur les versions récentes de Go.Parce que si ton programme plante avec un
panicavant la fin, le fichier reste ouvert. Ne jamais oublier le defer pour éviter les fuites de descripteurs.C'est quoi l'intérêt du
defer file.Close()? Pourquoi pas juste le fermer à la fin ?Exact. Et ajoute
os.O_TRUNCsi tu veux être sûr de repartir de zéro à chaque ouverture.