- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在构建一个简单的自上而下的基于图 block 的 2D 游戏,并且我正在尝试解析 Tiled Map Editor(.tmx 文件)的输出。对于那些不熟悉的人来说,TMX 文件是 XML 文件,它使用图像中重复使用的图 block 层来描述游戏 map 。以前除了解析简单文本外,我从未接触过任何东西,我想知道,对于一个相当简单的 XML 文件,使用 LINQ 是否是执行此操作的最合适方法。
这是一个删节的 .tmx 文件:
<?xml version="1.0" encoding="UTF-8"?>
<map width="100" height="100" tilewidth="16" tileheight="16">
<tileset>
<!-- This stuff in here is mostly metadata that the map editor uses -->
</tileset>
<layer name="Background" width="100" height="100">
<data>
<tile gid="1" />
<tile gid="2" />
<tile gid="3" />
<tile gid="1" />
<tile gid="1" />
<tile gid="1" />
<!-- etc... -->
</data>
</layer>
<layer name="Foreground" width="100" height="100">
<data>
<!-- gid="0" means don't load a tile there. It should be an empty cell for that layer (and the layers beneath this are visible) -->
<tile gid="0" />
<tile gid="4" />
<!-- etc. -->
</data>
</layer>
<!-- More layers.... -->
</map>
如您所见,它非常简单(请注意,每一层中的每个图 block (100x100) 都有一个“图 block ”元素)。现在在我看来,LINQ 的目的是从可能非常大且几乎类似于数据库的 xml 文件中获取非常具体的数据,而您实际上并不需要整个文件。我在这里要做的主要是遍历每个“tile”元素的 gid 并将其插入到一个数组中,该数组表示我的应用程序中的 map 。
这是我处理图层的代码:
public void AddLayer(XElement layerElement) {
TileMapLayer layer = new TileMapLayer(Convert.ToInt32(layerElement.Attribute("width")), Convert.ToInt32(layerElement.Attribute("height")));
layer.Name = (string)layerElement.Attribute("name");
layer.Opacity = Convert.ToDouble(layerElement.Attribute("opacity"));
layer.Visible = Convert.ToInt32(layerElement.Attribute("visible")) == 1;
if (layerElement.HasElements)
{
XElement data = layerElement.Element("data");
foreach (XElement tile in data.Elements())
{
layer.NextTile(Convert.ToInt32(tile.Attribute("gid")));
}
}
this.layers.Add(layer);
}
为了让我的问题更简洁:当我遍历并关心每条数据时(即我正在遍历并按顺序获取每个节点的所有子元素的数据),是否使用 LINQ to XML给我什么好处?比如 LINQ to XML 库的性能是否更好?我对 LINQ 的不熟悉是否会阻止我看到一种有效的方法来做我想做的事情?等等?或者我真的应该使用不同的 XML 实用程序吗?
最佳答案
对于简单的检索和更新数据,LINQ to XML 似乎是 XML 解析的方钉/圆孔解决方案。 XElement 当然可以递归迭代,但我发现 XPath 查询更加简洁易读。
我发现 LINQ to XML 非常有用的地方是在处理文档 namespace 方面。 .NET 提供的其他框架并没有以优雅的方式处理这个问题。根据http://msdn.microsoft.com/en-us/library/ecf3e2k0.aspx :
Changing the prefix of a node does not change its namespace. The namespace can only be set when the node is created. When you persist the tree, new namespace attributes may be persisted out to satisfy the prefix you set. If the new namespace cannot be created, then the prefix is changed so the node preserves its local name and namespace.
诚然,这是一个深奥的要求,但我发现 LINQ to XML 在为 XPath 查询准备命名空间管理器方面也很有用。
public XmlNamespaceManager NamespaceManager { get; set; }
public XPathNavigator Navigator { get; set; }
public SuperDuperXMLQueryingClass(System.IO.Stream stream)
{
var namespaces = RetrieveNameSpaceMapFromXml(XDocument.Load(stream).Root);
Navigator = new XPathDocument(stream).CreateNavigator();
NamespaceManager = new XmlNamespaceManager(Navigator.NameTable);
foreach (var t in namespaces)
{
NamespaceManager.AddNamespace(t.Key, t.Value.NamespaceName);
}
}
// LINQ to XML mostly used here.
private static Dictionary<string, XNamespace> RetrieveNamespaceMapFromXDocumentRoot(XElement root)
{
if (root == null)
{ throw new ArgumentNullException("root"); }
return root.Attributes().Where(a => a.IsNamespaceDeclaration)
.GroupBy(a => (
a.Name.Namespace == XNamespace.None ? String.Empty : a.Name.LocalName),
a => XNamespace.Get(a.Value)
)
.Where(g => g.Key != string.Empty)
.ToDictionary(g => g.Key, g => g.First());
}
public string DeliverFirstValueFromXPathQuery(string qry)
{
try
{
var iter = QueryXPathNavigatorUsingShortNamespaces(qry).GetEnumerator();
iter.MoveNext();
return iter.Current == null ? string.Empty : iter.Current.ToString();
}
catch (InvalidOperationException ex)
{
return "";
}
}
这使您可以选择使用 XPath 查询 XML 文档,而无需使用完整的 URI。
//For Example
DeliverFirstValueFromXPathQuery("/ns0:MyRoot/ns1:MySub/nsn:ValueHolder");
这里的总结是每个框架都有一些重叠的功能;但是,有些比其他的更适合用于专业工作。
关于c# - 将 LINQ to XML 用于简单的 XML 文件 : overkill? 还是更糟?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18746925/
我有一个功能是转换 ADO Recordset 进入html: class function RecordsetToHtml(const rs: _Recordset): WideString; 该函
经过几天的研究和讨论,我想出了这种方法来收集访客的熵(你可以看到我的研究历史here) 当用户访问时,我运行此代码: $entropy=sha1(microtime().$pepper.$_SERVE
给定一个无序列表 List ,我需要查找是否存在 String与提供的字符串匹配。 所以,我循环 for (String k : keys) { if (Utils.keysM
我已经搜索过这个问题,但没有找到我正在寻找的答案。 基本上,我想将类构造函数包装在 try/except 子句中,以便它忽略构造函数内特定类型的错误(但无论如何都会记录并打印它们)。我发现做到这一点的
我有一组三个数字,我想将一组数字与另一组数字进行比较。即,第一组中的每个数字小于另一组中的至少一个数字。需要注意的是,第一组中的下一个数字必须小于第二组中的不同数字(即,{6,1,6} 对 {8,8,
关闭。这个问题是off-topic .它目前不接受答案。 想改进这个问题吗? Update the question所以它是on-topic用于堆栈溢出。 关闭 9 年前。 Improve this
首先介绍一下背景: 我正在开发一个带有 EJB 模块和应用程序客户端模块的企业应用程序 (ear)。我还使用 hibernate JPA 来实现持久性,并使用 swingx 来实现 GUI。这些是唯一
我正在尝试在我的上网本上运行 Eclipse 以便能够为 Android 进行开发。 您可能已经猜到了,Eclipse 非常慢,并且不容易有效地开发。 我正在使用 Linux Ubuntu 并且我还有
for row, instrument in enumerate(instruments): for col, value in enumerate(instrument):
return not a and not b ^ 我如何以更好的格式表达它 最佳答案 DeMorgan's Law , 也许? return not (a or b) 我认为在这一点上已经足够简单了
我正在尝试让 Font Awesome 图标看起来更 slim https://jsfiddle.net/cliffeee/7L6ehw9r/1/ . 我尝试使用“-webkit-text-strok
假设我有一个名为 vals 的数据框,如下所示: id…………日期…………min_date…… .........最大日期 1…………2016/01/01…………2017/01/01…………2018/
是否有更 Pythonic 的方式来做到这一点?: if self.name2info[name]['prereqs'] is None: se
我有一个函数可以将一些文本打印到它接收到的 ostream&。如果 ostream 以终端为目标,我想让它适应终端宽度,否则默认为某个值。 我现在做的是: 从 ostream 中获取一个 ofstre
这个问题在这里已经有了答案: Should a retrieval method return 'null' or throw an exception when it can't produce
我有这个 bc = 'off' if c.page == 'blog': bc = 'on' print(bc) 有没有更 Pythonic(和/或更短)的方式在 Python 中编写? 最佳
输入:一个包含 50,000 行的 CSV;每行包含 910 列值 0/1。 输出:运行我的 CNN 的数据框。 我编写了一个逐行读取 CSV 的代码。对于每一行,我将数据分成两部分,称为神经元(90
据我所知,with block 会在您退出 block 后自动调用 close(),并且它通常用于确保不会忘记关闭一个文件。 好像没有技术上的区别 with open(file, 'r+') as f
我有一个使用 Entity Framework V6.1.1 的 MVC 5 网站。 Entity Framework DbContext 类和模型最初都在网站项目中。这个项目有 3 个 DbCont
我是编程新手,在尝试通过将 tableView 和关联 View 的创建移动到单独的类并将委托(delegate)和数据源从 VC 移动到单独的类来精简我的 ViewController 时遇到了一些
我是一名优秀的程序员,十分优秀!