Problème VBA
Résolu
dubolg
-
dubolg -
dubolg -
Bonjour,
Je suis à la recherche d'une idée pour résoudre un problème simple (sauf pour moi)
j'ai sous excel une suite de numéro :
10
10
12
14
16
16
16
16
18
20
...
J'aimerais dans un premier temps pouvoir garder tous les nombres en double et supprimer ceux qui son seul. Et dans un deuxième temps,en repartant de la même liste faire l'inverse, c'est à dire garder les nombres seul et supprimer ceux qui sont en double.
J'essaye depuis un certain temps de trouver une solution mais je me retrouve avec des if sans dessus dessous alors je vous demande si quelqu'un connait une méthode simple et efficace.
Merci
Je suis à la recherche d'une idée pour résoudre un problème simple (sauf pour moi)
j'ai sous excel une suite de numéro :
10
10
12
14
16
16
16
16
18
20
...
J'aimerais dans un premier temps pouvoir garder tous les nombres en double et supprimer ceux qui son seul. Et dans un deuxième temps,en repartant de la même liste faire l'inverse, c'est à dire garder les nombres seul et supprimer ceux qui sont en double.
J'essaye depuis un certain temps de trouver une solution mais je me retrouve avec des if sans dessus dessous alors je vous demande si quelqu'un connait une méthode simple et efficace.
Merci
6 réponses
Salut !
As-tu essayé avec la fonction Count If ?
Voilà un code qui pourrait résoudre ton premier problème si tu as une liste de 20 nombres située dans la colonne A par exemple :
Sub Test()
For i = 1 To 20
Cells(1, 2).Value = WorksheetFunction.CountIf(Columns(1), Cells(i, 1).Value)
If Cells(1, 2).Value = 1 Then
Cells(1, 2).Value = ""
End If
Next i
End Sub
Pour ton deuxième problème, tu n'as qu'à remplacer le "=1" par ">1" !
En espérant t'avoir aidé !
As-tu essayé avec la fonction Count If ?
Voilà un code qui pourrait résoudre ton premier problème si tu as une liste de 20 nombres située dans la colonne A par exemple :
Sub Test()
For i = 1 To 20
Cells(1, 2).Value = WorksheetFunction.CountIf(Columns(1), Cells(i, 1).Value)
If Cells(1, 2).Value = 1 Then
Cells(1, 2).Value = ""
End If
Next i
End Sub
Pour ton deuxième problème, tu n'as qu'à remplacer le "=1" par ">1" !
En espérant t'avoir aidé !
bonjour,
si j'ai bien compris ton problème, tu ne veux que garder les nombres qui apparaissent deux fois car:
_ tu ne gardes que ceux qui apparaissent deux fois
_ puis tu ne gardes que si ils sont seuls
Pour que la macro fonctionne, il faut que ta liste soit triée!!!
et en plus ca va inscrire une formule en colonne 2, j'espère qu'il n'y a pas de données.
Sub x()
i = 1
Do While i <= 60000 'va regarder ta liste de la première ligne à la 60000
Cells(i, 2).Select
ActiveCell.FormulaR1C1 = "=COUNTIF(R" & i + 1 & "C[-1]:R60000C[-1], rc[-1])"
If Cells(i, 2) <> 1 Then
Range(Rows(i), Rows(i + Cells(i, 2))).Select
Selection.Delete Shift:=xlUp
i = i - 1
Else
Rows(i).Select
Selection.Delete Shift:=xlUp
End If
i = i + 1
Loop
End Sub
si j'ai bien compris ton problème, tu ne veux que garder les nombres qui apparaissent deux fois car:
_ tu ne gardes que ceux qui apparaissent deux fois
_ puis tu ne gardes que si ils sont seuls
Pour que la macro fonctionne, il faut que ta liste soit triée!!!
et en plus ca va inscrire une formule en colonne 2, j'espère qu'il n'y a pas de données.
Sub x()
i = 1
Do While i <= 60000 'va regarder ta liste de la première ligne à la 60000
Cells(i, 2).Select
ActiveCell.FormulaR1C1 = "=COUNTIF(R" & i + 1 & "C[-1]:R60000C[-1], rc[-1])"
If Cells(i, 2) <> 1 Then
Range(Rows(i), Rows(i + Cells(i, 2))).Select
Selection.Delete Shift:=xlUp
i = i - 1
Else
Rows(i).Select
Selection.Delete Shift:=xlUp
End If
i = i + 1
Loop
End Sub
Non, n'est pas conforme a mon besoin.
Liste de départ :
15
10
10
10
8
20
20
15
Au final j'essaie d'avoir dans mon 1er cas (supprimer les uni-nombre):
10
10
10
20
20
Au final j'essaie d'avoir dans mon 2ème cas (supprimer les multi-nombre):
15
8
15
Je cherche également une solution de mon côté pour résoudre ce casse tête
Liste de départ :
15
10
10
10
8
20
20
15
Au final j'essaie d'avoir dans mon 1er cas (supprimer les uni-nombre):
10
10
10
20
20
Au final j'essaie d'avoir dans mon 2ème cas (supprimer les multi-nombre):
15
8
15
Je cherche également une solution de mon côté pour résoudre ce casse tête
Bonjour,
éventuellement tu peux le faire sans vba :
Insérer une colonne B et une ligne 1 pour les titres de colonne.
- en B2 : =ET(NB.SI(A:A;A2)>1;EQUIV(A2;A:A;0)=LIGNE()) à recopier vers le bas
- mettre un filtre automatique en ligne 1 et sélectionner les FAUX en colonne B
- sélectionner la 1ère ligne, shift+clic sur la dernière, clic-droit et 'supprimer la ligne'
Il ne reste plus que le 1er exemplaire des doublons.
http://www.cijoint.fr/cjlink.php?file=cj201108/cijEyxEtEC.xls
eric
éventuellement tu peux le faire sans vba :
Insérer une colonne B et une ligne 1 pour les titres de colonne.
- en B2 : =ET(NB.SI(A:A;A2)>1;EQUIV(A2;A:A;0)=LIGNE()) à recopier vers le bas
- mettre un filtre automatique en ligne 1 et sélectionner les FAUX en colonne B
- sélectionner la 1ère ligne, shift+clic sur la dernière, clic-droit et 'supprimer la ligne'
Il ne reste plus que le 1er exemplaire des doublons.
http://www.cijoint.fr/cjlink.php?file=cj201108/cijEyxEtEC.xls
eric
bonjour,
Maintenant que j'ai mieux compris ton besoin, essaie cette petite macro
en colonne 2, tu auras ta liste des multinombres
en colonn3, tu auras ta liste des nombres seuls.
Sub x()
i = 1
a=2
c=2
Do While i <= 60000 'va regarder ta liste de la première ligne à la 60000
'si ton nombre est égal au nombre de la ligne suivante
if cells(i,1) = cells(i+1,1) then
cells(a,2) = cells(i,1)
b=1
do while cells(i,1) = cells(i+b,1)
b=b+1
loop
i=i+b
a=a+1
else
cells(c,3) = cells(i,1)
end if
Loop
End Sub
Maintenant que j'ai mieux compris ton besoin, essaie cette petite macro
en colonne 2, tu auras ta liste des multinombres
en colonn3, tu auras ta liste des nombres seuls.
Sub x()
i = 1
a=2
c=2
Do While i <= 60000 'va regarder ta liste de la première ligne à la 60000
'si ton nombre est égal au nombre de la ligne suivante
if cells(i,1) = cells(i+1,1) then
cells(a,2) = cells(i,1)
b=1
do while cells(i,1) = cells(i+b,1)
b=b+1
loop
i=i+b
a=a+1
else
cells(c,3) = cells(i,1)
end if
Loop
End Sub
j'ai fait une petite bourde dans ma macro, copie celle là :
Sub x()
i = 1
a=2
c=2
Do While i <= 60000 'va regarder ta liste de la première ligne à la 60000
'si ton nombre est égal au nombre de la ligne suivante
if cells(i,1) = cells(i+1,1) then
cells(a,2) = cells(i,1)
b=1
do while cells(i,1) = cells(i+b,1)
b=b+1
loop
i=i+b
a=a+1
else
cells(c,3) = cells(i,1)
c=c+1
i=i+1
end if
Loop
End Sub
Sub x()
i = 1
a=2
c=2
Do While i <= 60000 'va regarder ta liste de la première ligne à la 60000
'si ton nombre est égal au nombre de la ligne suivante
if cells(i,1) = cells(i+1,1) then
cells(a,2) = cells(i,1)
b=1
do while cells(i,1) = cells(i+b,1)
b=b+1
loop
i=i+b
a=a+1
else
cells(c,3) = cells(i,1)
c=c+1
i=i+1
end if
Loop
End Sub
Vous n’avez pas trouvé la réponse que vous recherchez ?
Posez votre question
Vos techniques me paraissent la bonne méthode, je vais m'en inspirer. Je vous direz si j'ai réussi d'ici peu
Puis-je enregistrer la valeur d'une case (6 chiffres maximum, positif) de cette manière ? :
Puis-je enregistrer la valeur d'une case (6 chiffres maximum, positif) de cette manière ? :
DIM x as long x=cells(1;1).value
Bonjour,
Effectivement, un nombre à 6 chiffres, sans décimales, est stocké en VBA sous le type Long. Pour info, Long : Nombre entier codé sur 4 octets (32 bits) et dont la valeur est comprise entre -2 147 483 648 et 2 147 483 647.
Le code que tu donnes est donc exact.
Pour continuer sur ta question initiale, ta liste de nombres est longue? combien de lignes?
Effectivement, un nombre à 6 chiffres, sans décimales, est stocké en VBA sous le type Long. Pour info, Long : Nombre entier codé sur 4 octets (32 bits) et dont la valeur est comprise entre -2 147 483 648 et 2 147 483 647.
Le code que tu donnes est donc exact.
Pour continuer sur ta question initiale, ta liste de nombres est longue? combien de lignes?
Alors, si j'ai bien compris, il s'agit juste d'une partie d'une plus grosse procédure?
Si tel est le cas, j'alimenterais deux variables tableaux pour pouvoir les restituer ultérieurement dans le code.
Comme ceci :
Si tel est le cas, j'alimenterais deux variables tableaux pour pouvoir les restituer ultérieurement dans le code.
Comme ceci :
Option Explicit Sub ReconstitutionAvecEtSansDoubles() 'déclaration des variables Dim TabloDoublons() 'As Long 'apostrophe à ôter si on est sur que ce ne sont que des nombres Dim TabloSingle() 'As Long 'apostrophe à ôter si on est sur que ce ne sont que des nombres Dim DrLig As Integer, Lig As Integer, Ind1 As Integer, Ind2 As Integer 'les variables Ind servent d'indices à nos variables tableaux. Ind1 = 0 Ind2 = 0 'DrLig stocke le numéro de la dernière ligne col A de la feuille Feuil1 A ADAPTER DrLig = Sheets("Feuil1").Columns(1).Find("*", , , , xlByColumns, xlPrevious).Row 'on boucle sur toutes les lignes de la colonne A For Lig = 1 To DrLig 'Si NB.SI(ColonneA, ActiveCell) > 1 (si on a la même valeur plus d'une fois) If WorksheetFunction.CountIf(Columns(1), Cells(Lig, 1).Value) > 1 Then 'alors on la stocke dans la variable tableaux "doublons" ReDim Preserve TabloDoublons(Ind1) TabloDoublons(Ind1) = Cells(Lig, 1).Value Ind1 = Ind1 + 1 'sinon, donc si NB.SI(ColonneA, ActiveCell) = 1 (si on a la même valeur qu'une fois) Else 'alors on la stocke dans la variable tableaux "Single" ReDim Preserve TabloSingle(Ind2) TabloSingle(Ind2) = Cells(Lig, 1).Value Ind2 = Ind2 + 1 End If 'et on passe à la cellule suivante Next 'A partir d'ici, les 6 lignes de code suivants restituent les valeurs contenues dans nos variables 'tableaux respectivement en colB et col C. 'si tu veux voir le résultat, enlève les apostrophes au début de chacune de ces lignes 'For Lig = 0 To UBound(TabloDoublons) ' Cells(Lig + 1, 2) = TabloDoublons(Lig) 'Next 'For Lig = 0 To UBound(TabloSingle) ' Cells(Lig + 1, 3) = TabloSingle(Lig) 'Next End Sub
Voila je suis enfin parvenu à un résultat correct grâce à vos code et vos idées, mon code est basic mais fonctionne. Merci
J'ai fais encore plus simple, de ma liste, tout les numéros seul sont déplacés vers une autre page et de ce fait j'aurais ma séparation. Les lignes vides seront supprimer par la suite.
Sub Séparation_Multiconducteurs()
Si quelqu'un veut s'amuser à l'optimiser, je ne suis pas contre, car pour l'instant ce n'est pas dans mes priorités.
Sujet résolu
Merci à tous, bonne continuation
J'ai fais encore plus simple, de ma liste, tout les numéros seul sont déplacés vers une autre page et de ce fait j'aurais ma séparation. Les lignes vides seront supprimer par la suite.
Sub Séparation_Multiconducteurs()
Dim mémorisation As Long Dim nb_ligne As Integer Dim ligne As Integer Dim boucle_ligne As Boolean Sheets("M_3,2").Select nb_ligne = Cells.Find("*", , , , , xlPrevious).Row ligne = 1 mémorisation = Cells(ligne, 1).Value retour: If boucle_ligne = False Then If WorksheetFunction.CountIf(Columns(1), Cells(ligne, 1).Value) = 1 Then Rows(ligne).Select Application.CutCopyMode = False Selection.Copy Sheets("M_3,2_U").Select Rows(ligne).Select ActiveSheet.Paste Sheets("M_3,2").Select Rows(ligne).Select Selection.ClearContents End If If ligne = nb_ligne Then boucle_ligne = True ligne = ligne + 1 GoTo retour End If
Si quelqu'un veut s'amuser à l'optimiser, je ne suis pas contre, car pour l'instant ce n'est pas dans mes priorités.
Sujet résolu
Merci à tous, bonne continuation