我正在使用libxml2来编码XML文件中的数据。我的数据包含"<"之类的标签。和">"。当将其转换为XML时,这些标签也将转换为"& lt"one_answers"& gt"。有什么方法可以解决这个问题。我想在解码XML文件时将这些标签用作XML节点,因此CDATA不是解决此问题的解决方案。请为此提供任何解决方案。谢谢。
示例代码:
xmlNewChild(node, NULL, (xmlChar *)"ADDRESS", (xmlChar *)"<street>Park Street</street><city>kolkata</city>");
and output of above code is:
<person>
<ADDRESS><street>Park Street</street><city>Kolkata</city></ADDRESS>
如果您希望将字符串视为XML,则应使用XMLReadMemory将其解析并从中获得xmlDoc
。它可能适用于较大的字符串,但通常使用单步说明来构建文档,例如Joachim的答案。在这里,我提出 xmlAddChildFromString
函数以用字符串方式进行操作。
#include <stdio.h>
#include <string.h>
#include <libxml/parser.h>
#include <libxml/tree.h>
/// Returns 0 on failure, 1 otherwise
int xmlAddChildFromString(xmlNodePtr parent, xmlChar *newNodeStr)
{
int rv = 0;
xmlChar *newNodeStrWrapped = calloc(strlen(newNodeStr) + 10, 1);
if (!newNodeStrWrapped) return 0;
strcat(newNodeStrWrapped, "<a>");
strcat(newNodeStrWrapped, newNodeStr);
strcat(newNodeStrWrapped, "</a>");
xmlDocPtr newDoc = xmlReadMemory(
newNodeStrWrapped, strlen(newNodeStrWrapped),
NULL, NULL, 0);
free(newNodeStrWrapped);
if (!newDoc) return 0;
xmlNodePtr newNode = xmlDocCopyNode(
xmlDocGetRootElement(newDoc),
parent->doc,
1);
xmlFreeDoc(newDoc);
if (!newNode) return 0;
xmlNodePtr addedNode = xmlAddChildList(parent, newNode->children);
if (!addedNode) {
xmlFreeNode(newNode);
return 0;
}
newNode->children = NULL; // Thanks to milaniez
newNode->last = NULL; // for fixing
xmlFreeNode(newNode); // the memory leak.
return 1;
}
int
main(int argc, char **argv)
{
xmlDocPtr doc = xmlNewDoc(BAD_CAST "1.0");
xmlNodePtr root = xmlNewNode(NULL, BAD_CAST "root");
xmlDocSetRootElement(doc, root);
xmlAddChildFromString(root,
"<street>Park Street</street><city>kolkata</city>");
xmlDocDump(stdout, doc);
xmlFreeDoc(doc);
return(0);
}
您可以尝试使用函数xmlparseinnodecontext。它允许您在父节点的上下文中解析RAW XML,并构造一个可以连接到父的节点。
例如:
const char * xml = "<a><b><c>blah</c></b></a>";
xmlNodePtr new_node = NULL;
// we assume that 'parent' node is already defined
xmlParseInNodeContext(parent, xml, strlen(xml), 0, &new_node);
if (new_node) xmlAddChild(parent, new_node);
您必须在链条中调用xmlNewChild
,一个呼叫父节点的呼叫,每个子节点的呼叫:
xmlNodePtr *addressNode = xmlNewChild(node, NULL, (xmlChar *) "address", NULL);
xmlNewChild(addressNode, NULL, (xmlChar *) "street", "Park Street");
xmlNewChild(addressNode, NULL, (xmlChar *) "city", "Koltaka");
我现在正在使用以下代码将XML文本(可能包含多个元素)注入现有节点(感谢Nazar和Nellnhof的一个答案,并从我的问题中引用我(将字符串注入XML节点,而无需内容逃脱):
std::string xml = "<a>" + str + "</a>";
xmlNodePtr pNewNode = nullptr;
xmlParseInNodeContext(pParentNode, xml.c_str(), (int)xml.length(), 0, &pNewNode);
if (pNewNode != nullptr)
{
// add new xml node children to parent
xmlNode *pChild = pNewNode->children;
while (pChild != nullptr)
{
xmlAddChild(pParentNode, xmlCopyNode(pChild, 1));
pChild = pChild->next;
}
xmlFreeNode(pNewNode);
}
它采用字符串(str)添加一个周围元素(&lt; a> ...&lt; a/>),使用 xmlparseinnodecontext 解析字符串,然后添加新节点的孩子给父母。重要的是要添加新节点的孩子,而不是避免有LT的新节点;a> ...&lt;A/>在最终XML中。