ImmutableDictionary
This commit is contained in:
parent
b0c47947a6
commit
9e234f0663
2
.gitignore
vendored
2
.gitignore
vendored
|
@ -1,4 +1,4 @@
|
||||||
/tests_results
|
/tests_results
|
||||||
/main.exe
|
/main.exe
|
||||||
/*Generated.cs
|
/*Generated.cs
|
||||||
/*Generator.exe
|
/*Generator.exe
|
4
Lexer.cs
4
Lexer.cs
|
@ -72,7 +72,7 @@ public static partial class Lexer {
|
||||||
Rule(S.String, '"', S.StringClose, S.Space),
|
Rule(S.String, '"', S.StringClose, S.Space),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static Dictionary<S, List<Rule>> Dict =
|
public static DefaultDictionary<S, List<Rule>> Dict =
|
||||||
Default
|
Default
|
||||||
.GroupBy(r => r.oldState, r => r)
|
.GroupBy(r => r.oldState, r => r)
|
||||||
.ToDefaultDictionary(
|
.ToDefaultDictionary(
|
||||||
|
@ -81,7 +81,7 @@ public static partial class Lexer {
|
||||||
rs => rs.ToList()) ;
|
rs => rs.ToList()) ;
|
||||||
|
|
||||||
// This adds transitions through an implicit empty whitespace.
|
// This adds transitions through an implicit empty whitespace.
|
||||||
public static Dictionary<S, List<Rule>> WithEpsilonTransitions =
|
public static DefaultDictionary<S, List<Rule>> WithEpsilonTransitions =
|
||||||
Dict.ToDefaultDictionary(
|
Dict.ToDefaultDictionary(
|
||||||
new List<Rule>(),
|
new List<Rule>(),
|
||||||
kv => kv.Key,
|
kv => kv.Key,
|
||||||
|
|
2
Makefile
2
Makefile
|
@ -7,7 +7,7 @@ run: main.exe
|
||||||
mono main.exe
|
mono main.exe
|
||||||
|
|
||||||
main.exe: $(sort $(CS) $(GENERATED))
|
main.exe: $(sort $(CS) $(GENERATED))
|
||||||
mcs -out:$@ $^
|
mcs -out:$@ /reference:/usr/lib/mono/fsharp/FSharp.Core.dll $^
|
||||||
|
|
||||||
%Generated.cs: .%Generator.exe
|
%Generated.cs: .%Generator.exe
|
||||||
mono $<
|
mono $<
|
||||||
|
|
18
Parser.cs
18
Parser.cs
|
@ -12,8 +12,24 @@ public static partial class Parser {
|
||||||
|
|
||||||
public static PrecedenceDAG DefaultPrecedenceDAG = new PrecedenceDAG();
|
public static PrecedenceDAG DefaultPrecedenceDAG = new PrecedenceDAG();
|
||||||
|
|
||||||
public static PrecedenceDAG With(PrecedenceDAG precedenceDAG, Operator @operator) {
|
public static DAGNode With(DAGNode node, Operator op) {
|
||||||
|
/* var newOp = op.fixity.Match(
|
||||||
|
Closed: () => node.WithClosed(op),
|
||||||
|
InfixLeftAssociative: () => node.WithInfixLeftAssociative(op),
|
||||||
|
InfixRightAssociative: () => node.WithInfixRightAssociative(op),
|
||||||
|
InfixNonAssociative: () => node.WithInfixNonAssociative(op),
|
||||||
|
Prefix: () => node.WithPrefix(op),
|
||||||
|
Postfix: () => node.WithPostFix(op),
|
||||||
|
Terminal: () => node.WithTerminal(op)
|
||||||
|
);*/
|
||||||
|
// op.fixity, parts, holes
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static PrecedenceDAG With(PrecedenceDAG precedenceDAG, Operator @operator) {
|
||||||
|
/*precedenceDAG.update(
|
||||||
|
dagNode => dagNode.Add(@operator)
|
||||||
|
);*/
|
||||||
throw new NotImplementedException();
|
throw new NotImplementedException();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -24,21 +24,18 @@ public static class ParserGenerator {
|
||||||
Case("Terminal")),
|
Case("Terminal")),
|
||||||
|
|
||||||
Record("Operator",
|
Record("Operator",
|
||||||
Field("List<S>", "Parts"),
|
Field("Fixity", "fixity"),
|
||||||
Field("List<string>", "Holes")),
|
Field("List<S>", "parts"),
|
||||||
|
Field("List<string>", "holes")),
|
||||||
Record("Closed",
|
|
||||||
Field("S", "openSymbol"),
|
|
||||||
Field("S", "closedSymbol")),
|
|
||||||
|
|
||||||
Record("DAGNode",
|
Record("DAGNode",
|
||||||
Field("List<S>", "infixLeftAssociative"),
|
Field("List<Operator>", "infixLeftAssociative"),
|
||||||
Field("List<S>", "infixRightAssociative"),
|
Field("List<Operator>", "infixRightAssociative"),
|
||||||
Field("List<S>", "infixNonAssociative"),
|
Field("List<Operator>", "infixNonAssociative"),
|
||||||
Field("List<S>", "prefix"),
|
Field("List<Operator>", "prefix"),
|
||||||
Field("List<S>", "postfix"),
|
Field("List<Operator>", "postfix"),
|
||||||
Field("List<Closed>", "closed"),
|
Field("List<Operator>", "closed"),
|
||||||
Field("List<S>", "terminal"),
|
Field("List<Operator>", "terminal"),
|
||||||
Field("List<string>", "successors"))));
|
Field("List<string>", "successorNodes"))));
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -16,7 +16,7 @@ public static class Generator {
|
||||||
o.WriteLine($" /* To match against an instance of {name}, write:");
|
o.WriteLine($" /* To match against an instance of {name}, write:");
|
||||||
o.WriteLine($" x.Match(");
|
o.WriteLine($" x.Match(");
|
||||||
o.WriteLine(String.Join(",\n", variant.Select(@case =>
|
o.WriteLine(String.Join(",\n", variant.Select(@case =>
|
||||||
$" {@case.Key}: {@case.Value == null ? "()" : "value"} => throw new NotImplementedYetException(),")));
|
$" {@case.Key}: {@case.Value == null ? "()" : "value"} => throw new NotImplementedYetException()")));
|
||||||
o.WriteLine($" )");
|
o.WriteLine($" )");
|
||||||
o.WriteLine($" */");
|
o.WriteLine($" */");
|
||||||
|
|
||||||
|
@ -149,6 +149,16 @@ public static class Generator {
|
||||||
o.WriteLine(String.Join(",\n", record.Select(@field =>
|
o.WriteLine(String.Join(",\n", record.Select(@field =>
|
||||||
$" this.{@field.Key}")));
|
$" this.{@field.Key}")));
|
||||||
o.WriteLine($" );");
|
o.WriteLine($" );");
|
||||||
|
foreach (var @field in record) {
|
||||||
|
var F = @field.Key;
|
||||||
|
var noAtF = F.StartsWith("@") ? F.Substring(1) : F;
|
||||||
|
var caseF = Char.ToUpper(noAtF[0]) + noAtF.Substring(1);
|
||||||
|
var Ty = @field.Value;
|
||||||
|
o.Write($" public {name} With{caseF}({Ty} {F}) => new {name}(");
|
||||||
|
o.Write(String.Join(", ", record.Select(@f => $"{f.Key}: {f.Key}")));
|
||||||
|
o.WriteLine(");");
|
||||||
|
}
|
||||||
|
|
||||||
o.WriteLine($" }}");
|
o.WriteLine($" }}");
|
||||||
o.WriteLine($"{footer}");
|
o.WriteLine($"{footer}");
|
||||||
}
|
}
|
||||||
|
|
|
@ -152,7 +152,7 @@ public static class Collection {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Option<V> GetValue<K, V>(this Dictionary<K, V> d, K key) {
|
public static Option<V> GetValue<K, V>(this ImmutableDictionary<K, V> d, K key) {
|
||||||
V result = default(V);
|
V result = default(V);
|
||||||
if (d.TryGetValue(key, out result)) {
|
if (d.TryGetValue(key, out result)) {
|
||||||
return result.Some();
|
return result.Some();
|
||||||
|
@ -161,7 +161,7 @@ public static class Collection {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static V GetOrDefault<K, V>(this Dictionary<K, V> d, K key, V defaultValue) {
|
public static V GetOrDefault<K, V>(this ImmutableDictionary<K, V> d, K key, V defaultValue) {
|
||||||
V result = default(V);
|
V result = default(V);
|
||||||
if (d.TryGetValue(key, out result)) {
|
if (d.TryGetValue(key, out result)) {
|
||||||
return result;
|
return result;
|
||||||
|
@ -169,23 +169,4 @@ public static class Collection {
|
||||||
return defaultValue;
|
return defaultValue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class DefaultDictionary<TKey, TValue> : Dictionary<TKey, TValue> {
|
|
||||||
public readonly TValue defaultValue;
|
|
||||||
//public readonly Dictionary<TKey, TValue> dictionary;
|
|
||||||
|
|
||||||
public DefaultDictionary(TValue defaultValue, Dictionary<TKey, TValue> dictionary) : base(dictionary) {
|
|
||||||
this.defaultValue = defaultValue;
|
|
||||||
//this.dictionary = dictionary;
|
|
||||||
}
|
|
||||||
|
|
||||||
public new TValue this[TKey key] {
|
|
||||||
get {
|
|
||||||
return this.GetOrDefault(key, defaultValue);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static DefaultDictionary<UKey, UValue> ToDefaultDictionary<T, UKey, UValue>(this IEnumerable<T> e, UValue defaultValue, Func<T, UKey> key, Func<T, UValue> value)
|
|
||||||
=> new DefaultDictionary<UKey, UValue>(defaultValue, e.ToDictionary(key, value));
|
|
||||||
}
|
}
|
|
@ -1,6 +1,7 @@
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using Mutable = System.Collections.Generic;
|
||||||
|
|
||||||
public static class Func {
|
public static class Func {
|
||||||
// supply 1 argument to function of 2 arguments
|
// supply 1 argument to function of 2 arguments
|
||||||
|
@ -29,7 +30,7 @@ public static class Func {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Func<A, B> Memoize<A, B>(this Func<A, B> f) where A : IEquatable<A> {
|
public static Func<A, B> Memoize<A, B>(this Func<A, B> f) where A : IEquatable<A> {
|
||||||
var d = new Dictionary<A, B>();
|
var d = new Mutable.Dictionary<A, B>();
|
||||||
return a => {
|
return a => {
|
||||||
if (d.TryGetValue(a, out var b)) {
|
if (d.TryGetValue(a, out var b)) {
|
||||||
return b;
|
return b;
|
||||||
|
|
30
Utils/Immutable/DefaultDictionary.cs
Normal file
30
Utils/Immutable/DefaultDictionary.cs
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
//namespace Immutable {
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
|
public class DefaultDictionary<TKey, TValue> : ImmutableDictionary<TKey, TValue> {
|
||||||
|
public readonly TValue defaultValue;
|
||||||
|
|
||||||
|
public DefaultDictionary(TValue defaultValue, ImmutableDictionary<TKey, TValue> dictionary) : base(dictionary) {
|
||||||
|
this.defaultValue = defaultValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
public DefaultDictionary(DefaultDictionary<TKey, TValue> dictionary, TKey key, TValue value) : base(dictionary, key, value) {
|
||||||
|
this.defaultValue = dictionary.defaultValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
public new TValue this[TKey key] {
|
||||||
|
get {
|
||||||
|
return this.GetOrDefault(key, defaultValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public new DefaultDictionary<TKey, TValue> With(TKey key, TValue value)
|
||||||
|
=> new DefaultDictionary<TKey, TValue>(this, key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class DefaultDictionaryExtensionMethods {
|
||||||
|
public static DefaultDictionary<UKey, UValue> ToDefaultDictionary<T, UKey, UValue>(this IEnumerable<T> e, UValue defaultValue, Func<T, UKey> key, Func<T, UValue> value)
|
||||||
|
=> new DefaultDictionary<UKey, UValue>(defaultValue, e.ToImmutableDictionary(key, value));
|
||||||
|
}
|
||||||
|
//}
|
71
Utils/Immutable/Dictionary.cs
Normal file
71
Utils/Immutable/Dictionary.cs
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
//namespace Immutable {
|
||||||
|
using System;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Collections;
|
||||||
|
using Mutable = System.Collections.Generic;
|
||||||
|
using Microsoft.FSharp.Collections;
|
||||||
|
using System.Collections.Immutable;
|
||||||
|
|
||||||
|
// TODO: use Microsoft.FSharp.Collections.FSharpMap
|
||||||
|
public class ImmutableDictionary<TKey, TValue> : Mutable.IReadOnlyDictionary<TKey, TValue> {
|
||||||
|
private readonly Mutable.Dictionary<TKey, TValue> d;
|
||||||
|
private System.Collections.Immutable.ImmutableDictionary<TKey, TValue> i = System.Collections.Immutable.ImmutableDictionary<TKey, TValue>.Empty;
|
||||||
|
//private readonly FSharpMap<TKey, TValue> x = new FSharpMap<int, int>(Enumerable.Enpty<Tuple<int, int>>());
|
||||||
|
|
||||||
|
public ImmutableDictionary() {
|
||||||
|
d = new Mutable.Dictionary<TKey, TValue>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public ImmutableDictionary(Mutable.Dictionary<TKey, TValue> d) {
|
||||||
|
// Clone the mutable dictionary.
|
||||||
|
this.d = new Mutable.Dictionary<TKey, TValue>(d, d.Comparer);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ImmutableDictionary(ImmutableDictionary<TKey, TValue> immutableDictionary, TKey key, TValue value) {
|
||||||
|
// Clone the mutable dictionary contained within that immutable
|
||||||
|
// dictionary before updating it.
|
||||||
|
var clone = new Mutable.Dictionary<TKey, TValue>(immutableDictionary.d, immutableDictionary.d.Comparer);
|
||||||
|
clone.Add(key, value);
|
||||||
|
this.d = clone;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ImmutableDictionary(ImmutableDictionary<TKey, TValue> immutableDictionary) {
|
||||||
|
// No need to clone the mutable dictionary contained within
|
||||||
|
// that immutable dictionary, since we know it is never mutated.
|
||||||
|
this.d = immutableDictionary.d;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TValue this[TKey key] { get => d[key]; }
|
||||||
|
|
||||||
|
public IEnumerable<TKey> Keys { get => d.Keys; }
|
||||||
|
public IEnumerable<TValue> Values { get => d.Values; }
|
||||||
|
public int Count { get => d.Count; }
|
||||||
|
public IEqualityComparer<TKey> Comparer { get => d.Comparer; }
|
||||||
|
|
||||||
|
public bool TryGetValue(TKey key, out TValue value)
|
||||||
|
=> d.TryGetValue(key, out value);
|
||||||
|
|
||||||
|
public bool ContainsKey(TKey key)
|
||||||
|
=> d.ContainsKey(key);
|
||||||
|
|
||||||
|
public ImmutableDictionary<TKey, TValue> With(TKey key, TValue value)
|
||||||
|
=> new ImmutableDictionary<TKey, TValue>(this, key, value);
|
||||||
|
|
||||||
|
IEnumerator<KeyValuePair<TKey, TValue>> IEnumerable<KeyValuePair<TKey, TValue>>.GetEnumerator()
|
||||||
|
=> d.GetEnumerator();
|
||||||
|
|
||||||
|
IEnumerator IEnumerable.GetEnumerator()
|
||||||
|
=> d.GetEnumerator();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class ImmutableDictionaryExtensionMethods {
|
||||||
|
public static ImmutableDictionary<UKey, UValue> ToImmutableDictionary<T, UKey, UValue>(this IEnumerable<T> e, Func<T, UKey> key, Func<T, UValue> value)
|
||||||
|
=> new ImmutableDictionary<UKey, UValue>(e.ToDictionary(key, value));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prevent usage of the mutable dictionary.
|
||||||
|
public abstract class Dictionary<TKey, TValue> : ImmutableDictionary<TKey, TValue> {
|
||||||
|
}
|
||||||
|
|
||||||
|
//}
|
2
main.cs
2
main.cs
|
@ -81,7 +81,7 @@ public static class MainClass {
|
||||||
}
|
}
|
||||||
} catch (UserErrorException e) {
|
} catch (UserErrorException e) {
|
||||||
Console.WriteLine("");
|
Console.WriteLine("");
|
||||||
Console.WriteLine(e.Message);
|
Console.WriteLine($"\x1b[1;33m{e.Message}\x1b[m\n");
|
||||||
Environment.Exit(1);
|
Environment.Exit(1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user