IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

Langage PHP Discussion :

XML DomDocument optimisation de code


Sujet :

Langage PHP

  1. #1
    Nouveau membre du Club
    Homme Profil pro
    Développeur Web
    Inscrit en
    Décembre 2006
    Messages
    61
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 41
    Localisation : France, Indre et Loire (Centre)

    Informations professionnelles :
    Activité : Développeur Web
    Secteur : Communication - Médias

    Informations forums :
    Inscription : Décembre 2006
    Messages : 61
    Points : 39
    Points
    39
    Par défaut XML DomDocument optimisation de code
    Bonjour,

    J'ai un fichier xml à traiter qui possède plus de 29000 noeuds, et j'ai besoin de votre aide pour optimiser mon code qui le parcours.

    Pour le moment je peux traiter environ 14000 noeuds après quoi j'ai une erreur :
    Fatal error: Maximum execution time of 120 seconds exceeded....
    Voici le code que je souhaite optimiser

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
     
    // recuperation du nombre d'enregistrement
    $clients = $dom->getElementsByTagName("clients")->item(0);
    if ($clients->hasAttribute("nb"))
    {
    	$nbClient = $clients->getAttribute("nb");
    }
     
    for($i=0; $i<$nbClient; $i++)
    {
    	$listClient = $dom->getElementsByTagName('client')->item($i);
    	$arrSet = array();
    	$arrSet['tstamp'] = $time;
    	foreach($listClient->childNodes as $node)
    	{
    		if($node->nodeType == XML_ELEMENT_NODE)
                    {
    			$tagName = trim($node->tagName);
    			$nodeValue = trim($node->nodeValue);
    			if($tagName == 'datmod' or $tagName == 'datsup')
    			{
    				if(!empty($nodeValue)){
    					list($date, $Heure) = explode('T', $nodeValue);
    					list($annee, $mois, $jour) = explode('-', $date);
    					list($heure, $minute, $seconde) = explode(':', $Heure);
     
    					$nodeValue = mktime($heure, $minute, $seconde, $mois, $jour, $annee);
    				}
    				else{
    					$nodeValue = 0;
    				}
    			}
    			$arrSet[$tagName] = $nodeValue;
    		}
    	}
    	$this->Database->prepare("INSERT INTO `tl_ds_preregistration` %s")->set($arrSet)->execute();
    }

    Pour info j'ai fait des tests en mettant en commentaire le foreach et l'insert into et j'ai le même problème, je pense donc qu'il faut que je lise mon XML autrement sans faire de boucle.

    La ligne qui ne semble pas du tout être optimisé est celle ci :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    $listClient = $dom->getElementsByTagName('client')->item($i);
    Je vous joins un extrait du fichier XML
    Fichiers attachés Fichiers attachés

  2. #2
    Modératrice
    Avatar de Celira
    Femme Profil pro
    Développeuse PHP/Java
    Inscrit en
    Avril 2007
    Messages
    8 633
    Détails du profil
    Informations personnelles :
    Sexe : Femme
    Âge : 39
    Localisation : France

    Informations professionnelles :
    Activité : Développeuse PHP/Java
    Secteur : Industrie

    Informations forums :
    Inscription : Avril 2007
    Messages : 8 633
    Points : 16 372
    Points
    16 372
    Par défaut
    Le problème vient probablement du fait que $dom->getElementsByTagName('client') remonte tous les nodes <client>. Si tu as 29000 nodes <client>, ça fait un tableau de 29000 éléments à chaque tour de boucle... Tu peux commencer par sortir la lecture du xml de la boucle.
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
     
    $nodesClients = $dom->getElementsByTagName('client');
    for($i=0; $i<$nbClient; $i++)
    {
        $listClient = $nodesClients->item($i);
    Modératrice PHP
    Aucun navigateur ne propose d'extension boule-de-cristal : postez votre code et vos messages d'erreurs. (Rappel : "ça ne marche pas" n'est pas un message d'erreur)
    Cherchez un peu avant poser votre question : Cours et Tutoriels PHP - FAQ PHP - PDO une soupe et au lit !.

    Affichez votre code en couleurs : [CODE=php][/CODE] (bouton # de l'éditeur) et [C=php][/C]

  3. #3
    Nouveau membre du Club
    Homme Profil pro
    Développeur Web
    Inscrit en
    Décembre 2006
    Messages
    61
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 41
    Localisation : France, Indre et Loire (Centre)

    Informations professionnelles :
    Activité : Développeur Web
    Secteur : Communication - Médias

    Informations forums :
    Inscription : Décembre 2006
    Messages : 61
    Points : 39
    Points
    39
    Par défaut
    Bonjour,

    Ça aurai pu en effet être une solution et je te remercie de m'avoir répondu.

    Je suis du coup passé à simpleXML qui est bien plus rapide voici mon nouveau code :

    Code php : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
     
    $dom = new DomDocument();
    $dom->load($this->folderXml . DIRECTORY_SEPARATOR . $this->fileUser);
     
    if(!$dom->schemaValidate($this->folderXsd . DIRECTORY_SEPARATOR . $this->fileSchemaUser))
    {
        $doNotContinue = true;
    }
    if(!$doNotContinue)
    {
        $time = time();
        // Conversion élément DOM en SimpleXML => plus rapide
        $sxml = simplexml_import_dom($dom);
     
        // $sxml->results->clients['nb']; // Lecture attribut
     
        foreach ($sxml->xpath('//client') as $client) {
            $arrSet = array();
            $arrSet['tstamp'] = $time;
            $arrSet['code_cli'] = trim($client->code_cli);
            $arrSet['nom_cli'] = trim($client->nom_cli);
            $arrSet['codpos_cli'] = trim($client->codpos_cli);
            $arrSet['ville_cli'] = trim($client->ville_cli);
            $arrSet['code_site'] = trim($client->code_site);
            $arrSet['site'] = trim($client->site);
            $arrSet['datmod'] = $this->convertDateTime(trim($client->datmod));
            $arrSet['datsup'] = $this->convertDateTime(trim($client->datsup));
     
            $this->Database->prepare("INSERT INTO `tl_ds_preregistration` %s")->set($arrSet)->execute();
        }
        $this->log('L\'import utilisateur a été réalisé avec succès', 'dsAutomator importUser()', TL_CRON);
    }

    Avec ce code les 29000 noeuds sont traité en moins d'une seconde

+ Répondre à la discussion
Cette discussion est résolue.

Discussions similaires

  1. optimiser le code d'une fonction
    Par yanis97 dans le forum MS SQL Server
    Réponses: 1
    Dernier message: 15/07/2005, 08h41
  2. Optimiser mon code ASP/HTML
    Par ahage4x4 dans le forum ASP
    Réponses: 7
    Dernier message: 30/05/2005, 10h29
  3. optimiser le code
    Par bibi2607 dans le forum ASP
    Réponses: 3
    Dernier message: 03/02/2005, 14h30
  4. syntaxe et optimisation de codes
    Par elitol dans le forum Langage SQL
    Réponses: 18
    Dernier message: 12/08/2004, 11h54
  5. optimisation du code et var globales
    Par tigrou2405 dans le forum ASP
    Réponses: 2
    Dernier message: 23/01/2004, 10h59

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo