Saltar al contenido principal

📁 Entrada/Salida de Archivos

📖 ¿Qué es la Entrada/Salida de Archivos?

La Entrada/Salida de Archivos (File I/O) es la operación de leer y escribir archivos. ¡Es esencial para almacenar datos, gestionar configuraciones, registrar logs y más!

💡 Lectura de Archivos

Lectura Completa

import java.io.File

fun main() {
// Leer todo el archivo como cadena
val content = File("data.txt").readText()
println(content)

// Leer el archivo como lista de líneas
val lines = File("data.txt").readLines()
for (line in lines) {
println(line)
}
}

Lectura Segura

import java.io.File

fun readFileSafe(filename: String): String? {
return try {
File(filename).readText()
} catch (e: Exception) {
println("Error al leer el archivo: ${e.message}")
null
}
}

fun main() {
val content = readFileSafe("data.txt")
if (content != null) {
println("Contenido: $content")
} else {
println("No se puede leer el archivo")
}
}

Procesamiento Línea por Línea

import java.io.File

fun main() {
// Eficiente en memoria (bueno para archivos grandes)
File("data.txt").forEachLine { line ->
println(line)
}

// useLines - se cierra automáticamente
val lineCount = File("data.txt").useLines { lines ->
lines.count()
}
println("Total de líneas: $lineCount")
}

✏️ Escritura de Archivos

Sobrescribir

import java.io.File

fun main() {
// Escribir texto
File("output.txt").writeText("Hola\n¡Esto es Kotlin!")

// Escribir línea por línea
val lines = listOf("Primera línea", "Segunda línea", "Tercera línea")
File("output.txt").writeText(lines.joinToString("\n"))
}

Añadir Contenido

import java.io.File

fun main() {
val file = File("log.txt")

// Añadir contenido
file.appendText("Log 1\n")
file.appendText("Log 2\n")
file.appendText("Log 3\n")
}

Escritura Segura

import java.io.File

fun writeFileSafe(filename: String, content: String): Boolean {
return try {
File(filename).writeText(content)
true
} catch (e: Exception) {
println("Error al escribir el archivo: ${e.message}")
false
}
}

fun main() {
if (writeFileSafe("data.txt", "Hello Kotlin!")) {
println("Archivo guardado correctamente")
}
}

🎯 Ejemplos Prácticos

Bloc de Notas Simple

import java.io.File

class SimpleNotebook(private val filename: String) {
private val file = File(filename)

fun write(content: String) {
file.writeText(content)
println("Guardado completado")
}

fun append(content: String) {
file.appendText("$content\n")
println("Añadido completado")
}

fun read(): String {
return if (file.exists()) {
file.readText()
} else {
"El archivo no existe"
}
}

fun clear() {
file.writeText("")
println("Contenido eliminado")
}
}

fun main() {
val notebook = SimpleNotebook("mynotes.txt")

notebook.write("Primera nota")
notebook.append("Segunda nota")
notebook.append("Tercera nota")

println("\n=== Contenido del Bloc ===")
println(notebook.read())
}

Logger (Registrador)

import java.io.File
import java.time.LocalDateTime
import java.time.format.DateTimeFormatter

class Logger(private val filename: String) {
private val file = File(filename)
private val formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")

fun log(message: String) {
val timestamp = LocalDateTime.now().format(formatter)
file.appendText("[$timestamp] $message\n")
}

fun info(message: String) = log("INFO: $message")
fun error(message: String) = log("ERROR: $message")
fun warn(message: String) = log("WARN: $message")

fun readLogs(): String {
return if (file.exists()) {
file.readText()
} else {
"Sin logs"
}
}
}

fun main() {
val logger = Logger("app.log")

logger.info("Aplicación iniciada")
logger.warn("Memoria insuficiente")
logger.error("Fallo de conexión")

println(logger.readLogs())
}

Procesamiento de Archivos CSV

import java.io.File

data class Person(val name: String, val age: Int, val city: String)

object CsvHandler {
fun writeCsv(filename: String, people: List<Person>) {
val csv = StringBuilder()
csv.append("Nombre,Edad,Ciudad\n")

for (person in people) {
csv.append("${person.name},${person.age},${person.city}\n")
}

File(filename).writeText(csv.toString())
}

fun readCsv(filename: String): List<Person> {
val file = File(filename)
if (!file.exists()) return emptyList()

val lines = file.readLines()
if (lines.isEmpty()) return emptyList()

return lines.drop(1).map { line ->
val parts = line.split(",")
Person(parts[0], parts[1].toInt(), parts[2])
}
}
}

