Ayuda escritura archivo en C#

Iniciado por Kaxperday, 6 Mayo 2015, 09:49 AM

0 Miembros y 2 Visitantes están viendo este tema.

Kaxperday

Hola a todos ustedes, miren estoy usando un archivo en el que guardo los datos de una aplicación todos sus datos (login, información, autentificacion, entradas, configuración...), cada tipo de datos está delimitado por una linea única en el archivo.

Cuando se realiza un cambio sobre algo por pequeño que sea debo sobreescribir el archivo. Antes trabajaba con un archivo para cada cosa, pero no era el camino correcto tener 10 archivos a la vez.

Es por ello que decidí meterlo en uno y poder lineas delimitadoras, que se crean de la siguiente manera, y se generan al llamar al constructor de archivo:

Código (csharp) [Seleccionar]
public Archivo(String directorio)
        {
            this.lectura = new FileStream(directorio + "\\app.dat", FileMode.Open);
            this.escritura = new FileStream(directorio + "\\app.tmp", FileMode.Create);
            vector.Add("ENTRADA_AUTH_SALIDA");//0
            vector.Add("ENTRADA_LOGI_SALIDA");//1
            vector.Add("ENTRADA_CALC_SALIDA");//2
            vector.Add("ENTRADA_HOLA_SALIDA");//3
            vector.Add("ENTRADA_MENS_SALIDA");//4
            vector.Add("ENTRADA_ENTR_SALIDA");//5
            vector.Add("ENTRADA_PARA_SALIDA");//6
        }


Bien, ahora para sobreescribir el archivo llega mi duda. El archivo app.dat tiene los datos actuales, lo que pense era en leer todo su contenido y pasarlo a una cadena. Y sobreescibir esa cadena añadiendo los datos en el campo correspondiente y escribirla en otro archivo de salida.

Pero creo que es preferible utilizar solo un archivo ya puestos.

Es por ello que quiero leer el contenido del archivo, luego modificar la cadena leída y sobreescribirla en el mismo archivo. He probado esto:

Código (csharp) [Seleccionar]
public bool setAutentificacion(String certificado)
        {
            StreamWriter escritor = new StreamWriter(escritura);//intentar escritura sobremismo archivo
            StreamReader lector = new StreamReader(lectura);
            String contenido = lector.ReadToEnd();
            lector.Close();
            if(contenido.Contains(vector[0].ToString())
            {
                contenido.//aqui quiero saber el puntero de la cadena donde esta esa subcadena
            }
}


Podría hacerlo leyendo caracter a caracter hasta los \n o NewLine, pero tardaría mucho para archivos grandes. Además de la forma de hacerlo no me parece adecuada asi, no se si se podría hacer con algún metodo el identificar la subcadena y obtener el indice donde se encuentra para poder cortarla alli e introducir los nuevos datos.

Luego tengo también la duda del archivo, si puedo hacer de un FileStream un StreamReader y un StreamWriter a la vez, para leerlo todo con uno, y luego usar el otro para sobreescribir el mismo, no se si será posible.

Saludos y gracias por estar ahí de veras. Podéis contar conmigo.
Cuando el poder económico parasita al político ningún partido ni dictador podrá liberarnos de él. Se reserva el 99% ese poder.

Eleкtro

#1
Hola

Perdon que te lo diga pero tengo la sensación de que nunca das la información necesaria, o al menos yo nunca soy capaz de entender a la perfección el problema que tienes en las dudas que formulas, espero haber entendido bien lo que pretendes hacer esta vez.

En realidad no necesitas ningún archivo "app.dat" ni manipular el contenido de ningún archivo manuálmente, ya que puedes utilizar la infrastructura settings para almacenar la configuración de usuario, dicha configuración se escribe en un archivo del directorio Roaming pero su manipulación es muy sencilla ya que está orientada a objetos:
Using Settings in C#

De todas formas, si quieres generar un archivo legible/editable entonces dale una extensión común de texto plano, cómo "app.txt" o "app.ini", pero si utilizas la extensión ".dat" eso da a entender al end-user que debería tratarse de un archivo binario con contenido "ilegible", y por ende te sugiero aplicar la serialización de datos:

Primero, diseño el modelo de un objeto/contenedor cómo esta estructura de abajo, que servirá para almacenar los datos:
Código (vbnet) [Seleccionar]
''' <summary>
''' Structure that specifies the application configuration.
''' </summary>
<SerializableAttribute>
Public Structure ConfigStruct

   ''' <summary>
   ''' </summary>
   Public Auth As String

   ''' <summary>
   ''' </summary>
   Public Logi As String

   ''' <summary>
   ''' </summary>
   Public Calc As String

   ''' <summary>
   ''' </summary>
   Public Hola As String

   ''' <summary>
   ''' </summary>
   Public Mens As String

   ''' <summary>
   ''' </summary>
   Public Entr As String

   ''' <summary>
   ''' </summary>
   Public Para As String

End Structure


C#:
Código (csharp) [Seleccionar]
using Microsoft.VisualBasic;
using System;
using System.Collections;
using System.Collections.Generic;
using System.Data;
using System.Diagnostics;

/// <summary>
/// Structure that specifies the application configuration.
/// </summary>
[SerializableAttribute()]
public struct ConfigStruct
{
/// <summary>
/// </summary>
public string Auth;

/// <summary>
/// </summary>
public string Logi;

/// <summary>
/// </summary>
public string Calc;

/// <summary>
/// </summary>
public string Hola;

/// <summary>
/// </summary>
public string Mens;

/// <summary>
/// </summary>
public string Entr;

/// <summary>
/// </summary>
public string Para;
}

//=======================================================
//Service provided by Telerik (www.telerik.com)
//=======================================================


Segundo, desarrollo los métodos (genéricos) de serialización y deserialización los cuales me ayudarán a automatizar dicha tarea:
Código (vbnet) [Seleccionar]
Imports System.Runtime.Serialization.Formatters.Binary

''' <summary>
''' Exposes members for binary serialization.
''' </summary>
Public NotInheritable Class BinarySerializerUtil

   ''' <summary>
   ''' Serializes an object into the specified file, using binary serialization.
   ''' </summary>
   ''' <typeparam name="T"></typeparam>
   ''' <param name="object">The object to serialize.</param>
   ''' <param name="filepath">The filepath to save the obstore the serialized data.</param>
   Public Shared Sub Serialize(Of T)(ByVal [object] As T,
                                     ByVal filepath As String)

       Dim serializer As New BinaryFormatter

       Using writer As New FileStream(filepath, FileMode.Create)
           serializer.Serialize(writer, [object])
       End Using

   End Sub

   ''' <summary>
   ''' Deserializes an Object from the specified file, using binary deserialization.
   ''' </summary>
   ''' <typeparam name="T"></typeparam>
   ''' <param name="object">The ByRefered object to store the deserialized data.</param>
   ''' <param name="filepath">The filepath to deserialize its contents.</param>
   Public Shared Sub Deserialize(Of T)(ByRef [object] As T,
                                       ByVal filepath As String)

       Dim serializer As New BinaryFormatter

       Using reader As New FileStream(filepath, FileMode.Open)
           [object] = DirectCast(serializer.Deserialize(reader), T)
       End Using

   End Sub

End Class


C#:
Código (csharp) [Seleccionar]
using Microsoft.VisualBasic;
using System;
using System.Collections;
using System.Collections.Generic;
using System.Data;
using System.Diagnostics;
using System.Runtime.Serialization.Formatters.Binary;

/// <summary>
/// Exposes members for binary serialization.
/// </summary>
public sealed class BinarySerializerUtil
{
/// <summary>
/// Serializes an object into the specified file, using binary serialization.
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="object">The object to serialize.</param>
/// <param name="filepath">The filepath to save the obstore the serialized data.</param>
public static void Serialize<T>(T @object, string filepath)
{
BinaryFormatter serializer = new BinaryFormatter();

using (FileStream writer = new FileStream(filepath, FileMode.Create)) {
serializer.Serialize(writer, @object);
}
}

/// <summary>
/// Deserializes an Object from the specified file, using binary deserialization.
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="object">The ByRefered object to store the deserialized data.</param>
/// <param name="filepath">The filepath to deserialize its contents.</param>
public static void Deserialize<T>(ref T @object, string filepath)
{
BinaryFormatter serializer = new BinaryFormatter();

using (FileStream reader = new FileStream(filepath, FileMode.Open)) {
@object = (T)serializer.Deserialize(reader);
}
}
}

//=======================================================
//Service provided by Telerik (www.telerik.com)
//=======================================================


Por último, lo pongo todo en práctica.
(Si lo necesitases puedes adaptar el ejemplo para serializar también una colección uni o multi dimensional, claro está, aunque dependiendo del tipo de objeto la implementación de una serialización puede requerir más trabajo):
Código (vbnet) [Seleccionar]
Imports System.IO

Public Class Form1

   ''' <summary>
   ''' The filepath where to serialize de data.
   ''' </summary>
   Private ReadOnly datFilePath As String = Path.Combine(Application.StartupPath, "app.dat")

   ''' <summary>
   ''' A <see cref="ConfigStruct"/> instance that specifies the application config.
   ''' </summary>
   Private config As New ConfigStruct With
       {
           .Auth = "Auth value",
           .Calc = "Calc vaule",
           .Entr = "Entr vaule",
           .Hola = "Hola vaule",
           .Logi = "Logi vaule",
           .Mens = "Mens vaule",
           .Para = "Para vaule"
       }

   ''' <summary>
   ''' Handles the <see cref="Form.Shown"/> event of this form.
   ''' </summary>
   Private Sub Test() Handles MyBase.Shown

       ' Serializo los datos de la instancia del objeto, al archivo especificado.
       BinarySerializerUtil.Serialize(Of ConfigStruct)(Me.config, datFilePath)

       ' Limpio los datos del objeto.
       Me.config = Nothing

       ' Deserializo los datos del archivo, a la instancia del objeto especificado.
       BinarySerializerUtil.Deserialize(Of ConfigStruct)(Me.config, datFilePath)

       Debug.WriteLine(Me.config.Auth)
       Debug.WriteLine(Me.config.Calc)
       Debug.WriteLine(Me.config.Entr)
       Debug.WriteLine(Me.config.Hola)
       Debug.WriteLine(Me.config.Logi)
       Debug.WriteLine(Me.config.Mens)
       Debug.WriteLine(Me.config.Para)

   End Sub

End Class


C#:
Código (csharp) [Seleccionar]
using Microsoft.VisualBasic;
using System;
using System.Collections;
using System.Collections.Generic;
using System.Data;
using System.Diagnostics;
using System.IO;

public class Form1
{
/// <summary>
/// The filepath where to serialize de data.
/// </summary>

private readonly string datFilePath = Path.Combine(Application.StartupPath, "app.dat");
/// <summary>
/// A <see cref="ConfigStruct"/> instance that specifies the application config.
/// </summary>
private ConfigStruct config = new ConfigStruct {
Auth = "Auth value",
Calc = "Calc vaule",
Entr = "Entr vaule",
Hola = "Hola vaule",
Logi = "Logi vaule",
Mens = "Mens vaule",
Para = "Para vaule"
};

/// <summary>
/// Handles the <see cref="Form.Shown"/> event of this form.
/// </summary>
private void Test()
{
// Serializo los datos de la instancia del objeto, al archivo especificado.
BinarySerializerUtil.Serialize<ConfigStruct>(this.config, datFilePath);

// Limpio los datos del objeto.
this.config = null;

// Deserializo los datos del archivo, a la instancia del objeto especificado.
BinarySerializerUtil.Deserialize<ConfigStruct>(this.config, datFilePath);

Debug.WriteLine(this.config.Auth);
Debug.WriteLine(this.config.Calc);
Debug.WriteLine(this.config.Entr);
Debug.WriteLine(this.config.Hola);
Debug.WriteLine(this.config.Logi);
Debug.WriteLine(this.config.Mens);
Debug.WriteLine(this.config.Para);

}

public Form1()
{
Shown += Test;
}
}

//=======================================================
//Service provided by Telerik (www.telerik.com)
//=======================================================


No estoy seguro de si algo así es lo que pretendías hacer.

Saludos








Kaxperday

#2
Hola elektro me ha parecido muy interesante tu respuesta. No sabía que se podía hacer eso, probaré a ver que tal se me da eso del settings, aunque no se si se podrá hacer para variables string muy largas, porque tengo pensado meter links de temas de foros en esas variables y pueden ser cientos de lineas incluso quizas miles. Es por eso que ahí ya dudo para eso, aunque para user/pas me parece mejor asi, la idea mia tambien era que si lo hago todo en un archivo encriptarlo todo.

Cuando vuelva con esto te comento, gracias.

Edito: Me costará comprenderlo, siempre hablas con un nivel experto y yo que soy un aprendiz me lleva tiempo analizar tus códigos, pero me ayudas mucho aun asi.

Aquí vuelvo, la idea tuya es bastante buena pero ya te dije que trabajo con cadenas muy largas y prefiero crear un archivo con toda la información y encriptarlo.

Tengo esto:

Código (csharp) [Seleccionar]
public bool setAutentificacion(String certificado)
       {
           int index;
           bool correcto = false;;
           StreamWriter escritor = new StreamWriter(escritura);//intentar escritura sobremismo archivo
           StreamReader lector = new StreamReader(lectura);
           String contenido = lector.ReadToEnd();
           lector.Close();
           if (contenido.Contains(vector[0].ToString()))
           {
               index = contenido.IndexOf(vector[0].ToString());
               if (index != -1)
               {
                   correcto = true;
                   contenido = contenido.Insert(contenido.IndexOf(vector[0].ToString()), certificado);
                   int entrada = contenido.LastIndexOf(vector[0].ToString()+certificado);
                   int salida = contenido.IndexOf(vector[1].ToString());
                   contenido.Remove(entrada, salida-entrada);
                   escritor.Write(contenido);
               }
               else
               {
                   correcto = false;
               }
           }
           else
               correcto = false;
           return correcto;
       }


Ahí ando, aunque como alguien toque el archivo adios, aunque pondre algo para comprobar si esta corrupto y que genere uno nuevo y ya esta, aunque sera dificil.

Me parece buena manera con los indexof() imaginaba que ahi estaria la clave ahora que se usarlos ya veo que es buen camino para identifica lugares intracatenarios.

Por cierto, no entiendo porque esto:

Código (csharp) [Seleccionar]
contenido.LastIndexOf(vector[0].ToString())

Vale cero cuando la cadena contenido contiene:

CitarENTRADA_AUTH_SALIDA
ENTRADA_LOGI_SALIDA
ENTRADA_CALC_SALIDA
ENTRADA_HOLA_SALIDA
ENTRADA_MENS_SALIDA
ENTRADA_ENTR_SALIDA
ENTRADA_PARA_SALIDA
Cuando el poder económico parasita al político ningún partido ni dictador podrá liberarnos de él. Se reserva el 99% ese poder.

Eleкtro

#3
Cita de: Kaxperday en  6 Mayo 2015, 11:56 AMno se si se podrá hacer para variables string muy largas, tengo pensado meter links de temas de foros en esas variables y pueden ser cientos de lineas incluso quizas miles.

El límite teórico de longitud para un String es el equivalente a 'Int32.MaxValue', es decir, 2.147,483,647 de caracteres, pero ya que por defecto .Net utiliza 2 bytes por caracter, el límite sería 1.073,741,823 de caracteres.

Mientras esas variables no superen el límite de longitud, o no superen los 2GB de asignación memoria, entonces no te preocupes ...hay potencia de sobra xD.




Cita de: Kaxperday en  6 Mayo 2015, 11:56 AMpueden ser cientos de lineas incluso quizas miles.

Por lo que comentas tienes un String que tiene una url por cada linea ...y son miles de lineas,
está claro que irás añadiendo urls a ese String, lo que no tengo claro es si también necesitas manipular constantemente ese string para buscar, eliminar o modificar alguna url, pero si llegas a necesitar eso entonces, ¿has pensado en utilizar una colección genérica de tipo List? (List<T>):

La Class List expone varios miembros que te facilitarían todas esas tareas y más (Add, AddRange, Remove, RemoveRange, RemoveAt, RemoveAll, Insert, IndexOf, Find, FindAll, FindIndex, etc...), me parece mucho más productivo que estar partiendo el String para buscar, eliminar o modificar urls, aunque siempre puedes hacerlo también bonito usando LINQ xD.




Cita de: Kaxperday en  6 Mayo 2015, 11:56 AMla idea mia tambien era que si lo hago todo en un archivo encriptarlo todo.

Perfecto, mi ejemplo entonces no ha sido en vano xD por que la serialización binaria le daría una seguridad adicional de ingeniería inversa, ten en cuenta que siempre puedes cifrar el string antes de serializarlo, y desenctriptarlo al deserializar. (al igual que si prefieres usar las settings).

Ah, y también puedes utilizar otro tipo de serializaciones para estructurar el archivo de otra manera, por ejemplo la serialización XML, eso ya depende de tus preferencias:
XmlSerializer Class - MSDN

Ejemplo del archivo "app.dat" usando serialización Binaria:
0001 0000 00ff ffff ff01 0000 0000 0000
000c 0200 0000 4a57 696e 646f 7773 4170
706c 6963 6174 696f 6e31 2c20 5665 7273
696f 6e3d 312e 302e 302e 302c 2043 756c
7475 7265 3d6e 6575 7472 616c 2c20 5075
626c 6963 4b65 7954 6f6b 656e 3d6e 756c
6c05 0100 0000 2057 696e 646f 7773 4170
706c 6963 6174 696f 6e31 2e43 6f6e 6669
6753 7472 7563 7407 0000 0004 4175 7468
044c 6f67 6904 4361 6c63 0448 6f6c 6104
4d65 6e73 0445 6e74 7204 5061 7261 0101
0101 0101 0102 0000 0006 0300 0000 0a41
7574 6820 7661 6c75 6506 0400 0000 0a4c
6f67 6920 7661 756c 6506 0500 0000 0a43
616c 6320 7661 756c 6506 0600 0000 0a48
6f6c 6120 7661 756c 6506 0700 0000 0a4d
656e 7320 7661 756c 6506 0800 0000 0a45
6172 6120 7661 756c 650b


Ejemplo del archivo "app.dat" usando serialización XML:
Código (xml) [Seleccionar]
<?xml version="1.0"?>
<ConfigStruct xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
 <Auth>Auth value</Auth>
 <Logi>Logi vaule</Logi>
 <Calc>Calc vaule</Calc>
 <Hola>Hola vaule</Hola>
 <Mens>Mens vaule</Mens>
 <Entr>Entr vaule</Entr>
 <Para>Para vaule</Para>
</ConfigStruct>


Por cierto, ahora que reviso el código que te mostré en el primer post, me he dado cuenta de que cometí un fallo de escritura en la documentación XML este método:
Citar
Código (csharp,1) [Seleccionar]
/// <param name="filepath">The filepath to save the obstore the serialized data.</param>
public static void Serialize<T>(T @object, string filepath)
...

Quise escribir esto:
Código (csharp,1) [Seleccionar]
/// <param name="filepath">The filepath to store the serialized data.</param>
public static void Serialize<T>(T @object, string filepath)
...


Saludos








Eleкtro

#4
En el código que has añadido en la edición de tu comentario, no liberas el objeto 'escritor', aparte de eso ...alguna otra cosa de menor importancia es que deberías aprovechar la declaración Using, y seguir las buenas prácticas de convenciones de nomenclatura en C# por que 'setAutentificacion' es incorrecto, el Camel-Casing se utiliza al declarar variables (y depende del tipo de visibilidad con la que la declares), los nombres de método se escriben utilizando Word-Casing.

Esto no soluciona el problema que tengas, pero por el momento el código queda más simplificado:
Código (csharp) [Seleccionar]
public bool SetAutentificacion(string certificado)
{
   int index = 0;
   string contenido = null;
   System.Text.Encoding encoding = System.Text.Encoding.Default;

   using (StreamWriter escritor = new StreamWriter(path: escritura, append: false, encoding: encoding))
   {

       using (StreamReader lector = new StreamReader(path: lectura, encoding: encoding))
       {
           contenido = lector.ReadToEnd();
       }

       index = contenido.IndexOf(vector(0));

       if ((index != -1))
       {
           contenido = contenido.Insert(contenido.IndexOf(vector(0)), certificado);
           int entrada = contenido.LastIndexOf(vector(0) + certificado);
           int salida = contenido.IndexOf(vector(1));
           contenido.Remove(entrada, salida - entrada);
           escritor.Write(contenido);
       }
   }
   return (index != -1);
}


PD: Omito la extensión '.ToString' por que doy por hecho que estás utilizando una colección de Strings, y sino, deberías hacerlo. Especifica que tipo de objeto es "vector".




Cita de: Kaxperday en  6 Mayo 2015, 11:56 AMPor cierto, no entiendo porque esto:

Código (csharp) [Seleccionar]
contenido.LastIndexOf(vector[0].ToString())

Vale cero cuando la cadena contenido contiene:

Creo que estás confundiendo la utilización de la función 'LastIndexOf', ¿que valor esperas obtener al llamar a contenido.LastIndexOf?, te está devolviendo el valor correcto, '0' es la posición del primer caracter de la última coindidencia del string que estás buscando.

Si lo que pretendes es obtener la posición del último caracter de la última coincidencia del string que buscas, entonces debes añadirle la longitud ese se string:
Código (vbnet) [Seleccionar]
Dim str As String = "Hello World"
Dim find As String = "Hello"
Dim index As Integer

index = str.LastIndexOf(find) + find.Length ' Resultado: 5


O si pretendes obtener la posición del último caracter de la primera coincidencia del string que buscas, sería así:
Código (vbnet) [Seleccionar]
index = str.IndexOf(find) + find.Length ' Resultado: 5

En este ejemplo ambos resultados son el mismo por que solo se puede encontrar una coincidencia de la palabra "Hello" en el string "Hello World". ¿entiendes?.

Saludos








Kaxperday

#5
Lo primero muchas gracias a elektro, eres un autentico crack de veras gracias por la ayuda.

Lo de settigs es muy atractivo, pero no deja de ser un archivo mas junto a la app, y si podria cargar variables y ponerlas en binario etc, pero ya en el punto que estaba no iba a cambiarlo todo la proxima vez lo tendré en cuenta.

Ya tengo la clase archivo con metodos get y set que me escriben en el archivo los datos en cada apartado guiándose por los indexof(), y cadenas separadoras. Y funciona bien.

Tengo que meter un control de errores para comprobar si el archivo esta corrompido y restaurarlo en el caso.

En lo del método setAutentificación, puedo decirte que en clase siempre hemos usado Camel-Case para Java y de ahí lo lleve a C#, aunque no siempre me han gustado

Y por lo de indexof() muchas gracias, pensaba que ambos eran para la primera coincidencia, y que uno se situaba al principio de la coincidencia indexof y otro al inal lastindexof.

Ahora lo que me pasa no se si tendrá remedio, quiero trabajar sobre un solo archivo, tengo que abrirlo leerlo y sobreescribirlo, es decir leer su contenido y escribir lo que quiero cambiar en el solo. No sé si es posible, antes hice 2 archivos distintos un StreamWriter y un StreamReader.

Código (csharp) [Seleccionar]
private FileStream lectura;
       private FileStream escritura;

       public Archivo(String directorio)
       {
           this.lectura = new FileStream(directorio + "\\app1.txt", FileMode.Open);
           this.escritura = new FileStream(directorio + "\\app2.txt", FileMode.Create);
       }


¿No se puede hacer de alguna manera con un solo FileStream leer el archivo y a la vez escribirlo?
Antes de nada voy a probar a hacerlo con un solo FileStream archivo.

Creo que así es posible, fuera StreamReader y StreamWriter. Un solo FileStream tiene métodos de lectura y escritura, creo que ahí estará lo que busco, de todas formas trabaja con bytes y me costará un poco, si creéis que hay otro camino mejor no estaría demás comentarlo.

Gracias estoy con ello socios.

Edito: Luego tengo pensado  cifrar todo el archivo y puede quedar bastante elegante. Saludos.

Ya encontre la respuesta :
http://stackoverflow.com/questions/605685/how-to-both-read-and-write-a-file-in-c-sharp

Se olvidan de FileStream, StreamReader y StreamWriter, una maravilla. Aunque habrá que controlar excepciones. gg.

Código (csharp) [Seleccionar]
static void Main(string[] args)
   {
       var text = File.ReadAllText(@"C:\words.txt");
       File.WriteAllText(@"C:\words.txt", text + "DERP");
   }


Simplemente leo todo en una cadena, la modifico y escribo la nueva cadena en el mismo archivo sobreescribiendo los datos anteriores :))

Por cierto elektro tienes razón con los metodos, pues los metodos por defecto de las clases salen ambas palabras comenando en mayuscula.

He avanzado bastante con el proyecto, asi da gusto, espero seguir dándole caña, siempre mejorando.
Cuando el poder económico parasita al político ningún partido ni dictador podrá liberarnos de él. Se reserva el 99% ese poder.

Eleкtro

#6
Cita de: Kaxperday en  7 Mayo 2015, 16:21 PMSe olvidan de FileStream, StreamReader y StreamWriter, una maravilla. Aunque habrá que controlar excepciones. gg.

Y la Class StringReader, StringWritter, y StringBuilder, miratelas (esta última para construir un string multilinea viene muy bien con el método Append, AppendFormat y/o AppendLine).

StringReader Class - MSDN
StringWriter Class - MSDN
StringBuilder Class - MSDN

PD: La Class StringWritter expone su StringBuilder llamando a StringWriter.GetStringBuilder.

Saludos