我从我的 Silverlight 应用程序中获得了以下代码片段:
var messages = from message in XcurrentMsg.Descendants("message")
where DateTime.Parse(message.Attribute("timestamp").Value).CompareTo(DateTime.Parse(MessageCache.Last_Cached())) > 0
select new
{
ip = message.Attribute("ip").Value,
timestamp = message.Attribute("timestamp").Value,
text = message.Value,
};
if (messages == null)
throw new SystemException("No new messages recorded. Application tried to access non existing resources!");
foreach (var message in messages)
{
XElement temporaryElement = new XElement("message", message.text.ToString(), new XAttribute("ip", message.ip.ToString()), new XAttribute("timestamp", message.timestamp.ToString()));
XcurrentMsg.Element("root").Element("messages").Add(temporaryElement);
AddMessage(BuildMessage(message.ip, message.timestamp, message.text));
msgCount++;
}
MessageCache.CacheXML(XcurrentMsg);
MessageCache.Refresh();
XcurrentMsg 是从我的服务器获取的包含消息的 XDocument: 结构
<root>
<messages>
<message ip="" timestamp=""> Text </message>
</messages>
</root>
我想获得比上次缓存 XcurrentMsg 时更新的所有“消息”。 只要我删除“XElement temporaryElement”和“XcurrentMsg.Element ....”,这就可以正常工作。 并简单地使用 currentMsg 字符串作为输出。 但我想将“新消息”保存在我的 XcurrentMsg/缓存中。 现在,如果我不删除这部分内容,我的应用程序就会变得非常疯狂。 我认为它会不停地向 XcurrentMsg 写入无限元素。
我不知道是什么问题。
问候,
最佳答案
这是一个经典的 LINQ 问题。
变量messages
持有对 IEnumerable<someAnonymousType>
的引用.您犯的错误是假设在分配给 messages
之后所有Descendents
已被枚举和 someAnonymousType
的集合已经建成。
实际上此时什么都没有发生。只有当您开始使用 foreach 枚举集合时,才会出现 Descendents
。枚举并创建匿名类型。连点全套Descendents
没有被枚举,事实上直到第一个满足 where
的项目才被枚举。子句条件,然后在 foreach 循环时一次只创建一个投影(选择的结果)。
因此,您添加的消息也包含在枚举中,并且因为这会引发其他消息,您会以无限循环结束,或者至少会出现错误,提示您试图修改正在枚举的集合。
如果您想在开始循环之前确保要枚举的项目列表是固定的,请添加 ToList()
对于查询,这会创建一个 List<T>
.
var messages = (from message in XcurrentMsg.Descendants("message")
where DateTime.Parse(message.Attribute("timestamp").Value).CompareTo(DateTime.Parse(MessageCache.Last_Cached())) > 0
select new
{
ip = message.Attribute("ip").Value,
timestamp = message.Attribute("timestamp").Value,
text = message.Value,
}).ToList();
注意不用测试messages
对于 null 在任何一种情况下,它都不会为 null,它可能是一个空的枚举或列表。
关于c# - 操纵现有的 XDocument(失败),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2856426/