fun main() {
val people = listOf(
Person("Hong Gildong", 25, "Seúl"),
Person("Kim Chulsoo", 30, "Busan"),
Person("Lee Younghee", 28, "Daegu")
)

// Guardar CSV
CsvHandler.writeCsv("people.csv", people)
println("CSV guardado correctamente")

// Leer CSV
val loaded = CsvHandler.readCsv("people.csv")
println("\n=== Contenido del CSV ===")
loaded.forEach { println(it) }
}

Gestión de Archivos de Configuración

import java.io.File

class Config(private val filename: String) {
private val settings = mutableMapOf<String, String>()

init {
load()
}

private fun load() {
val file = File(filename)
if (file.exists()) {
file.forEachLine { line ->
val parts = line.split("=")
if (parts.size == 2) {
settings[parts[0].trim()] = parts[1].trim()
}
}
}
}

fun save() {
val content = settings.entries.joinToString("\n") {
"${it.key}=${it.value}"
}
File(filename).writeText(content)
}

fun set(key: String, value: String) {
settings[key] = value
}

fun get(key: String): String? {
return settings[key]
}

fun getAll(): Map<String, String> {
return settings.toMap()
}
}

fun main() {
val config = Config("app.config")

// Guardar configuración
config.set("host", "localhost")
config.set("port", "8080")
config.set("timeout", "3000")
config.save()

// Leer configuración
println("Host: ${config.get("host")}")
println("Port: ${config.get("port")}")

println("\n=== Configuración Completa ===")
config.getAll().forEach { (key, value) ->
println("$key = $value")
}
}

📂 Gestión de Archivos/Directorios

Información de Archivos

import java.io.File

fun main() {
val file = File("data.txt")

println("Existe: ${file.exists()}")
println("Es archivo: ${file.isFile}")
println("Es directorio: ${file.isDirectory}")
println("Tamaño: ${file.length()} bytes")
println("Ruta absoluta: ${file.absolutePath}")
println("Nombre: ${file.name}")
println("Carpeta superior: ${file.parent}")
}

Operaciones con Directorios

import java.io.File

fun main() {
// Crear directorio
val dir = File("mydata")
dir.mkdir()

// Lista de archivos
dir.listFiles()?.forEach { file ->
println("${file.name} (${if (file.isDirectory) "carpeta" else "archivo"})")
}

// Buscar archivos en subdirectorios
dir.walk().forEach { file ->
println(file.absolutePath)
}
}

Copiar/Eliminar Archivos

import java.io.File

fun main() {
val source = File("source.txt")
val dest = File("dest.txt")

// Copiar
source.copyTo(dest, overwrite = true)

// Mover
source.renameTo(File("moved.txt"))

// Eliminar
dest.delete()

// Eliminar directorio completo
File("mydata").deleteRecursively()
}

🤔 Preguntas Frecuentes

P1. ¿Qué pasa si el archivo no existe?

R: ¡El manejo de excepciones es esencial!

fun readFileSafe(filename: String): String {
val file = File(filename)

if (!file.exists()) {
return "Archivo no encontrado"
}

return try {
file.readText()
} catch (e: Exception) {
"Error de lectura: ${e.message}"
}
}

P2. ¿Qué hacer con archivos grandes?

R: ¡Procese línea por línea!

// ❌ Riesgo de falta de memoria
val content = File("huge.txt").readText()

// ✅ Procesamiento línea por línea
File("huge.txt").forEachLine { line ->
processLine(line)
}

P3. ¿Qué separador de ruta usar?

R: ¡Use File.separator!

// ❌ Diferente en cada SO
val path = "data/files/text.txt"

// ✅ Funciona en cualquier SO
val file = File("data", "files").resolve("text.txt")
// O
val path2 = listOf("data", "files", "text.txt").joinToString(File.separator)

🎬 Conclusión

¡Almacene y gestione datos con entrada/salida de archivos!

Resumen:
✅ Leer archivos con readText()
✅ Escribir archivos con writeText()
✅ Procesar línea por línea con forEachLine()
✅ Verificar existencia con exists()
✅ El manejo de excepciones es esencial

Siguiente paso: ¡Aprenda sobre coincidencia de patrones en Expresiones Regulares!