Optimize mxmlIndexFind().

Doco updates.
web
Michael R Sweet 20 years ago
parent 46085f9327
commit e6c90615c6
  1. 29
      doc/reference.html
  2. 159
      mxml-index.c
  3. 29
      mxml.xml

@ -307,7 +307,7 @@ mxmlFindElement(
<h3><a name='mxmlIndexDelete'>mxmlIndexDelete()</a></h3> <h3><a name='mxmlIndexDelete'>mxmlIndexDelete()</a></h3>
<hr noshade/> <hr noshade/>
<h4>Description</h4> <h4>Description</h4>
<p>Sort attribute...</p> <p>Delete an index.</p>
<h4>Syntax</h4> <h4>Syntax</h4>
<pre> <pre>
void void
@ -326,7 +326,9 @@ mxmlIndexDelete(
<h3><a name='mxmlIndexEnum'>mxmlIndexEnum()</a></h3> <h3><a name='mxmlIndexEnum'>mxmlIndexEnum()</a></h3>
<hr noshade/> <hr noshade/>
<h4>Description</h4> <h4>Description</h4>
<p>Return the next node in the index.</p> <p>Return the next node in the index.
Nodes are returned in the sorted order of the index.</p>
<h4>Syntax</h4> <h4>Syntax</h4>
<pre> <pre>
<a href='#mxml_node_t'>mxml_node_t</a> * <a href='#mxml_node_t'>mxml_node_t</a> *
@ -345,7 +347,12 @@ mxmlIndexEnum(
<h3><a name='mxmlIndexFind'>mxmlIndexFind()</a></h3> <h3><a name='mxmlIndexFind'>mxmlIndexFind()</a></h3>
<hr noshade/> <hr noshade/>
<h4>Description</h4> <h4>Description</h4>
<p>Find the next matching node.</p> <p>Find the next matching node.
You should call mxmlIndexReset() prior to using this function for
the first time with a particular set of &quot;element&quot; and &quot;value&quot;
strings. Passing NULL for both &quot;element&quot; and &quot;value&quot; is equivalent
to calling mxmlIndexEnum().</p>
<h4>Syntax</h4> <h4>Syntax</h4>
<pre> <pre>
<a href='#mxml_node_t'>mxml_node_t</a> * <a href='#mxml_node_t'>mxml_node_t</a> *
@ -368,7 +375,13 @@ mxmlIndexFind(
<h3><a name='mxmlIndexNew'>mxmlIndexNew()</a></h3> <h3><a name='mxmlIndexNew'>mxmlIndexNew()</a></h3>
<hr noshade/> <hr noshade/>
<h4>Description</h4> <h4>Description</h4>
<p>Create a new index.</p> <p>Create a new index.
The index will contain all nodes that contain the named element and/or
attribute. If both &quot;element&quot; and &quot;attr&quot; are NULL, then the index will
contain a sorted list of the elements in the node tree. Nodes are
sorted by element name and optionally by attribute value if the &quot;attr&quot;
argument is not NULL.</p>
<h4>Syntax</h4> <h4>Syntax</h4>
<pre> <pre>
<a href='#mxml_index_t'>mxml_index_t</a> * <a href='#mxml_index_t'>mxml_index_t</a> *
@ -391,7 +404,11 @@ mxmlIndexNew(
<h3><a name='mxmlIndexReset'>mxmlIndexReset()</a></h3> <h3><a name='mxmlIndexReset'>mxmlIndexReset()</a></h3>
<hr noshade/> <hr noshade/>
<h4>Description</h4> <h4>Description</h4>
<p>Return the first node in the index.</p> <p>Reset the enumeration/find pointer in the index and
return the first node in the index.
This function should be called prior to using mxmlIndexEnum() or
mxmlIndexFind() for the first time.</p>
<h4>Syntax</h4> <h4>Syntax</h4>
<pre> <pre>
<a href='#mxml_node_t'>mxml_node_t</a> * <a href='#mxml_node_t'>mxml_node_t</a> *
@ -1208,7 +1225,7 @@ static int num_callbacks = 1;
<h3><a name='sort_attr_name'>sort_attr_name</a></h3> <h3><a name='sort_attr_name'>sort_attr_name</a></h3>
<hr noshade/> <hr noshade/>
<h4>Description</h4> <h4>Description</h4>
<p>Delete an index.</p> <p>Current attribute to sort on</p>
<h4>Definition</h4> <h4>Definition</h4>
<pre> <pre>
static const char * sort_attr_name; static const char * sort_attr_name;

@ -1,5 +1,5 @@
/* /*
* "$Id: mxml-index.c,v 1.1 2004/05/16 05:25:38 mike Exp $" * "$Id: mxml-index.c,v 1.2 2004/05/16 13:09:44 mike Exp $"
* *
* Index support code for Mini-XML, a small XML-like file parsing library. * Index support code for Mini-XML, a small XML-like file parsing library.
* *
@ -21,7 +21,8 @@
* mxmlIndexEnum() - Return the next node in the index. * mxmlIndexEnum() - Return the next node in the index.
* mxmlIndexFind() - Find the next matching node. * mxmlIndexFind() - Find the next matching node.
* mxmlIndexNew() - Create a new index. * mxmlIndexNew() - Create a new index.
* mxmlIndexReset() - Return the first node in the index. * mxmlIndexReset() - Reset the enumeration/find pointer in the index and
* return the first node in the index.
* sort_attr() - Sort by attribute value... * sort_attr() - Sort by attribute value...
* sort_attr() - Sort by element name... * sort_attr() - Sort by element name...
* sort_element_attr() - Sort by element name and attribute value... * sort_element_attr() - Sort by element name and attribute value...
@ -39,6 +40,8 @@
* Sort functions... * Sort functions...
*/ */
static int index_compare(mxml_index_t *ind, const char *element,
const char *value, mxml_node_t *node);
static int sort_attr(const void *a, const void *b); static int sort_attr(const void *a, const void *b);
static int sort_element(const void *a, const void *b); static int sort_element(const void *a, const void *b);
static int sort_element_attr(const void *a, const void *b); static int sort_element_attr(const void *a, const void *b);
@ -48,7 +51,7 @@ static int sort_element_attr(const void *a, const void *b);
* Sort attribute... * Sort attribute...
*/ */
static const char *sort_attr_name; static const char *sort_attr_name; /* Current attribute to sort on */
/* /*
@ -81,6 +84,8 @@ mxmlIndexDelete(mxml_index_t *ind) /* I - Index to delete */
/* /*
* 'mxmlIndexEnum()' - Return the next node in the index. * 'mxmlIndexEnum()' - Return the next node in the index.
*
* Nodes are returned in the sorted order of the index.
*/ */
mxml_node_t * /* O - Next node or NULL if there is none */ mxml_node_t * /* O - Next node or NULL if there is none */
@ -106,6 +111,11 @@ mxmlIndexEnum(mxml_index_t *ind) /* I - Index to enumerate */
/* /*
* 'mxmlIndexFind()' - Find the next matching node. * 'mxmlIndexFind()' - Find the next matching node.
*
* You should call mxmlIndexReset() prior to using this function for
* the first time with a particular set of "element" and "value"
* strings. Passing NULL for both "element" and "value" is equivalent
* to calling mxmlIndexEnum().
*/ */
mxml_node_t * /* O - Node or NULL if none found */ mxml_node_t * /* O - Node or NULL if none found */
@ -113,7 +123,10 @@ mxmlIndexFind(mxml_index_t *ind, /* I - Index to search */
const char *element, /* I - Element name to find, if any */ const char *element, /* I - Element name to find, if any */
const char *value) /* I - Attribute value, if any */ const char *value) /* I - Attribute value, if any */
{ {
int i, j; /* Looping vars */ int diff, /* Difference between names */
current, /* Current entity in search */
first, /* First entity in search */
last; /* Last entity in search */
/* /*
@ -132,58 +145,85 @@ mxmlIndexFind(mxml_index_t *ind, /* I - Index to search */
return (mxmlIndexEnum(ind)); return (mxmlIndexEnum(ind));
/* /*
* Find the node... * If there are no nodes in the index, return NULL...
*/
if (!ind->num_nodes)
return (NULL);
/*
* If cur_node == 0, then find the first matching node...
*/ */
for (i = ind->cur_node; i < ind->num_nodes; i ++) if (ind->cur_node == 0)
{ {
/* /*
* Check the element name... * Find the first node using a modified binary search algorithm...
*/ */
if (element) first = 0;
last = ind->num_nodes - 1;
while (last > first)
{ {
if ((j = strcmp(element, ind->nodes[i]->value.element.name)) > 0) current = (first + last) / 2;
if ((diff = index_compare(ind, element, value, ind->nodes[current])) == 0)
{ {
/* /*
* No more elements <= name... * Found a match, move back to find the first...
*/ */
ind->cur_node = ind->num_nodes; while (current > 0 &&
!index_compare(ind, element, value, ind->nodes[current - 1]))
current --;
/*
* Return the first match and save the index to the next...
*/
return (NULL); ind->cur_node = current + 1;
return (ind->nodes[current]);
} }
else if (j < 0) else if (diff < 0)
continue; last = current;
else
first = current;
} }
/* /*
* Check the attribute value... * If we get this far, then we found exactly 0 or 1 matches...
*/ */
if (value) current = (first + last) / 2;
{ ind->cur_node = ind->num_nodes;
if ((j = strcmp(value, mxmlElementGetAttr(ind->nodes[i], ind->attr))) > 0)
{
/*
* No more attributes <= value...
*/
ind->cur_node = ind->num_nodes; if (!index_compare(ind, element, value, ind->nodes[current]))
{
/*
* Found exactly one match...
*/
return (NULL); return (ind->nodes[current]);
}
else if (j != 0)
continue;
} }
else
{
/*
* No matches...
*/
return (NULL);
}
}
else if (ind->cur_node < ind->num_nodes &&
!index_compare(ind, element, value, ind->nodes[ind->cur_node]))
{
/* /*
* Got a match, return it... * Return the next matching node...
*/ */
ind->cur_node = i + 1; return (ind->nodes[ind->cur_node ++]);
return (ind->nodes[i]);
} }
/* /*
@ -198,6 +238,12 @@ mxmlIndexFind(mxml_index_t *ind, /* I - Index to search */
/* /*
* 'mxmlIndexNew()' - Create a new index. * 'mxmlIndexNew()' - Create a new index.
*
* The index will contain all nodes that contain the named element and/or
* attribute. If both "element" and "attr" are NULL, then the index will
* contain a sorted list of the elements in the node tree. Nodes are
* sorted by element name and optionally by attribute value if the "attr"
* argument is not NULL.
*/ */
mxml_index_t * /* O - New index */ mxml_index_t * /* O - New index */
@ -306,7 +352,11 @@ mxmlIndexNew(mxml_node_t *node, /* I - XML node tree */
/* /*
* 'mxmlIndexReset()' - Return the first node in the index. * 'mxmlIndexReset()' - Reset the enumeration/find pointer in the index and
* return the first node in the index.
*
* This function should be called prior to using mxmlIndexEnum() or
* mxmlIndexFind() for the first time.
*/ */
mxml_node_t * /* O - First node or NULL if there is none */ mxml_node_t * /* O - First node or NULL if there is none */
@ -330,12 +380,53 @@ mxmlIndexReset(mxml_index_t *ind) /* I - Index to reset */
*/ */
if (ind->num_nodes) if (ind->num_nodes)
return (ind->nodes); return (ind->nodes[0]);
else else
return (NULL); return (NULL);
} }
/*
* 'index_compare()' - Compare a node with index values.
*/
static int /* O - Result of comparison */
index_compare(mxml_index_t *ind, /* I - Index */
const char *element, /* I - Element name or NULL */
const char *value, /* I - Attribute value or NULL */
mxml_node_t *node) /* I - Node */
{
int diff; /* Difference */
/*
* Check the element name...
*/
if (element)
{
if ((diff = strcmp(element, node->value.element.name)) != 0)
return (diff);
}
/*
* Check the attribute value...
*/
if (value)
{
if ((diff = strcmp(value, mxmlElementGetAttr(node, ind->attr))) != 0)
return (diff);
}
/*
* No difference, return 0...
*/
return (0);
}
/* /*
* 'sort_attr()' - Sort by attribute value... * 'sort_attr()' - Sort by attribute value...
*/ */
@ -383,5 +474,5 @@ sort_element_attr(const void *a, /* I - First node */
/* /*
* End of "$Id: mxml-index.c,v 1.1 2004/05/16 05:25:38 mike Exp $". * End of "$Id: mxml-index.c,v 1.2 2004/05/16 13:09:44 mike Exp $".
*/ */

@ -156,7 +156,7 @@ constrains the search to a particular node's children.</description>
</argument> </argument>
</function> </function>
<function name="mxmlIndexDelete"> <function name="mxmlIndexDelete">
<description>Sort attribute...</description> <description>Delete an index.</description>
<argument name="ind" direction="I"> <argument name="ind" direction="I">
<type>mxml_index_t *</type> <type>mxml_index_t *</type>
<description>Index to delete</description> <description>Index to delete</description>
@ -167,7 +167,9 @@ constrains the search to a particular node's children.</description>
<type>mxml_node_t *</type> <type>mxml_node_t *</type>
<description>Next node or NULL if there is none</description> <description>Next node or NULL if there is none</description>
</returnvalue> </returnvalue>
<description>Return the next node in the index.</description> <description>Return the next node in the index.
Nodes are returned in the sorted order of the index.</description>
<argument name="ind" direction="I"> <argument name="ind" direction="I">
<type>mxml_index_t *</type> <type>mxml_index_t *</type>
<description>Index to enumerate</description> <description>Index to enumerate</description>
@ -178,7 +180,12 @@ constrains the search to a particular node's children.</description>
<type>mxml_node_t *</type> <type>mxml_node_t *</type>
<description>Node or NULL if none found</description> <description>Node or NULL if none found</description>
</returnvalue> </returnvalue>
<description>Find the next matching node.</description> <description>Find the next matching node.
You should call mxmlIndexReset() prior to using this function for
the first time with a particular set of &quot;element&quot; and &quot;value&quot;
strings. Passing NULL for both &quot;element&quot; and &quot;value&quot; is equivalent
to calling mxmlIndexEnum().</description>
<argument name="ind" direction="I"> <argument name="ind" direction="I">
<type>mxml_index_t *</type> <type>mxml_index_t *</type>
<description>Index to search</description> <description>Index to search</description>
@ -197,7 +204,13 @@ constrains the search to a particular node's children.</description>
<type>mxml_index_t *</type> <type>mxml_index_t *</type>
<description>New index</description> <description>New index</description>
</returnvalue> </returnvalue>
<description>Create a new index.</description> <description>Create a new index.
The index will contain all nodes that contain the named element and/or
attribute. If both &quot;element&quot; and &quot;attr&quot; are NULL, then the index will
contain a sorted list of the elements in the node tree. Nodes are
sorted by element name and optionally by attribute value if the &quot;attr&quot;
argument is not NULL.</description>
<argument name="node" direction="I"> <argument name="node" direction="I">
<type>mxml_node_t *</type> <type>mxml_node_t *</type>
<description>XML node tree</description> <description>XML node tree</description>
@ -216,7 +229,11 @@ constrains the search to a particular node's children.</description>
<type>mxml_node_t *</type> <type>mxml_node_t *</type>
<description>First node or NULL if there is none</description> <description>First node or NULL if there is none</description>
</returnvalue> </returnvalue>
<description>Return the first node in the index.</description> <description>Reset the enumeration/find pointer in the index and
return the first node in the index.
This function should be called prior to using mxmlIndexEnum() or
mxmlIndexFind() for the first time.</description>
<argument name="ind" direction="I"> <argument name="ind" direction="I">
<type>mxml_index_t *</type> <type>mxml_index_t *</type>
<description>Index to reset</description> <description>Index to reset</description>
@ -826,6 +843,6 @@ the walk to the node's children.</description>
</variable> </variable>
<variable name="sort_attr_name"> <variable name="sort_attr_name">
<type>static const char *</type> <type>static const char *</type>
<description>Delete an index.</description> <description>Current attribute to sort on</description>
</variable> </variable>
</mxmldoc> </mxmldoc>

Loading…
Cancel
Save