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

VB.NET Discussion :

Problème avec System.OutOfMemoryException


Sujet :

VB.NET

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    Membre éclairé
    Profil pro
    Inscrit en
    Février 2006
    Messages
    505
    Détails du profil
    Informations personnelles :
    Localisation : Canada

    Informations forums :
    Inscription : Février 2006
    Messages : 505
    Par défaut Problème avec System.OutOfMemoryException
    Bonjour à tous,

    Voilà, j'utilise une BD Access local pour effectuer des opérations d'analyses. j'ajoute à cette BD environ 4000 enregistrements chaque jour. Lors de mes analyses je récupère l'ensemble de ma BD dans un DataSet pour ensuite faire mes opérations d'analyse sur l'ensemble des mes enregistrements.

    Depuis une semaine je ne suis plus capable de faire l'analyse sur l'ensemble des enregistrments car l'exception System.OutOfMemoryException apparaità environ 90% du travail fait. Voici le message reçu.

    L'exception System.OutOfMemoryException n'a pas été gérée
    Message=Une exception de type 'System.OutOfMemoryException' a été levée.
    Source=System.Data
    StackTrace:
    à System.Data.Common.DoubleStorage.SetCapacity(Int32 capacity)
    à System.Data.RecordManager.set_RecordCapacity(Int32 value)
    à System.Data.RecordManager.GrowRecordCapacity()
    à System.Data.RecordManager.NewRecordBase()
    à System.Data.DataTable.NewRecord(Int32 sourceRecord)
    à System.Data.DataRow.BeginEditInternal()
    à System.Data.DataRow.set_Item(DataColumn column, Object value)
    à System.Data.DataRow.set_Item(Int32 columnIndex, Object value)
    à BourseV3.ScanMarche.RemplirDatatable(String symb) dans C:\Users\Mario\Documents\Visual Studio 2010\Projects\BourseV3\BourseV3\ScanMarche.vb:ligne 551
    à BourseV3.ScanMarche.DoTheTask() dans C:\Users\Mario\Documents\Visual Studio 2010\Projects\BourseV3\BourseV3\ScanMarche.vb:ligne 359
    à System.Threading.ThreadHelper.ThreadStart_Context(Object state)
    à System.Threading.ExecutionContext.runTryCode(Object userData)
    à System.Runtime.CompilerServices.RuntimeHelpers.ExecuteCodeWithGuaranteedCleanup(TryCode code, CleanupCode backoutCode, Object userData)
    à System.Threading.ExecutionContext.RunInternal(ExecutionContext executionContext, ContextCallback callback, Object state)
    à System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean ignoreSyncCtx)
    à System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state)
    à System.Threading.ThreadHelper.ThreadStart()
    InnerException:
    J'imagine que je dois libérer de la memoire, j'ai essayé de vider mes datatables après chaque analyse mais ça fonctionne pas. Alors je suis un peu perdu.

    Le bout de code qui suis est utilisé pour charger mes datatables une après l'autre en faire l'analyse et détruire les datatables.

    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
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    'Action des threads
        Sub DoTheTask()
            'boucle pour modifier les critères d'analyses 
            SyncLock Lock
                Do Until QuSymbole.Count = 0
                    'Création de la Datatable
                    Dim SymbA As String = String.Empty
                    Dim Sq As New Dictio
                    Dim Th As String = Thread.CurrentThread.Name
                    Dim Doubl As Boolean = False
                    SymbA = QuSymbole.Dequeue
                    For Each DtTest As DataTable In ObjDataSet.Tables
                        If DtTest.TableName = SymbA Then
                            Doubl = True
                            DtTest.Clear()
                        End If
                    Next
                    If Not Doubl = True Then
                        Sq.Create(Th, SymbA)
                        ObjDataSet.Tables.Add(Sq(Th))
                        'Formation des columns
                        MiseEnForme(ObjDataSet.Tables(SymbA))
                    End If
                    'Ajouter les données
                    RemplirDatatable(SymbA)
                    'Analyser le Datatable
                    If ObjDataSet.Tables(SymbA).Rows.Count > 0 Then
                        Dim RCount As Integer = ObjDataSet.Tables(SymbA).Rows.Count - 1
                        'Calcul de tendences
                        Dim Tend30 As Decimal = TendencePrix(30, SymbA, ObjDataSet.Tables(SymbA))
                        Dim Tend180 As Decimal = TendencePrix(180, SymbA, ObjDataSet.Tables(SymbA))
                        Dim Tend360 As Decimal = TendencePrix(360, SymbA, ObjDataSet.Tables(SymbA))
                        'Calcule de la variation entre aujourd'hui et hier
                        Dim Var As Double = ObjDataSet.Tables(SymbA).Rows(RCount).Item(9) - ObjDataSet.Tables(SymbA).Rows(RCount - 1).Item(9)
                        Dim VarPC As Double = Var / ObjDataSet.Tables(SymbA).Rows(RCount - 1).Item(9)
                        If ObjDataSet.Tables(SymbA).Rows(RCount).Item(0) = ALDate.Item(ALDate.Count - 1) Then
                            Dim MaStr As String = (ObjDataSet.Tables(SymbA).Rows(RCount).Item(0) & ";" & ObjDataSet.Tables(SymbA).Rows(RCount).Item(1) & ";" & ObjDataSet.Tables(SymbA).Rows(RCount).Item(2) & ";" & ObjDataSet.Tables(SymbA).Rows(RCount).Item(3) & ";" & ObjDataSet.Tables(SymbA).Rows(RCount).Item(5) & ";" & ObjDataSet.Tables(SymbA).Rows(RCount).Item(6) & ";" & ObjDataSet.Tables(SymbA).Rows(RCount).Item(7) & ";" & ObjDataSet.Tables(SymbA).Rows(RCount).Item(8) & ";" & ObjDataSet.Tables(SymbA).Rows(RCount).Item(9) & ";" & Var & ";" & VarPC & ";" & ObjDataSet.Tables(SymbA).Rows(RCount).Item(10) & ";" & Tend30 & ";" & Tend180 & ";" & Tend360)
                            DelListVScan(MaStr)
                            AnalyseSymb(ObjDataSet.Tables(SymbA))
                        End If
                    End If
     
                    'ObjDataSet.Tables(SymbA).Rows.Clear()
                    ObjDataSet.Tables(SymbA).Clear()
     
                Loop
            End SyncLock
        End Sub
    Vos suggestions sont les bienvenus pour régler ou pour améliorer mon code.

    Merci de votre soutien

    Mario

  2. #2
    Expert éminent Avatar de Pol63
    Homme Profil pro
    .NET / SQL SERVER
    Inscrit en
    Avril 2007
    Messages
    14 197
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Puy de Dôme (Auvergne)

    Informations professionnelles :
    Activité : .NET / SQL SERVER

    Informations forums :
    Inscription : Avril 2007
    Messages : 14 197
    Par défaut
    il faut appeler dispose pour détruire les objets qui ont la méthode dispose

    travailler sur des dataset/datatable n'est pas ce qu'il y a de plus performant et c'est gourmand en mémoire

    il faudrait savoir ce que fait l'analyse, bien souvent une requete peut faire le travail sans ramener les données et les parcourir

    si tu veux savoir où est la fuite de mémoire, il y a des profiler .net qui peuvent te dire à un instant T de quoi est composé la ram prise par l'appli (par type)
    Cours complets, tutos et autres FAQ ici : C# - VB.NET

  3. #3
    Membre éclairé
    Profil pro
    Inscrit en
    Février 2006
    Messages
    505
    Détails du profil
    Informations personnelles :
    Localisation : Canada

    Informations forums :
    Inscription : Février 2006
    Messages : 505
    Par défaut
    Oui Pol63, tu réponds exactement à ce que j'avais besoin.
    j'essai avec Dispose, et je fouille pour trouver la façon de connaitre l'état de la mémoire RAM (Mémoire totale & mémoire utilisée)

    merci je reviens après les tests

  4. #4
    Membre éclairé
    Profil pro
    Inscrit en
    Février 2006
    Messages
    505
    Détails du profil
    Informations personnelles :
    Localisation : Canada

    Informations forums :
    Inscription : Février 2006
    Messages : 505
    Par défaut
    Zut, ça marche pas

    Je te remercie pour ta suggestion Pol63, malheureusement une requète ne pourrait être utile car je fait plusieur opération lors de l'analyse genre ( moyenne, tendence, etc). Avec mes recherches je croyais qu'il était avantageux de passer par un dataset car je n'avais pas besoin d'ouvrir 4000 fois ma connection avec ma BD.

    Ce que j'ai lu est que l'avantage de faire ainsi est que j'ai besoin une fois d'ouvrir ma connection BD pour charger mon DataSet pour ensuite travailler sur ce dataset.

    Ainsi, que me suggères-tu pour améliorer mon programme

    merci

  5. #5
    Membre Expert

    Homme Profil pro
    Développeur .NET
    Inscrit en
    Novembre 2010
    Messages
    2 067
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Rhône (Rhône Alpes)

    Informations professionnelles :
    Activité : Développeur .NET

    Informations forums :
    Inscription : Novembre 2010
    Messages : 2 067
    Par défaut
    une liste des données qui sont utile a ton calcul, si par exemple tu as 80 colonnes dans une table et que t'en as besoin que de 10 pour effectuer tes calculs tu récupèrent que ces 10 colonnes et tu crées une liste de ces données

  6. #6
    Membre éclairé
    Profil pro
    Inscrit en
    Février 2006
    Messages
    505
    Détails du profil
    Informations personnelles :
    Localisation : Canada

    Informations forums :
    Inscription : Février 2006
    Messages : 505
    Par défaut
    oui merci c'est une solution,

    j'aimerais bien identifier par contre ce qui manque à mon code pour pouvoir tourner en libérant la mémoire qui n'est plus utile. La memoire virtuelle au bébut de ma boucle est à 957*779*968 et j'ai un arret au lorsqu'elle arrive à 261*287*936.

  7. #7
    Expert éminent Avatar de Pol63
    Homme Profil pro
    .NET / SQL SERVER
    Inscrit en
    Avril 2007
    Messages
    14 197
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Puy de Dôme (Auvergne)

    Informations professionnelles :
    Activité : .NET / SQL SERVER

    Informations forums :
    Inscription : Avril 2007
    Messages : 14 197
    Par défaut
    Citation Envoyé par Mario Rousson Voir le message
    Je te remercie pour ta suggestion Pol63, malheureusement une requète ne pourrait être utile car je fais plusieurs opérations lors de l'analyse genre ( moyenne, tendence, etc)
    sur access je veux bien croire, c'est limité, mais une requete ce n'est pas forcément qu'un simple select, ca peut faire plusieurs dizaines de ligne de code et faire des calculs dans tous les sens, des regroupements, des agrégats ...

    Citation Envoyé par Mario Rousson Voir le message
    Avec mes recherches je croyais qu'il était avantageux de passer par un dataset car je n'avais pas besoin d'ouvrir 4000 fois ma connection avec ma BD.
    et quel serait l'utilité de n'ouvrir la connexion qu'une fois ?

    en rapatriant tout, tu peux modifier de la facon suivante (POO standard) :
    tu fais une classe avec en propriétés les colonnes qui te sont utiles
    tu créé un dictionary(of long, taclasse)
    tu fais un select lescolonnes from latable que tu parcours via un oledbdatareader pour remplir la collecion de classe
    à partir de là ton code de calcul devrait être à peu le même, mais sans l'outofmemory
    si tu as des jointures, il faut plusieurs classes, et soit avoir des propriétés qui sont du type de l'autre classe, soit avoir un autre dico
    Cours complets, tutos et autres FAQ ici : C# - VB.NET

  8. #8
    Membre éclairé
    Profil pro
    Inscrit en
    Février 2006
    Messages
    505
    Détails du profil
    Informations personnelles :
    Localisation : Canada

    Informations forums :
    Inscription : Février 2006
    Messages : 505
    Par défaut
    Je te remercie beaucoup Pol63 c'est toujours intéressant quand un connaisseur nous donne son opinion sur la meilleur façon de coder, je prend note de ta suggestion pour ma prochaine version car je crois avoir résolu mon problème sans trop de modification.

    j'ai ajouté ceci à la fin de chaque boucle et ma memoire virtuelle ne varie beaucoup moins.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    ObjDataSet.Tables(SymbA).Rows.Clear()
                    ObjDataSet.Tables(SymbA).Dispose()
                    ObjDataSet.Tables(SymbA).Reset()
    Cela dit, je doute pas qu'utiliser un Dictionary comme tu l'a décris est beaucoup mieux que le Dataset, mais pour l'instant point de vue vitesse d'exécution et stabilité du programme me satisfont.

    merci beaucoup
    Mario

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

Discussions similaires

  1. Problème avec system()
    Par cmoibal dans le forum Bibliothèque standard
    Réponses: 1
    Dernier message: 08/01/2008, 19h36
  2. Problème avec System.CodeDom
    Par spidey87 dans le forum Windows Forms
    Réponses: 4
    Dernier message: 20/11/2007, 09h45
  3. Problème avec System.Data.Odbc
    Par Chii-san dans le forum VB.NET
    Réponses: 2
    Dernier message: 14/08/2007, 14h41
  4. [Système] Problème avec system()
    Par bartrik dans le forum Langage
    Réponses: 2
    Dernier message: 07/06/2007, 10h46
  5. problème avec System.management
    Par arabimouh dans le forum Windows Mobile
    Réponses: 2
    Dernier message: 24/05/2007, 16h50

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