PDA

Ver la Versión Completa : Leer fichero ini


Angel.Matilla
31-03-2015, 10:14:57
Teniendo puesto el lógico include (#include <inifiles.hpp>), tengo este fichero ini:
[Datos]
Entorno=C:\DatAfi21\Gia.ini
Acceso=C:\DatAfi21\Acceso
Tablas=C:\DatAfi21\Tablasque he grabado con este código:
TIniFile *fIni = new TIniFile(ChangeFileExt(Application->ExeName, ".ini"));
fIni->WriteString("Datos", "Entorno", "C:\DatAfi21\\DatAfi21\\Gia.ini");
fIni->WriteString("Datos", "Acceso" , "C:\DatAfi21\\DatAfi21\\Acceso");
fIni->WriteString("Datos", "Tablas" , "C:\DatAfi21\\DatAfi21\\Tablas");
delete fIni;Sin embargo si trato de leerlo con este otro código:
TIniFile *fIni = new TIniFile(ChangeFileExt(Application->ExeName, ".ini"));
cEntorno = fIni->ReadString("Datos", "Entorno", cEntorno);
cAcceso = fIni->ReadString("Datos", "Acceso" , cAcceso);
cTablas = fIni->ReadString("Datos", "Tablas" , cTablas);
delete fIni;Siempre me devuelve cadenas vacías y no entiendo el motivo. Y me da lo mismo hacerlo sobre Builde 6 que sobre Builder XE3.

duilioisola
31-03-2015, 11:14:46
No tengo mucha experiencia con C, pero:

Veo que la primera barra separadora de carpetas es simple (una sola) y esto puede que C lo considere como terminador de cadena.


fIni->WriteString("Datos", "Entorno", "C:\DatAfi21\\DatAfi21\\Gia.ini");

Angel.Matilla
31-03-2015, 11:55:12
No tengo mucha experiencia con C, pero:

Veo que la primera barra separadora de carpetas es simple (una sola) y esto puede que C lo considere como terminador de cadena.


fIni->WriteString("Datos", "Entorno", "C:\DatAfi21\\DatAfi21\\Gia.ini");

Ha sido una erra al copiar; están puestas las barras dobles. De hecho, esa primera parte (C:\DatAfi21) está guardada en una variable que capturo a través de un SHBrowseForFolder.

Casimiro Notevi
31-03-2015, 13:04:57
¿El fichero ini está bien? Si está mal es que no lo grabas bien. Si está bien es que no lo lees bien.
Sigue con el depurador paso a paso hasta llegar hasta esa línea y mira el contenido de la variable.

Angel.Matilla
31-03-2015, 13:46:55
Gracias por la respuesta. Centrándome en que ya estoy probando sólo con Builder XE3.

La grabación es la que he puesto antes y lo hago con el primer trozo de código que puse, de lo que deduzco que está bien hecho. En la segunda parte, cuando intento leerlo, incializo las variables (por si acaso)
cAcceso = "";
cEntorno = "";
cTablas = "";y ejecuto el segundo código que he puesto. Me he dado cuenta que si pongo esto:
TStringList *slIniFile = new TStringList();
TIniFile *fIni = new TIniFile(ChangeFileExt(Application->ExeName, ".ini"));
fIni->ReadSectionValues(UnicodeString("DATOS"), slIniFile);
cAux = slIniFile->Text;en este slIniFile->Text aparece esto mirando con el depurador:
"Entorno=C:\\DatAfi21\\Gia.ini\r\nAcceso=C:\\DatAfi21\\Acceso\r\nTablas=C:\\DatAfi21\\Tablas\r\n"lo que significa que estar la información está, pero si hago esto:
cEntorno = slIniFile->Strings[0];que en teoría debería valer Entorno=C:\\DatAfi21\\Gia.ini, ¡no hay ningún valor en la variable!Me devuelve NULL. Me tiene total y aboslutamente despistado.

Casimiro Notevi
31-03-2015, 14:03:04
La grabación .. deduzco que está bien hecho.

Ángel, antes de probar la lectura, debes asegurarte, no solamente deducir, que se han grabado bien los datos.
Lo mínimo a probar es buscar el fichero ini en el disco y abrirlo con cualquier editor de textos para ver si está correcto.

Luego, si está bien, es cuando hay que seguir con el problema de la lectura.

ecfisa
01-04-2015, 03:41:49
Hola Angel.

Estuve probando, básicamente con tu código y no tengo ningún problema, te adjunto el código:

#include <inifiles.hpp>

AnsiString iniName = ChangeFileExt(Application->ExeName, ".ini");
AnsiString cEntorno, cAcceso, cTablas;

void __fastcall TForm1::btnSaveClick(TObject *Sender) {
TIniFile* ini = new TIniFile (iniName);

ini->WriteString("Datos", "Entorno", "C:\\DatAfi21\\DatAfi21\\Gia.ini");
ini->WriteString("Datos", "Acceso" , "C:\\DatAfi21\\DatAfi21\\Acceso");
ini->WriteString("Datos", "Tablas" , "C:\\DatAfi21\\DatAfi21\\Tablas");

delete ini;
}

void __fastcall TForm1::btnToStringClick(TObject *Sender) {

TIniFile* ini = new TIniFile(iniName);

cEntorno = ini->ReadString("Datos", "Entorno", cEntorno);
cAcceso = ini->ReadString("Datos", "Acceso" , cAcceso);
cTablas = ini->ReadString("Datos", "Tablas" , cTablas);

delete ini;
Memo1->Clear();
Memo1->Lines->Add(cEntorno);
Memo1->Lines->Add(cAcceso);
Memo1->Lines->Add(cTablas);
}

void __fastcall TForm1::btnToStringsClick(TObject *Sender) {
TIniFile* ini = new TIniFile(iniName);
TStrings* sl = new TStringList;

ini->ReadSectionValues("Datos", sl);
Memo2->Clear();
Memo2->Lines->Assign(sl);

delete ini;
delete sl;
}


Muestra:
http://s18.postimg.org/jrd8smp21/Angel_Matilla.png

Saludos :)

