• ベストアンサー
※ ChatGPTを利用し、要約された質問です(原文:エクセルでDictionaryオブジェクト)

エクセルでDictionaryオブジェクトを使った人名ごとの金額集計方法

このQ&Aのポイント
  • エクセルでDictionaryオブジェクトを利用して、AN列にある人名データをKeyにしてT列にある金額の集計を行う方法を紹介します。
  • AN列のデータには重複するものが多いが同姓同名は存在しないため、重複した人名をまとめて金額を集計します。
  • また、新たな条件として、同じ表のN列にあるコードのうち特定のコードが存在する場合に、追加したワークシートのC列にフラグを立てる方法も紹介します。

質問者が選んだベストアンサー

  • ベストアンサー
  • watabe007
  • ベストアンサー率62% (476/760)
回答No.4

myDicを2個、作っちゃいました。 Sub test03()   Dim myDic1 As Object, myDic2 As Object   Dim ws As Worksheet   Set myDic1 = CreateObject("Scripting.Dictionary")   Set myDic2 = CreateObject("Scripting.Dictionary")   For Each myC In Range("AN1", Cells(Rows.Count, "AN").End(xlUp))     myDic1(myC.Value) = myDic1(myC.Value) + Cells(myC.Row, "T").Value     If Cells(myC.Row, "N").Value = 77777777 Then       myDic2(myC.Value) = "○"     ElseIf myDic2(myC.Value) <> "○" Then       myDic2(myC.Value) = Empty     End If   Next   Set ws = Worksheets.Add(After:=ActiveSheet)   With ws '新規ワークシート     .Range("A1").Resize(myDic1.Count, 1).Value = Application.Transpose(myDic1.keys)     .Range("B1").Resize(myDic1.Count, 1).Value = Application.Transpose(myDic1.items)     .Range("C1").Resize(myDic2.Count, 1).Value = Application.Transpose(myDic2.items)   End With   Set myDic1 = Nothing   Set myDic2 = Nothing End Sub

emaxemax
質問者

お礼

dictionaryを2ついっぺんに作ることができるんですね!知りませんでした。 これはmyDic1のKeyとmyDic2のkeyは同一で、itemがそれぞれ金額とフラグなんですね。 やってみました。 大成功です。 ありがとうございました。

その他の回答 (5)

回答No.6

もう、遅いのかもしれませんが、このままでは終われませんので、#1のコードを直しました。 Keyの方で、重みをつけることにしました。こちらは、元のコードから改編が少ないです。 '// Sub test01R2()  Dim myDic As Object  Dim x As Long  Dim myC As Range  Dim i As Long  Dim ws As Worksheet  Set myDic = CreateObject("Scripting.Dictionary")  x = Cells(Rows.Count, "AN").End(xlUp).Row  For Each myC In Range("AN1:AN" & x) '受取人名列   If Not myDic.Exists(myC.Value) Then    myDic.Add myC.Value, Cells(myC.Row, "T").Value 'T列の金額   Else    myDic(myC.Value) = myDic(myC.Value) + Cells(myC.Row, "T").Value '金額を加算   End If  Next  For Each myC In Range("AN1:AN" & x)   If Cells(myC.Row, "N").Value = 77777777 Then    If Not myDic.Exists(myC.Value & "_") Then     myDic.Key(myC.Value) = myC.Value & "_"    End If   End If  Next    Set ws = Worksheets.Add(After:=ActiveSheet)  With ws '新規ワークシート   .Range("A1").Resize(myDic.Count, 1).Value = Application.Transpose(myDic.keys) 'A列にデータ   .Range("B1").Resize(myDic.Count, 1).Value = Application.Transpose(myDic.Items) 'B列に結びつけ金額データ   For i = 1 To myDic.Count    If Right(.Cells(i, "A").Value, 1) = "_" Then     .Cells(i, "C").Value = "○"     .Cells(i, "A").Value = Replace(.Cells(i, "A").Value, "_", "")    End If   Next i  End With  Set myDic = Nothing End Sub '//

emaxemax
質問者

お礼

ありがとうございます。 勉強させていただきます。

回答No.5

こんにちは。 むろん、小数方式は、小数点第何位なら大丈夫という所があるはずですから、それを使っていいはずですが、前に触れていた方法です。書いてみて、正攻法かは知りませんが、あまりマクロの価値は少ないです。[AZ列]を使っていないという条件ですから、そこを使っているなら、別の場所を探さなくてはなりません。しかし、もう何の為に書いているのか分からなくなりました。 '// Sub Test02() Dim x As Long Dim i As Long Dim j As Long Dim ws As Worksheet Dim rngC As Range   x = Cells(Rows.Count, "AN").End(xlUp).Row 'データの数を調べる      Range("AN1", "AN" & x).Copy Range("AZ1") 'AZ1 にデータをコピー   Range("AZ1", "AZ" & x).RemoveDuplicates Columns:=1, Header:=xlNo '重複の削除   Set rngC = Range("AZ1", Cells(Rows.Count, "AZ").End(xlUp))      Application.ScreenUpdating = False   For i = 1 To rngC.Rows.Count     Cells(i, "AZ").Offset(, 1).FormulaLocal = "=SUMIF(AN1:AN" & x & ",AZ" & i & ",T1:T" & x & ")"   Next i   rngC.Offset(, 1).Value = rngC.Offset(, 1).Value '数式を値にする   For i = 1 To x     If Cells(i, "N").Value = 77777777 Then      j = Application.Match(Cells(i, "AN").Value, rngC, False)      rngC.Cells(j, 1).Offset(0, -1).Value = "○"     End If   Next i   Set ws = Worksheets.Add(After:=ActiveSheet)   With ws '新規ワークシート     rngC.CurrentRegion.Copy .Range("A1")   End With   rngC.CurrentRegion.Clear      Application.ScreenUpdating = True End Sub '//

