2019-05-23 14:11:00 +02:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
2019-11-22 05:44:02 +01:00
|
|
|
|
using BrotliLib.Collections.Huffman;
|
2019-05-23 14:11:00 +02:00
|
|
|
|
|
|
|
|
|
namespace BrotliLib.Collections{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Collection that counts the amount of symbol occurrences.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public sealed class FrequencyList<T> : IReadOnlyCollection<T> where T : IComparable<T>{
|
2020-04-05 17:34:10 +02:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Constructs and initializes an array of empty <see cref="FrequencyList{T}"/> objects.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public static FrequencyList<T>[] Array(int size){
|
|
|
|
|
var array = new FrequencyList<T>[size];
|
|
|
|
|
|
|
|
|
|
for(int index = 0; index < array.Length; index++){
|
|
|
|
|
array[index] = new FrequencyList<T>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return array;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Constructs a <see cref="FrequencyList{T}"/> and immediately counts all symbols in the <paramref name="sequence"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public static FrequencyList<T> FromSequence(List<T> sequence){
|
|
|
|
|
return new FrequencyList<T>{ sequence };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Instance
|
|
|
|
|
|
2019-05-23 14:11:00 +02:00
|
|
|
|
public int Count => frequencies.Count;
|
2020-04-05 17:34:10 +02:00
|
|
|
|
public int Sum => frequencies.Sum(kvp => kvp.Value);
|
2019-05-23 14:11:00 +02:00
|
|
|
|
|
2020-04-07 17:51:34 +02:00
|
|
|
|
public IEnumerable<HuffmanGenerator<T>.SymbolFreq> HuffmanFreq{
|
|
|
|
|
get => frequencies.Select(kvp => new HuffmanGenerator<T>.SymbolFreq(kvp.Key, kvp.Value));
|
2019-05-23 14:11:00 +02:00
|
|
|
|
}
|
|
|
|
|
|
2020-04-05 18:11:26 +02:00
|
|
|
|
public int this[T symbol]{
|
|
|
|
|
get{
|
|
|
|
|
return frequencies.TryGetValue(symbol, out int count) ? count : 0;
|
|
|
|
|
}
|
|
|
|
|
set{
|
|
|
|
|
if (value < 0){
|
|
|
|
|
throw new ArgumentOutOfRangeException(nameof(value), "Symbol cannot have a negative frequency.");
|
|
|
|
|
}
|
|
|
|
|
else if (value == 0){
|
|
|
|
|
frequencies.Remove(symbol);
|
|
|
|
|
}
|
|
|
|
|
else{
|
|
|
|
|
frequencies[symbol] = value;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-23 14:11:00 +02:00
|
|
|
|
private readonly Dictionary<T, int> frequencies = new Dictionary<T, int>();
|
|
|
|
|
|
|
|
|
|
public bool Contains(T symbol){
|
|
|
|
|
return frequencies.ContainsKey(symbol);
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-05 17:34:10 +02:00
|
|
|
|
public void Clear(){
|
|
|
|
|
frequencies.Clear();
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-23 14:11:00 +02:00
|
|
|
|
public void Add(T symbol){
|
|
|
|
|
if (frequencies.ContainsKey(symbol)){
|
|
|
|
|
++frequencies[symbol];
|
|
|
|
|
}
|
|
|
|
|
else{
|
|
|
|
|
frequencies[symbol] = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-05 17:34:10 +02:00
|
|
|
|
public void Add(List<T> sequence){
|
|
|
|
|
foreach(T symbol in sequence){
|
|
|
|
|
Add(symbol);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Add(FrequencyList<T> other){
|
|
|
|
|
foreach(var (symbol, freq) in other.frequencies){
|
|
|
|
|
if (frequencies.ContainsKey(symbol)){
|
|
|
|
|
frequencies[symbol] += freq;
|
|
|
|
|
}
|
|
|
|
|
else{
|
|
|
|
|
frequencies[symbol] = freq;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-23 14:11:00 +02:00
|
|
|
|
public IEnumerator<T> GetEnumerator() => frequencies.Keys.GetEnumerator();
|
|
|
|
|
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
|
|
|
|
|
}
|
|
|
|
|
}
|