Angel.Matilla
01-04-2015, 09:58:40
Ángel, antes de probar la lectura, debes asegurarte, no solamente deducir, que se han grabado bien los datos.
Lo mínimo a probar es buscar el fichero ini en el disco y abrirlo con cualquier editor de textos para ver si está correcto.
Siento haberme expresado mal. El contenido del fichero que puse en el primer mensaje es un copiado y pegado a partir del bloc de notas, lo que me dice que el fichero está bien grabado.

Estuve probando, básicamente con tu código y no tengo ningún problema, te adjunto el código:
Entonces, básicamente, no lo entiendo. :confused:
Lo he probado yo también en un proyecto vacío y efectivamente funciona; ¿por qué demonios no me funciona a mi un código que como dices es en lo esencial igual? Seguiré probando.

Casimiro Notevi
01-04-2015, 10:58:19
¿No limpiarás las variables antes de volver a leerlas?

Angel.Matilla
01-04-2015, 11:13:22
¿No limpiarás las variables antes de volver a leerlas?
No. Las inicializo al principio de la función. No quería extenderme en exceso proque hay llamadas a funciones externas y demás, pero este es el código completo que estoy usando:
void __fastcall TfPersona::FormCreate(TObject *Sender)
{
BROWSEINFO bi;
int nDirectorio = 0;
LPITEMIDLIST ItemID;
TStringList *slIniFile = new TStringList();
UnicodeString cInicio;
wchar_t WDir[MAX_PATH], FolderName[MAX_PATH];

UnicodeString cAcceso = "";
UnicodeString cEntorno = "";
UnicodeString cTablas = "";

cAux = ChangeFileExt(Application->ExeName, ".ini");
if (!FileExists(cAux))
{
slBotones->Clear();
slBotones->Text = "Buscar\r\nCancelar";
if (Mensaje(this, String("Instalación"), String("Indique la situación de la base de datos"), tdiShield, slBotones) == mrCancel)
{
slBotones->Text = "Abandonar";
Mensaje(this, String("Instalación"), String("No puede iniciarse la sesión.\r\nDesconozco la situación de los ficheros."), tdiError, slBotones);
Application->Terminate();
return;
}

while (nDirectorio == 0)
{
memset(&bi, 0, sizeof(BROWSEINFO));
memset(WDir, 0, MAX_PATH);
bi.hwndOwner = NULL;
bi.pszDisplayName = FolderName;
bi.lpszTitle = String("Seleccione la situación de la base de datos").c_str();
ItemID = SHBrowseForFolder(&bi);
SHGetPathFromIDList(ItemID, WDir);
GlobalFreePtr(ItemID);
cInicio = String(WDir);

if (cInicio == "")
{
slBotones->Text = "Abandonar";
Mensaje(this, String("Instalación"), String("No puede iniciarse la sesión.\r\nDesconozco la situación de los ficheros."), tdiError, slBotones);
Application->Terminate();
return;
}

if (cInicio[1] > 'B')
{
if (cInicio.LastDelimiter("\\") == cInicio.Length())
cInicio = cInicio.SubString(1, cInicio.LastDelimiter("\\") - 1);

cAux = "La base de datos de afiliados se ";
if (DirectoryExists(cInicio + "\\DatAfi21"))
cAux = cAux + "encuentra";
else
cAux = cAux + "creará";
cAux = cAux + " en:\r\n\r\n" + cInicio + "\\DatAfi21\r\n\r\n¿Es correcta la elección?";
slBotones->Text = "Es correcta\r\nCambiar ubicación";
if (Mensaje(this, String("Instalación"), cAux, tdiNone, slBotones) == 100)
{
nDirectorio = 1;
try
{
System::ChDir(cInicio);
}
catch(...)
{
slBotones->Text = "Volver";
Mensaje(this, String("Instalación"), String("La vía seleccionada no está accesible.\r\nRevise su elección, por favor."), tdiError, slBotones);
nDirectorio = 0;
}
}
}
else
{
slBotones->Text = "Volver";
Mensaje(this, String("Instalación"), String("La vía seleccionada no está accesible.\r\nRevise su elección, por favor."), tdiError, slBotones);
nDirectorio = 0;
}
}

TIniFile *fIni = new TIniFile(ChangeFileExt(Application->ExeName, ".ini"));
fIni->WriteString("DATOS", "Entorno", cInicio + "\\DatAfi21\\Gia.ini");
fIni->WriteString("DATOS", "Acceso" , cInicio + "\\DatAfi21\\Acceso");
fIni->WriteString("DATOS", "Tablas" , cInicio + "\\DatAfi21\\Tablas");
delete fIni;
}

TIniFile *fIni = new TIniFile(ChangeFileExt(Application->ExeName, ".ini"));
fIni->ReadSectionValues(UnicodeString("DATOS"), slIniFile);
cEntorno = UnicodeString(fIni->ReadString(UnicodeString("DATOS"), "Entorno", cEntorno));
cAcceso = UnicodeString(fIni->ReadString(UnicodeString("DATOS"), "Acceso" , cAcceso));
cTablas = UnicodeString(fIni->ReadString(UnicodeString("DATOS"), "Tablas" , cTablas));
delete fIni;
}