emaxemax
質問者

お礼

WindFaller さん、何度もありがとうございます。 勉強になりました。

  • watabe007
  • ベストアンサー率62% (476/760)
回答No.3

myDicに配列を格納してみました。参考に Sub test02()   Dim myDic As Object   Dim myC As Range   Dim ws As Worksheet   Dim tmp As Variant, d As Variant, v, i As Long   Set myDic = CreateObject("Scripting.Dictionary")   For Each myC In Range("AN1", Cells(Rows.Count, "AN").End(xlUp))     If Not myDic.Exists(myC.Value) Then       tmp = Array(Cells(myC.Row, "T").Value, "")     Else       tmp = myDic(myC.Value)       tmp(0) = tmp(0) + Cells(myC.Row, "T").Value     End If     If Cells(myC.Row, "N").Value = 77777777 Then tmp(1) = "○"       myDic(myC.Value) = tmp     Next     ReDim v(1 To myDic.Count, 1 To 2)     For Each d In myDic.Items       i = i + 1       v(i, 1) = d(0)       v(i, 2) = d(1)     Next     Set ws = Worksheets.Add(After:=ActiveSheet)     With ws '新規ワークシート       .Range("A1").Resize(myDic.Count, 1).Value = Application.Transpose(myDic.keys) 'A列にデータ       .Range("B1").Resize(myDic.Count, 2).Value = v 'B列に結びつけ金額データ   End With   Set myDic = Nothing End Sub

emaxemax
質問者

お礼

メリークリスマス! ありがとうございます。 ただ、使い方がよくわかりませんでした。すみません。

回答No.2

#1の回答者です。 >これだと77777777コードが同一人物で10個あった場合、少数じゃなくなってしまいますよね? これは、小数の値は何でもよいわけですから、10個あるようなら、0.01にすればよいのでは? そうすれば、100個入りますよね。 今回は、あくまでも、全バージョン対象ということですが、それ以外にも、Excelの機能(Ver.2010以上?)を使った正攻法がありますね。たぶん、そういう回答も必ず出てくるはずですが、これは、あくまでも、Dictionary オブジェクトを使った場合です。私個人は、多少は、動きが遅いようですが、バージョンに影響されないDictinary オブジェクトの方が万能型に軍配をあげます。

emaxemax
質問者

お礼

ありがとうございます。 データをよく見たら、例外的に小数点がつくものがありました。 すみません。 エクセルのバージョンは2010です。

回答No.1

こんにちは。 少し質問が分かっていないかもしれませんが、結果的には人名に対してフラグが立つことを意味しているわけですね。以下は、金額が整数であるという条件で、小数があるとうまくないのですが、重み付けという手法です。 '// Sub test01R()   Dim myDic As Object   Dim x As Long   Dim myC As Range   Dim i As Long   Dim ws As Worksheet   Set myDic = CreateObject("Scripting.Dictionary")   x = Cells(Rows.Count, "AN").End(xlUp).Row   For Each myC In Range("AN1:AN" & x) '受取人名列     If Not myDic.Exists(myC.Value) Then        myDic.Add myC.Value, Cells(myC.Row, "T").Value 'T列の金額     Else       myDic(myC.Value) = myDic(myC.Value) + Cells(myC.Row, "T").Value '金額を加算     End If     If Cells(myC.Row, "N").Value = 77777777 Then       myDic(myC.Value) = myDic(myC.Value) + 0.1 '重み付け     End If   Next   Set ws = Worksheets.Add(After:=ActiveSheet)   With ws '新規ワークシート     .Range("A1").Resize(myDic.Count, 1).Value = Application.Transpose(myDic.keys) 'A列にデータ     .Range("B1").Resize(myDic.Count, 1).Value = Application.Transpose(myDic.Items) 'B列に結びつけ金額データ     For i = 1 To myDic.Count      If .Cells(i, 2).Value - Int(.Cells(i, 2).Value) <> 0 Then       .Cells(i, 3).Value = "○" 'フラグ       .Cells(i, 2).Value = Int(.Cells(i, 2).Value)      End If     Next i   End With   Set myDic = Nothing End Sub '//

emaxemax
質問者

お礼

諦めてワークシート関数を使い、手作業で終了させたところです。 なるほど・・・・・。 数値は金額なのですべて整数ですのでおもしろいアイデアです。 ありがとうございます。 ただ、これだと77777777コードが同一人物で10個あった場合、少数じゃなくなってしまいますよね?実際、10個程度はざらにあるんです。 今後、毎月出てくる作業なのでなんとか自動化したいと思っております。 何はともあれありがとうございました。 メリークリスマス。

関連するQ&A

専門家に質問してみよう