Windows Forms应用程序用于读取和写入非常大的XML文件,但它是悬挂的

时间:2013-01-25 05:58:48

标签: c# xml winforms

我尝试开发一个Windows Forms应用程序来读取和编写一个非常大的XML文件。我有两个按钮,1)用于编写XML文件,**和2)读取XML文件。 当我点击写入XML按钮时,我的winForm应用程序正在挂起。写入XML文件时不允许执行任何其他操作,但我想读取和写入相同的XML文件。

阅读

void btnReading_Click(object sender, EventArgs e)
{
    strXpathQuery = "/AppXmlLogWritter/LogData[substring(LogDateTime, 1, 8)  >='" +
                    dateTimePickerFromDate.Value.ToString("yyyyMMdd") +
                    "' and substring(LogDateTime, 1, 8)  <='" +
                    dateTimePickerToDate.Value.ToString("yyyyMMdd") +
                    "']";
    XmlElement objXmlRoot = null;
    XmlNodeList objxmlNodeList = objXmlRoot.SelectNodes(strXpathQuery);
}

void   BindData(objxmlNodeList);
{
    BindData(XmlNodeList objxmlNodeList)
    DataTable dataTable = new DataTable();
    dataTable = XmlNodeListToDataTable(objxmlNodeList, new string[] { "LogID", "LogDateTime"});

    lstViewInfo.View = View.Details;
    lstViewInfo.Clear();
    lstViewInfo.Columns.Add("LogID", Convert.ToInt32(lstViewInfo.Width * 0.20));
    lstViewInfo.Columns.Add("LogDateTime", Convert.ToInt32(lstViewInfo.Width * 0.20));
    ListViewItem objListViewitem = null;
    for (int i = 0; i < dataTable.Rows.Count; i++)
    {
        objListViewitem = new ListViewItem();
        objListViewitem.Text = dataTable.Rows[i]["LogID"].ToString();
        objListViewitem.SubItems.Add(dataTable.Rows[i]["LogDateTime"].ToString());
        lstViewInfo.Items.Add(objListViewitem);
    }
}

写作

void button1_Click(object sender, EventArgs e)
{
    Mutex objMutex = new Mutex(false, @"Global\MySharedLog");
    XmlDocument xmlDoc = new XmlDocument();
    string currentDateTime = DateTime.Now.ToString("yyyyMMddHHmmss");
    XmlElement newelement = xmlDoc.CreateElement("LogData");
    XmlElement xmlLogID = xmlDoc.CreateElement("LogID");
    XmlElement xmlLogDateTime = xmlDoc.CreateElement("LogDateTime");

    int randomNumber = random.Next(9999);
    xmlLogID.InnerText = _logIDPrefix + currentDateTime + DateTime.UtcNow.Ticks + randomNumber;
    xmlLogDateTime.InnerText = currentDateTime;

    newelement.AppendChild(xmlLogID);
    newelement.AppendChild(xmlLogDateTime);

    try
    {
        objMutex.WaitOne();
        if (!File.Exists(_logFilePath))
        {
            File.WriteAllText(
              _logFilePath,
              "<?xml version='1.0' encoding='utf-8' standalone='yes'?>\r\n<AppXmlLogWritter><objMutex></objMutex></AppXmlLogWritter>");
        }

        using (FileStream fileStream = new FileStream(_logFilePath,
               FileMode.OpenOrCreate,
               FileAccess.ReadWrite,
               FileShare.ReadWrite))
        {
            xmlDoc.Load(fileStream);
            xmlDoc.DocumentElement.AppendChild(newelement);
            fileStream.SetLength(0);
            xmlDoc.Save(fileStream);
        }
    }
    finally
    {
        objMutex.ReleaseMutex();
    }
 }

4 个答案:

答案 0 :(得分:2)

应用程序挂起,因为您使用UI线程完成所有工作,而不是将其移动到单独的线程。

这需要时间(作为操作)并且只要UI线程忙,它就无法处理重要的消息以重绘,移动,对鼠标做出反应等。

后台工作者,在处理程序的.NET 4.5异步方法中,将立即修复它。

答案 1 :(得分:0)

应用程序正在挂起,因为它正在尝试打开该文件。你在“XmlNodeListToDataTable”中使用了什么方法? MSXML?如果是这样,请尝试使用XPath,它会更快并且会更快地加载XML。此外,“XmlNodeListToDataTable”可以在一个单独的线程中完成,以便您的GUI在需要时仍然可以响应。

答案 2 :(得分:0)

是的,因为您用来编写的方法XmlNodeListToDataTable()绑定整个xml节点。您创建行,然后添加coloumns,然后将值添加到相应的coloumns。 想象一下,如果XML文件的文件大小是1 GB。然后有大量的XML节点,因此将整个XML数据绑定到数据表需要花费大量的时间。

因此,更好的方法是在数据集中获取XML文件并显示,您可以遍历数据集中的数据表。具体而言,与将XML绑定到数据表相比,它花费的时间更短。

答案 3 :(得分:0)

只是开箱即用......你也可以试试Xsd2Code。它是一个开源工具,甚至我们在我们的项目中使用。它易于维护和管理。即使我们处理大型XML文件。

关于Windows窗体挂起,首先,使用XPath可能会使您的操作更快。关于挂起,在新线程中操作是我能看到的唯一希望。当您编写XML文件时,您的主窗口线程正忙。线程的骨架代码可能是:

private void WriteXML_Click(object sender, EventArgs e)
{
    Thread t = new Thread(new ThreadStart(WriteXMLFile));
    t.Start();
}

private void WriteXMLFile()
{
    // Write an XML file
}

此代码只是使用线程的示例代码。这些postsposts也可以提供帮助。