Angel.Matilla
01-04-2015, 12:36:59
Después de toda la mañana dando vueltas y probando diferentes códigos he encontardo, más o menos. una solución.
TIniFile *fIni = new TIniFile(cIniFile);
TStringList *Entorno = new TStringList();
fIni->ReadSectionValues("Datos", Entorno);

Entorno->Strings[0] = Entorno->Strings[0].SubString(Entorno->Strings[0].Pos("=") + 1, Entorno->Strings[0].Length());
Entorno->Strings[1] = Entorno->Strings[1].SubString(Entorno->Strings[1].Pos("=") + 1, Entorno->Strings[1].Length());
Entorno->Strings[2] = Entorno->Strings[2].SubString(Entorno->Strings[2].Pos("=") + 1, Entorno->Strings[2].Length());
delete fIni;No es la forma que más me seduce pero no he encontrado otra ya que si trato de hacer esto:
UnicodeString cEntorno = Entorno->Strings[0].SubString(Entorno->Strings[0].Pos("=") + 1, Entorno->Strings[0].Length());dentro de la variable me devuelve siempre una cadena vacía.

ecfisa
01-04-2015, 13:37:09
Hola Angel.Matilla


...
Sin embargo si trato de leerlo con este otro código:

TIniFile *fIni = new TIniFile(ChangeFileExt(Application->ExeName, ".ini"));
cEntorno = fIni->ReadString("Datos", "Entorno", cEntorno);
cAcceso = fIni->ReadString("Datos", "Acceso" , cAcceso);
cTablas = fIni->ReadString("Datos", "Tablas" , cTablas);
delete fIni;

Siempre me devuelve cadenas vacías y no entiendo el motivo. Y me da lo mismo hacerlo sobre Builde 6 que sobre Builder XE3.

