我正在构建一个WPF程序,该程序正在为模拟程序编辑保存文件,以便更容易地调整某些参数。该(txt)文件的结构如下所示
{Verdieping 1}
A bunch of data
'Max'
15
20
10
]);
More data
{Verdieping 2}
A bunch of data
'Max'
25
10
40
20
]);
More data
{Verdieping 3}
etc.
因为我需要的列表名称总是'Max'我不能使用Replace("Old_Value", "New_Value");
因此我需要找到标题{Verdieping 1}然后查找'Max'并将值写下来直到达到“]);”所以我最终得到这样的东西:
Verdieping_1[0] = 15
Verdieping_1[1] = 20
Verdieping_1[0] = 10
Verdieping_2[0] = 25
Verdieping_2[1] = 10
Verdieping_2[2] = 40
Verdieping_2[3] = 20
然后,当我更改这些值时,我需要将它们写回到正确位置的文件中。现在我正在使用此代码来读取并最终写入新文件:
var Model = System.IO.File.ReadAllText(@"C:\Base_Model.txt");
//format code here...
System.IO.File.WriteAllText(@"C:\New_Model.txt", Model);
有人可以给我一些提示或帮助,以便我能最好地解决这个问题吗?
修改 这是我想编辑的文件的实际文件结构:
{Atom: Verdieping 2}
int015(0, 64, [
1
2
3
]);
int015(1, 64, [`ActivityID`
20
21
33
]);
int015(2, 64, [`vtp`
139830296
189540320
169919424
]);
int015(3, 64, [`%`
100
100
100
]);
int015(4, 64, [`Max`
15
5
20
]);
int015(5, 64, [`Fill (1 = fill, 0 = empty)`
1
1
0
]);
SetStatus(0);
int018;
{Atom: Verdieping 2}
int015(0, 64, [
1
2
3
]);
int015(1, 64, [`ActivityID`
22
23
24
]);
int015(2, 64, [`vtp`
166058172
165557860
155960564
]);
int015(3, 64, [`%`
100
100
100
]);
int015(4, 64, [`Max`
15
20
10
]);
int015(5, 64, [`Fill (1 = fill, 0 = empty)`
0
1
1
]);
SetStatus(0);
int018;
我之前谈到的'Max'实际上被称为:
int015(4,64,[Max
正如您所看到的那样,每个图层的名称都是相同的,变量的数量是不同的。
感谢您的帮助!
答案 0 :(得分:1)
var result = Regex.Matches(File.ReadAllText(fileName), @"\[`Max`(.*?)\]", RegexOptions.Singleline)
.Cast<Match>()
.Select(m => m.Groups[1].Value.Split(" \t\n\r".ToArray(), StringSplitOptions.RemoveEmptyEntries))
.ToList();
foreach (var v in result)
{
Console.WriteLine(String.Join(", ", v));
}
输出:
15, 5, 20
15, 20, 10
答案 1 :(得分:0)
正则表达可能吗?我做了这个快速的答案,不应该复制粘贴;你可以考虑结果。
在这个答案中,我希望你的数组按时间顺序排列,以防止源代码太长而且凌乱。
我为'DataContainer'对象读取的文件看起来与下面的文件相同:
{Verdieping 1}
A bunch of data
'Max'
15
20
10
]);
More data
{Verdieping 2}
A bunch of data
'Max'
25
10
40
20
]);
首先,我在字符串中找到每个'Max' 然后我将下面的所有数据打印出来
用于查找此内容的正则表达式为:\'Max\'.*?\]\);
尝试使用此代码查看结果
// fill our container
string DataContainer = string.Empty;
using (StreamReader sr = new StreamReader(Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.Desktop), "TestFile.txt"))) {
DataContainer = sr.ReadToEnd();
}
// print data
Match regMatch = Regex.Match(DataContainer, @"\'Max\'.*?\]\);", RegexOptions.Singleline);
int verdiepc = 0;
while (regMatch.Success) {
string[] Data = regMatch.Value.Replace("\r", "").Split('\n');
for (int I = 1; I < Data.Length - 1; I++) {
Console.WriteLine("Verdieping_{0}[{1}] = {2}"
, verdiepc, I, Data[I]);
}
Console.Write("\n");
verdiepc ++;
regMatch = regMatch.NextMatch();
}
// prevent immidiate exit
Console.ReadKey();
输出将是:
Verdieping_0[1] = 15
Verdieping_0[2] = 20
Verdieping_0[3] = 10
Verdieping_1[1] = 25
Verdieping_2[2] = 10
........ etc..
我希望你能用这个答案。这是一个如何使用正则表达式来解析文件的草图
答案 2 :(得分:0)
string LoadedFile = File.ReadAllText(@"C:\Base_Model.txt");
StringBuilder result = new StringBuilder();
for (int i = 1; i < Int32.MaxValue; i++)
{
Match ma = Regex.Match(LoadedFile, "Verdieping " + i.ToString() + ".+?'Max'.*?([0-9].+?)[]][)];", RegexOptions.Singleline);
if(!ma.Success)
{
break;
}
string a = ma.Result("$1");
MatchCollection mc = Regex.Matches(a, "(\\d+?)\r\n", RegexOptions.Singleline);
for (int j = 0; j < mc.Count; j++)
{
result.AppendLine(String.Format("Verdieping_{0}[{1}] = {2}", i, j, mc[j].Result("$1")));
}
result.AppendLine(String.Empty);
}
File.WriteAllText(@"C:\New_Model.txt", result.ToString());
答案 3 :(得分:0)
您可以初始化包含所需输出的新List
string
。然后,阅读文件,获取目标部分之间的信息,并将[
和]);
之间的值插入List
的{{1}}。
示例强>
string
示例输入
static List<string> ReadFile(string Path, string Section)
{
#region Declare Variables
string[] Lines = File.ReadAllLines(Path); //Read Path into Lines
int CurrentValue = 1; //Declare CurrentValue as an Integer of value 1
List<string> RequiredValues = new List<string>(); //Initialize a new List of string of name RequiredValues
#endregion
#region Collect Information
//Look through the section getting the data between [ and ]);
for (int i = 0; i < Lines.Length; i++)
{
if (Lines[i].StartsWith("{" + Section + "}"))
{
i++;
while (!Lines[i].StartsWith("{") && !Lines[i].EndsWith("}"))
{
i++;
if (Lines.Length > i)
{
if (Lines[i].Contains('['))
{
while (true)
{
i++;
if (Lines[i].Contains("]);"))
{
break;
}
RequiredValues.Add(Section + "[" + CurrentValue.ToString() + "] = " + Lines[i]);
CurrentValue++;
}
}
RequiredValues.Add("");
}
else
{
break; //Reached the end of the document
}
}
}
}
#endregion
return RequiredValues; //Return RequiredValues
}
private void Window_Loaded(object sender, RoutedEventArgs e)
{
List<string> Contents = ReadFile(@"C:\Base_Model.txt", "Atom: Verdieping 2");
foreach (string str in Contents)
{
System.Diagnostics.Debug.WriteLine(str); //Write str to the trace listeners
}
}
样本输出
从阅读{Atom: Verdieping 2}
int015(0, 64, [
4
5
6
]);
int015(1, 64, [`ActivityID`
20
21
33
]);
int015(2, 64, [`vtp`
139830296
189540320
169919424
]);
int015(3, 64, [`%`
100
100
100
]);
int015(4, 64, [`Max`
15
5
20
]);
int015(5, 64, [`Fill (1 = fill, 0 = empty)`
1
1
0
]);
SetStatus(0);
int018;
{Atom: Verdieping 1}
int015(0, 64, [
1
2
3
]);
int015(1, 64, [`ActivityID`
22
23
24
]);
int015(2, 64, [`vtp`
166058172
165557860
155960564
]);
int015(3, 64, [`%`
100
100
100
]);
int015(4, 64, [`Max`
15
20
10
]);
int015(5, 64, [`Fill (1 = fill, 0 = empty)`
0
1
1
]);
SetStatus(0);
int018;
Atom: Verdieping 2
从阅读Atom: Verdieping 2[1] = 4
Atom: Verdieping 2[2] = 5
Atom: Verdieping 2[3] = 6
Atom: Verdieping 2[4] = 20
Atom: Verdieping 2[5] = 21
Atom: Verdieping 2[6] = 33
Atom: Verdieping 2[7] = 139830296
Atom: Verdieping 2[8] = 189540320
Atom: Verdieping 2[9] = 169919424
Atom: Verdieping 2[10] = 100
Atom: Verdieping 2[11] = 100
Atom: Verdieping 2[12] = 100
Atom: Verdieping 2[13] = 15
Atom: Verdieping 2[14] = 5
Atom: Verdieping 2[15] = 20
Atom: Verdieping 2[16] = 1
Atom: Verdieping 2[17] = 1
Atom: Verdieping 2[18] = 0
Atom: Verdieping 1
注意:使用此方法,即使您有两个具有相同名称的部分,应用程序也只会收集它遇到的第一部分之间的信息。
谢谢, 我希望你觉得这很有帮助:)