Сравнить / посчитать значения в System.Collections.ArrayList - PullRequest
9 голосов
/ 24 января 2010

Я очищаю 5 файлов для определенного значения.Я не ожидаю каких-либо других значений, НО, поскольку это для моих собственных образовательных целей, я хотел бы, чтобы приложение подсчитывало, сравнивало и печатало наиболее популярные значения.

например:

ArrayList arrName = new ArrayList();
arrName.Add("BOB")
arrName.Add("JOHN")
arrName.Add("TOM")
arrName.Add("TOM")
arrName.Add("TOM")

Результатом, которого я хотел бы, будет TOM, но, будучи новичком, я действительно не знаю, как двигаться дальше.

Любые мысли, предложения или примеры очень ценятся.Спасибо.

Ответы [ 6 ]

4 голосов
/ 24 января 2010

Вы можете легко сделать это с LINQ, если вы можете использовать его с запросом, похожим на

names.Distinct().OrderByDescending(s => names.Count(u => u == s))).FirstOrDefault();

Возвращает значение с наибольшим количеством или default(Type). В случае эквивалентных подсчетов, он вернет первый с наибольшим количеством. Вы можете поместить этот метод в свои расширения с обобщениями для общего использования.

class Program
{
    static void Main(string[] args)
    {

        IEnumerable<String> names = new String[] { "BOB", 
                                                   "JOHN", 
                                                   "TOM", 
                                                   "TOM", 
                                                   "TOM" };
        var res = names.Top(); //returns "TOM"
    }
}

public static class Extensions
{

    public static T Top<T>(this IEnumerable<T> values)
    {
        return values.Distinct().OrderByDescending(s => values.Count(u => u.Equals(s))).FirstOrDefault();
    }
}

Если вам нужны все значения с наибольшим числом, например, если ваш список был "BOB", "JOHN", "JOHN", "TOM", "TOM" Полагаю, вы могли бы вместо этого использовать эту версию, чтобы вернуть и JOHN, и TOM:

    public static IEnumerable<T> Top<T>(this IEnumerable<T> values)
    {
        List<T> ret = new List<T>();
        int max = -1;

        foreach (var val in values.Distinct())
        {
            int count = values.Count(t => t.Equals(val));

            if (count >= max)
            {
                if (count > max)
                {
                    ret.Clear();
                    max = count;
                }
                ret.Add(val); //stacks equivalent count, if applicable
            }
        }

        return ret;
    }
2 голосов
/ 24 января 2010

Это та задача, для которой LINQ хорошо подходит.

Сначала давайте определим, что мы делаем:

  1. Группировка товаров по значению
  2. Подсчитайте каждую группу
  3. Вернуть предмет, группа которого имеет наибольшее количество

Этот запрос реализует вышеуказанное:

private string GetMostFrequent(IEnumerable<string> items)
{
    var itemsOrderedByCount =
        from item in items
        group item by item into itemGroup
        orderby itemGroup.Count() descending, itemGroup.Key
        select itemGroup.Key;

    return itemsOrderedByCount.FirstOrDefault();
}

Реализация очень похожа на высокоуровневое описание, приятный побочный эффект декларативного синтаксиса. Вот краткое объяснение каждой части:

from item in items

похоже на объявление цикла; item относится к переменной цикла.

group item by item into itemGroup

Это помещает каждый item в группу на основе его значения.

orderby itemGroup.Count() descending, itemGroup.Key

Это подсчитывает каждую группу и сортирует их так, что наиболее часто встречается первое. Если есть две группы с одинаковым количеством, выбирается меньшее значение. (Поскольку каждая группа содержит все одинаковые значения, ключ - это подсчитанный элемент.)

select itemGroup.Key

Это говорит о том, что для каждой группы мы просто хотим подсчитать элемент.

return itemsOrderedByCount.FirstOrDefault();

Это захватывает первый элемент в упорядоченном списке (тот, который имеет наибольшее количество). Если исходная последовательность пуста, возвращается ноль.

Использование:

var items = new[] { "BOB", "JOHN", "TOM", "TOM", "TOM" };

Assert.AreEqual("TOM", GetMostFrequent(items));
2 голосов
/ 24 января 2010

Вы не указали версию .Net / C #, которую вы используете, поэтому я рассмотрю это для каждой версии C #: v1, v2 и v3.

C # V1:

class CountValueComparer : IComparer
{
    public int Compare(object x, object y)
    {
        DictionaryEntry left = (DictionaryEntry)x;
        DictionaryEntry right = (DictionaryEntry)y;

        return ((int)left.Value).CompareTo((int)right.Value);
    }
}

Hashtable counts = new Hashtable();

foreach(String value in arrName)
{
    if (counts.ContainsKey(value))
    {
        int valueCount = (int)counts[value];
        ++valueCount;
        counts[value] = valueCount;
    }
    else
    {
        counts[value] = 1;
    }
}

DictionaryEntry[] sorted = new DictionaryEntry[counts.Count];
counts.CopyTo(sorted, 0);
Array.Sort(sorted, new CountValueComparer());

foreach (DictionaryEntry entry in sorted)
{
    Console.Writeline("Name: {0}; Count: {1}", entry.Key, entry.Value);
}

C # V2:

class CountValueComparer : IComparer<KeyValuePair<String, int>>
{
    public int Compare(int x, int y)
    {
        return x.Value.CompareTo(y.Value);
    }
}

// if v2, use the List<T> class!
List<String> arrName = new List<String>();

arrName.Add("TOM");
// etc...

Dictionary<String, int> counts = new Dictionary<String, int>();

foreach(String value in arrName)
{
    int count;
    if (counts.TryGetValue(value, out count))
    {
        counts[value] = ++count;
    }
    else
    {
        counts[value] = 1;
    }
}

KeyValuePair<String, int>[] sorted = new KeyValuePair<String, int>[counts.Count];
counts.CopyTo(sorted, 0);
Array.Sort(sorted, new CountValueComparer());

C # V3:

// if v3, use the List<T> class!
var arrName = new List<String>();

arrName.Add("TOM");
// etc...

var counts = (from n in arrName 
              group n by n into g 
              select new { Name = g.Key, Count = g.Count() })
              .OrderByDescending(x => x.Count);
var top = counts.FirstOrDefault();
Console.WriteLine("Name: {0}; Count: {1}", top.Name, top.Count);
1 голос
/ 24 января 2010

Вы можете использовать словарь (.NET 2.0+) для хранения повторного подсчета каждого значения:

Dictionary<string, int> counts = new Dictionary<string, int>();
foreach (string name in arrName) {
   int count;
   if (counts.TryGetValue(name, out count)) {
      counts[name] = count + 1;
   } else {
      counts.Add(name, 1);
   }
}

// and then look for the most popular value:

string mostPopular;
int max = 0;
foreach (string name in counts.Keys) {
   int count = counts[name];
   if (count > max) {
       mostPopular = name;
       max = count;
   }
}

// print it
Console.Write("Most popular value: {0}", mostPopular);

Если вы используете C # 3.0 (.NET 3.5 +), используйте:

var mostPopular = (from name in arrName.Cast<string>()
                   group name by name into g
                   orderby g.Count() descending
                   select g.Key).FirstOrDefault();

Console.Write("Most popular value: {0}", mostPopular ?? "None");
1 голос
/ 24 января 2010
    public static string GetMostPopular(ArrayList vals)
    {
        IDictionary<string, int> dict = new Dictionary<string, int>();
        int mx = 0;
        string ret = "";
        foreach (string x in vals)
        {
            if (!dict.ContainsKey(x))
            {
                dict[x] = 1;
            }
            else
            {
                dict[x]++;
            }
            if (dict[x] > mx)
            {
                mx = dict[x];
                ret = x;
            }
        }
        return ret;
    }

    static void Main()
    {
        ArrayList arrName = new ArrayList();
        arrName.Add("BOB");
        arrName.Add("JOHN");
        arrName.Add("TOM");
        arrName.Add("TOM");
        arrName.Add("TOM");
        string ans = GetMostPopular(arrName);
        Console.WriteLine(ans);
    }
0 голосов
/ 24 января 2010

Для перемещения по циклу вы можете использовать foreach:

foreach (string name in arrName) {
    Console.WriteLine(i);
}

И для подсчета значений вы можете использовать Hashtable, который сопоставляет ключи со значениями. Ключом может быть имя, а значением может быть то, сколько раз вы видели это имя в списке.

Hashtable nameHash = new Hashtable();
foreach (string name in arrName) {
    if (!nameHash.ContainsKey(name)) {
        nameHash.Add(name, 1);
    }
    else {
        int num = nameHash[name];
        nameHash.Add(name, num + 1);
    }
}
...