En C++ Builder 6 ese código no devuelve cadenas vacías (#7 (http://www.clubdelphi.com/foros/showpost.php?p=490723&postcount=7)).


...
Me he dado cuenta que si pongo esto:

TStringList *slIniFile = new TStringList();
TIniFile *fIni = new TIniFile(ChangeFileExt(Application->ExeName, ".ini"));
fIni->ReadSectionValues(UnicodeString("DATOS"), slIniFile);
cAux = slIniFile->Text;

en este slIniFile->Text aparece esto mirando con el depurador:
"Entorno=C:\\DatAfi21\\Gia.ini\r\nAcceso=C:\\DatAfi21\\Acceso\r\nTablas=C:\\DatAfi21\\Tablas\r\n "
lo que significa que estar la información está, pero si hago esto:
cEntorno = slIniFile->Strings[0];que en teoría debería valer Entorno=C:\\DatAfi21\\Gia.ini, ¡no hay ningún valor en la variable!Me devuelve NULL.
Me tiene total y aboslutamente despistado.

En C++ Builder 6, slIniFile->Strings[0] tiene el valor: Entorno=C:\DatAfi21\DatAfi21\Gia.ini

En resumen, tal como ya te comente en el mensaje #7 (http://www.clubdelphi.com/foros/showpost.php?p=490723&postcount=7), en C++ Builder 6 ambos casos funcionan.

Después de toda la mañana dando vueltas y probando diferentes códigos he encontardo, más o menos. una solución.
TIniFile *fIni = new TIniFile(cIniFile);
TStringList *Entorno = new TStringList();
fIni->ReadSectionValues("Datos", Entorno);

Entorno->Strings[0] = Entorno->Strings[0].SubString(Entorno->Strings[0].Pos("=") + 1, Entorno->Strings[0].Length());
Entorno->Strings[1] = Entorno->Strings[1].SubString(Entorno->Strings[1].Pos("=") + 1, Entorno->Strings[1].Length());
Entorno->Strings[2] = Entorno->Strings[2].SubString(Entorno->Strings[2].Pos("=") + 1, Entorno->Strings[2].Length());
delete fIni;
...

Pero eso es otra cosa que lo que mencionas en el primer mensaje. Ahora de lo obtenido con ReadSectionValues queres obtener la parte de la cadena posterior al "="...

La forma que expones es correcta, aunque podes ahorrar un poco de código usando un ciclo:

void __fastcall TForm1::afterEqual(TObject *Sender) {
TIniFile* ini = new TIniFile (iniName);
TStrings* sl = new TStringList;

ini->ReadSectionValues("Datos", sl);
for(int i = 0; i < sl->Count; i++)
sl->Strings[i] = sl->Strings[i].SubString(sl->Strings[i].Pos("=")+1, MaxInt);

Memo2->Clear();
Memo2->Lines->Assign(sl);

// (*)
AnsiString Entorno = sl->Strings[0].SubString(sl->Strings[0].Pos("=")+1, MaxInt);
ShowMessage(Entorno);

delete ini;
delete sl;
}


...
No es la forma que más me seduce pero no he encontrado otra ya que si trato de hacer esto:
UnicodeString cEntorno = Entorno->Strings[0].SubString(Entorno->Strings[0].Pos("=") + 1, Entorno->Strings[0].Length());
dentro de la variable me devuelve siempre una cadena vacía.
Como muestro bajo el comentario (*), en C++ Builder 6 no devuelve cadena vacía

Sobre si C++ Builder XE3 o el comportamiento de la función UnicodeString varían de algún modo el resultado lo ignoro, ya que mi versión de C++ Builder no la incluye.

Saludos :)

Kiranov
21-04-2015, 22:58:47
Buenas tardes,

La verdad hace mucho que no programo en C, pero se que la sintaxis para ReadString es la misma tanto en C como en Delphi.
La sintaxis es la siguiente:

Delphi
function ReadString(const Section, Ident, Default: string): string; override;

C++
virtual System::UnicodeString __fastcall ReadString(const System::UnicodeString Section, const System::UnicodeString Ident, const System::UnicodeString Default);

Section= La sección del archivo INI que vamos a leer.
Ident= La clave que estamos buscando.
Default= Al valor que va a tomar en casi de que no exista la clave.

En tu caso tienes asignado la misma valor en donde va "default":

TIniFile *fIni = new TIniFile(ChangeFileExt(Application->ExeName, ".ini"));
cEntorno = fIni->ReadString("Datos", "Entorno", cEntorno);
cAcceso = fIni->ReadString("Datos", "Acceso" , cAcceso);
cTablas = fIni->ReadString("Datos", "Tablas" , cTablas);
delete fIni;

Intenta cambiando el valor default por un valor vacio por ejemplo:

TIniFile *fIni = new TIniFile(ChangeFileExt(Application->ExeName, ".ini"));
cEntorno = fIni->ReadString("Datos", "Entorno", "");
cAcceso = fIni->ReadString("Datos", "Acceso" , "");
cTablas = fIni->ReadString("Datos", "Tablas" , "");
delete fIni;

Saludos!

Casimiro Notevi
21-04-2015, 23:28:27
Recuerda poner los tags al código fuente, ejemplo:

http://www.clubdelphi.com/images/UtilizarTAGs.png

Gracias :)