Ich habe ein Problem beim Lesen von IniFiles mit verschiedenen Codierungen. Wenn ich eine Unicode-Datei lese, scheint GetPrivateProfileSectionNamesA über die erste Zeile zu stolpern. ASCII oder ANSI funktioniert gut. Ich habe ein kleines Programm geschrieben, um mein Problem zu illustrieren. Zuerst die Ausgabe, dann das Programm. Ich interessiere mich nicht wirklich für UTF7 und UTF32, aber was ich nicht bekomme, ist der UTF8-Teil. Muss ich eine andere Funktion verwenden, um Unicode IniFiles zu lesen? Tue ich etwas falsch? Hoffnung jemand kann mir helfen, dank Norbertkernel32.dll - GetPrivateProfileSectionNamesA
, was ich bekommen:
IniEntriesWithSectionInFirstLine
first section using System.Text.ASCIIEncoding is FirstSectionInFirstLine
first section using System.Text.Latin1Encoding is FirstSectionInFirstLine
first section using System.Text.UTF7Encoding is
first section using System.Text.UTF8Encoding is SecondSection
first section using System.Text.UTF32Encoding is SecondSectio????????????
IniEntriesWithFirstLineEmpty
first section using System.Text.ASCIIEncoding is FirstSectionInSecondLine
first section using System.Text.Latin1Encoding is FirstSectionInSecondLine
first section using System.Text.UTF7Encoding is
first section using System.Text.UTF8Encoding is FirstSectionInSecondLine
first section using System.Text.UTF32Encoding is FirstSectionInSecondLin????????
das Programm:
using System;
using System.Collections.Generic;
using System.IO;
using System.Runtime.InteropServices;
using System.Text;
namespace TestIniRead
{
internal class Program
{
[DllImport("kernel32.dll", EntryPoint = "GetPrivateProfileSectionNamesA")]
private static extern int GetSectionNamesListA(
byte[] lpszReturnBuffer,
int nSize,
string lpFileName);
private static readonly string[] IniEntriesWithSectionInFirstLine = {
"[FirstSectionInFirstLine]",
"value=firsValue",
"",
"[SecondSection]",
"value=secondValue",
"",
"[ThirdSection]",
"value=secondValue",
""
};
private static readonly string[] IniEntriesWithFirstLineEmpty = {
"",
"[FirstSectionInSecondLine]",
"value=firsValue",
"",
"[SecondSection]",
"value=secondValue",
"",
"[ThirdSection]",
"value=secondValue",
""
};
private static void Main()
{
var fileInfo = new FileInfo("test.ini");
Console.WriteLine("IniEntriesWithSectionInFirstLine");
TestEncodings(fileInfo, IniEntriesWithSectionInFirstLine);
Console.WriteLine("");
Console.WriteLine("IniEntriesWithFirstLineEmpty");
TestEncodings(fileInfo, IniEntriesWithFirstLineEmpty);
Console.ReadLine();
}
private static void TestEncodings(FileInfo fileInfo, IEnumerable<string> iniEntries)
{
TestEncoding(fileInfo, iniEntries, Encoding.ASCII);
TestEncoding(fileInfo, iniEntries, Encoding.GetEncoding("ISO-8859-1"));
TestEncoding(fileInfo, iniEntries, Encoding.UTF7);
TestEncoding(fileInfo, iniEntries, Encoding.UTF8);
TestEncoding(fileInfo, iniEntries, Encoding.UTF32);
}
private static void TestEncoding(FileInfo fileInfo, IEnumerable<string> iniEntries, Encoding encoding)
{
CreateIniFile(fileInfo, iniEntries, encoding);
if (fileInfo.Exists)
{
var buffer = new byte[fileInfo.Length];
GetSectionNamesListA(buffer, (int) fileInfo.Length, fileInfo.FullName);
String s = encoding.GetString(buffer);
String[] names = s.Split('\0');
Console.WriteLine("first section using {0} is {1}", encoding, names[0]);
}
}
private static void CreateIniFile(FileSystemInfo fileInfo, IEnumerable<string> iniEntries, Encoding encoding)
{
using (var sw = new StreamWriter(File.Open(fileInfo.FullName, FileMode.Create), encoding))
{
foreach (string line in iniEntries)
{
sw.WriteLine(line);
}
}
}
}
}
Reaktion auf die ersten drei Antworten:
Sie sind natürlich richtig. Ich sollte GetPrivateProfileSectionNamesW für Unicode-Dateien verwenden. Ich fügte eine Methode hinzu, um die Codierung der IniFile zu erhalten, und verwendete A oder W entsprechend. Das Problem bleibt gleich. Die Funktion erhält den ersten Abschnitt nicht. Unter sehen Sie neuen Code nur für UTF8.
, was ich bekommen:
IniEntriesWithSectionInFirstLine
first section using System.Text.UTF8Encoding is SecondSection
das Programm:
using System;
using System.Collections.Generic;
using System.IO;
using System.Runtime.InteropServices;
using System.Text;
namespace TestIniRead
{
internal class Program
{
[DllImport("kernel32.dll", EntryPoint = "GetPrivateProfileSectionNamesA")]
private static extern int GetSectionNamesListA(
byte[] lpszReturnBuffer,
int nSize,
string lpFileName);
[DllImport("kernel32", EntryPoint = "GetPrivateProfileSectionNamesW", CharSet = CharSet.Unicode)]
private static extern int GetSectionNames
(
[MarshalAs(UnmanagedType.LPWStr)] string szBuffer,
int nlen,
string filename
);
private static readonly string[] IniEntriesWithSectionInFirstLine = {
"[FirstSectionInFirstLine]",
"value=firsValue",
"",
"[SecondSection]",
"value=secondValue",
"",
"[ThirdSection]",
"value=secondValue",
""
};
private static void Main()
{
var fileInfo = new FileInfo("test.ini");
Console.WriteLine("IniEntriesWithSectionInFirstLine");
TestEncodings(fileInfo, IniEntriesWithSectionInFirstLine);
Console.WriteLine("");
Console.ReadLine();
}
private static void TestEncodings(FileInfo fileInfo, IEnumerable<string> iniEntries)
{
TestEncoding(fileInfo, iniEntries, Encoding.UTF8);
}
private static readonly char[] separator = { '\0' };
private static void TestEncoding(FileInfo fileInfo, IEnumerable<string> iniEntries, Encoding encoding)
{
CreateIniFile(fileInfo, iniEntries, encoding);
if (fileInfo.Exists)
{
int len = (int)fileInfo.Length;
var buffer = new string('\0', len);
int nlen = GetSectionNames(buffer, len, fileInfo.FullName);
if (nlen <= 0)
{
Environment.Exit(nlen);
}
String[] names = buffer.Substring(0, nlen).Split(separator);
Console.WriteLine("first section using {0} is {1}", encoding, names[0]);
}
}
private static void CreateIniFile
(
FileSystemInfo fileInfo,
IEnumerable<string> iniEntries,
Encoding encoding)
{
using (var sw = new StreamWriter(File.Open(fileInfo.FullName, FileMode.Create), encoding))
{
foreach (string line in iniEntries)
{
sw.WriteLine(line);
}
}
}
}
}
Sie haben natürlich Recht. Ich sollte GetPrivateProfileSectionNamesW für Unicode-Dateien verwenden. Ich fügte eine Methode hinzu, um die Codierung der IniFile zu erhalten, und verwendete A oder W entsprechend. Das Problem bleibt gleich. Die Funktion erhält den ersten Abschnitt nicht. Siehe Änderungen oben. –
Ich vermute immer noch, dass die API-Funktion Probleme beim Verarbeiten der Byte-Reihenfolgezeichen am Anfang der Datei hat. Versuchen Sie, Ihren Texteditor nicht einzuschließen. Sie können überprüfen, ob sie vorhanden sind, indem Sie die Datei in einem Hex-Editor öffnen. (Textpad kann Textdateien in Hex-Ansicht öffnen und Sie können steuern, was, wenn Byte-Reihenfolge Marken enthalten sind) – pipTheGeek