Um eine Funktionsbibliothek zu verwenden muss zunächst Build-System vorbereitet werden. Dazu müssen entsprechende Compiler- und Linker-Flags gesetzt werden. Ohne diese Anpassungen wird die Kompilierung davon ausgehen, dass keine externen Funktionsbibliotheken verwendet werden. Die Folge ist meist ein "undefined reference"-Fehler.

Im CMake sind folgende Anpassungen notwendig:

  • Funktionsbibliothek über ein Find-Script auffinden lassen
  • An das Target des Programms, das die Funktionsbibliothek verwenden soll, folgende Informationen anfügen:
    • zusätzliches Include-Verzeichnis, in dem die Header der Funktionsbibliothek enthalten sind
    • Pfad zur Funktionsbibliothek
    • Kompiler-Flags, die für die Verwendung der Bibliothek notwendig sind

Hier ein Beispiel für CMake-Projekt, dass die "LibXml2" als Funktionsbibliothek einbettet:

CMakeLists.txt

cmake_minimum_required(VERSION 3.10)
project(demo C)

set(CMAKE_C_STANDARD 11)

find_package(LibXml2 REQUIRED)

add_executable(demo main.c)
target_include_directories(demo PRIVATE ${LIBXML2_INCLUDE_DIR})
target_link_libraries(demo PRIVATE ${LIBXML2_LIBRARIES})
target_compile_definitions(demo PRIVATE ${LIBXML2_DEFINITIONS})

Im Source Code des Programms können nun die Header-Dateien der Bibliotheken eingebunden, und die Funktionen, Typen und Variablen verwendet werden.

main.c

/**
 * section: Tree
 * synopsis: Navigates a tree to print element names
 * purpose: Parse a file to a tree, use xmlDocGetRootElement() to
 *          get the root element, then walk the document and print
 *          all the element name in document order.
 * usage: tree1 filename_or_URL
 * test: tree1 test2.xml > tree1.tmp && diff tree1.tmp $(srcdir)/tree1.res
 * author: Dodji Seketeli
 * copy: see Copyright for the status of this software.
 */
#include <stdio.h>
#include <libxml/parser.h>
#include <libxml/tree.h>

/*
 *To compile this file using gcc you can type
 *gcc `xml2-config --cflags --libs` -o xmlexample libxml2-example.c
 */

/**
 * print_element_names:
 * @a_node: the initial xml node to consider.
 *
 * Prints the names of the all the xml elements
 * that are siblings or children of a given xml node.
 */
static void
print_element_names(xmlNode * a_node)
{
    xmlNode *cur_node = NULL;

    for (cur_node = a_node; cur_node; cur_node = cur_node->next) {
        if (cur_node->type == XML_ELEMENT_NODE) {
            printf("node type: Element, name: %s\n", cur_node->name);
        }

        print_element_names(cur_node->children);
    }
}


/**
 * Simple example to parse a file called "file.xml",
 * walk down the DOM, and print the name of the
 * xml elements nodes.
 */
int
main(int argc, char **argv)
{
    xmlDoc *doc = NULL;
    xmlNode *root_element = NULL;

    if (argc != 2)
        return(1);

    /*
     * this initialize the library and check potential ABI mismatches
     * between the version it was compiled for and the actual shared
     * library used.
     */
    LIBXML_TEST_VERSION

    /*parse the file and get the DOM */
    doc = xmlReadFile(argv[1], NULL, 0);

    if (doc == NULL) {
        printf("error: could not parse file %s\n", argv[1]);
    }

    /*Get the root element node */
    root_element = xmlDocGetRootElement(doc);

    print_element_names(root_element);

    /*free the document */
    xmlFreeDoc(doc);

    /*
     *Free the global variables that may
     *have been allocated by the parser.
     */
    xmlCleanupParser();

    return 0;
}

Funktionsbibliotheken können selbst andere Funktionsbibliotheken verwenden. Das Einbinden einer Funktionsbibliothek sorgt dafür, dass alle Abhängigkeiten automatisch nachgezogen werden.

HINWEIS: Unter Linux reicht das Installieren der benötigten Funktionsbibliotheken meist nicht aus, da diese keine Header enthalten. Die Header sind separat in einem Development-Paket enthalten. Beispiel: libev4 enthält nur die Funktionsbibliothek. Die Header müssen über libev-dev nachinstalliert werden.

Zuletzt geändert: Freitag, 22. Juni 2018, 15:29