C# : Optimization

[".0s4*0|collections;datetime-format;optimization",["F@eBCDEV","PGGABEHAJILCDCEEMGLCECJEGAONKA","OOOCCCCBSTUUUUTTUUUUYFRDRDGDGDRDGDGDRDGDRDGDGDGDRDGDRDRDGDGDADADADGDADADFOOBBCCEOCOBOBOCCOBCCBEECOBOCCPRDGDGDGDGDRDADGDADFGDRDADGDADFADADOOBIBWSTTUUUUTTUUUU","....wshd..fttyyhrh.",".NET","Array","Dictionary","List","String","2D","Async","Console","DataTable","Dates","DateTime","Enum","File","For","Foreach","Format","IEnumerable","If","IndexOf","Lambda","LINQ","Optimization","Parse","Path","Process","Property","Random","Regex","Replace","Sort","Split","Static","Substring","Switch","Tuple","While","Tree, directed acyclic graph."," A directed acyclic graph contains nodes that do not form cycles. It can be used to store strings from a word list\u2014each letter is one node.","For search performance,"," an optimized tree is often the best solution. It has an initialization cost. And it requires some extra code to create. The C# version here is not optimal.","Example code."," In the C# language a tree can be implemented with classes (nodes) that point to further nodes. In this simple implementation, we use a Dictionary. ","AddWord: ","This method takes a string argument and adds it to the tree by creating (or using existing) nodes (one per character).","ContainsWord: ","This method meanwhile is similar to AddWord, but does not add anything. It just does lookups. It returns a bool.","Add: ","This method will add a new DictionaryNode to the Dictionary if one does not exist. It one already exists, it will not add another.","Get: ","Retrieves a node from the tree from the current node\u2014it uses a Dictionary lookup (TryGetValue).","TryGetValue ","trygetvalue","ins","class","adsbygoogle","data-ad-client","ca-pub-4712093147740724","data-ad-slot","6227126509","data-ad-format","auto","br","ins","class","adsbygoogle","data-ad-client","ca-pub-4712093147740724","data-ad-slot","6227126509","data-ad-format","auto","Based on:"," .NET 4.6\n\n","C# program that uses tree class","\n\nusing System;\nusing System.Collections.Generic;\n\nclass ","DictionaryNodeRoot","\n{\n DictionaryNode _root = new DictionaryNode();\n\n public void ","AddWord","(string value)\n {","\n // Add nodes from string chars.\n ","DictionaryNode current = this._root;\n for (int i = 0; i < value.Length; i++)\n {\n current = current.Add(value[i]);\n }","\n // Set state.\n ","current.SetWord(value);\n }\n\n public bool ","ContainsWord","(string value)\n {","\n // Get existing nodes from string chars.\n ","DictionaryNode current = this._root;\n for (int i = 0; i < value.Length; i++)\n {\n current = current.Get(value[i]);\n if (current == null)\n {\n return false;\n }\n }","\n // Return state.\n ","return current != null && current.GetWord() != null;\n }\n}\n\nclass ","DictionaryNode","\n{\n string _word;\n Dictionary<char, DictionaryNode> _dict;"," // Slow.\n\n ","public DictionaryNode ","Add","(char value)\n {","\n // Add individual node as child.\n // ... Handle null field.\n ","if (this._dict == null)\n {\n this._dict = new Dictionary<char, DictionaryNode>();\n }","\n // Look up and return if possible.\n ","DictionaryNode result;\n if (this._dict.TryGetValue(value, out result))\n {\n return result;\n }","\n // Store.\n ","result = new DictionaryNode();\n this._dict[value] = result;\n return result;\n }\n\n public DictionaryNode ","Get","(char value)\n {","\n // Get individual child node.\n ","if (this._dict == null)\n {\n return null;\n }\n DictionaryNode result;\n if (this._dict.TryGetValue(value, out result))\n {\n return result;\n }\n return null;\n }\n\n public void ","SetWord","(string word)\n {\n this._word = word;\n }\n\n public string ","GetWord","()\n {\n return this._word;\n }\n}\n\nclass Program\n{\n static void Main()\n {","\n // Create a tree.\n ","DictionaryNodeRoot tree = new DictionaryNodeRoot();","\n // Add three words to the tree.\n ","tree.AddWord(","\"dot\"",");\n tree.AddWord(","\"net\"",");\n tree.AddWord(","\"perls\"",");","\n\n // Search for strings in the tree.\n ","Console.WriteLine(tree.ContainsWord(","\"perls\"","));\n Console.WriteLine(tree.ContainsWord(","\"nothing\"","));\n }\n}\n\n","Output","\n\nTrue\nFalse","Notes, above program."," DictionaryNodeRoot serves as the root of the tree. It provides helper methods\u2014AddWord and ContainsNode. We use this class to add strings to the tree.","Notes, AddWord."," We add a string char-by-char with a for-loop in AddWord. The Add method from DictionaryNode.cs returns the newly-added or accessed node. We call Add repeatedly. ","Char ","char","For ","for","Info: ","This logic ends up building the tree. Nodes that already exist are not added again. They are instead shared.","Finally: ","The SetWord method is used. This signals to DictionaryType that a complete word has been reached at this node.","Bool ","bool","Notes, DictionaryNode."," On each node, we use a Dictionary instance. The Dictionary provides references to child nodes. The string field tells us whether a complete word here exists. ","Key: ","The Dictionary field uses a char key type. This is not essential. With this tree design, any value type could be used.","Notes, Add and Get."," Add and Get are similar. Then they do a lookup in the Dictionary for the value specified. They return the DictionaryNode found, or null. ","Null ","null","Discussion."," In software many applications require collections of strings. An obvious example is a spell-checker. All those words are stored somehow in memory. ","Spell-Checker ","spell-checker","Notes, advantages."," The tree accommodates vast data sets. This tree could handle any number of words within the limits of memory without a big performance drop. ","Simple: ","The design goal for this tree was simplicity. There are many ways to optimize the tree\u2014see the performance section.","Warning: ","Optimizations make the tree's design less obvious. They might not work for every requirement.","Performance."," Using the Dictionary as a field for child nodes is a slow implementation. An array, on each node, would be faster. Each element could instead be indexed in this array. ","Arrays ","array","Tip: ","Further optimizations are possible. The entire data structure can be collapsed into 1 or 2 int arrays.","Tip 2: ","These arrays essentially are a flattened form of the Dictionaries. The arrays reference one another.","Int Array ","int-array","Flatten Array ","flatten-array","Optimization ","optimization","And: ","A tree can be reduced so that the prefixes and suffixes are shared. The implementation here shares only prefixes.","Note, optimization."," Premature optimization is the root of something (I think). I have designed this same tree many times. And in none of those times has the performance been that critical. ","Benchmark ","benchmark","Performance test."," Here we compare the performance of an optimized version of our tree against a Dictionary generic. The tree only supports 26 lowercase letters. ","Result: ","The tree is almost twice as fast on a simple benchmark. This is in-line with other benchmark results.","Also: ","The tree is not done with optimization. An array of \"initial nodes\" can be used to optimize the first letter's look up.","C# program that benchmarks tree with nodes array","\n\nusing System;\n\nclass DictionaryNodeRoot\n{\n DictionaryNode _root = new DictionaryNode();\n\n public void AddWord(string value)\n {\n DictionaryNode current = this._root;\n for (int i = 0; i < value.Length; i++)\n {\n current = current.Add(value[i]);\n }\n current.SetWord(value);\n }\n\n public bool ContainsWord(string value)\n {\n DictionaryNode current = this._root;\n for (int i = 0; i < value.Length; i++)\n {\n current = current.Get(value[i]);\n if (current == null)\n {\n return false;\n }\n }\n return current != null && current.GetWord() != null;\n }\n}\n\nclass DictionaryNode\n{\n string _word;\n ","DictionaryNode[]"," _dict;"," // Use array for performance boost.\n\n ","public DictionaryNode Add(char value)\n {\n if (this._dict == null)\n {\n this._dict = new DictionaryNode[26];\n }","\n // Look up and return if possible.\n ","int key = value % 26;\n if (this._dict[key] != null)\n {\n return this._dict[key];\n }","\n // Store.\n ","var result = new DictionaryNode();\n this._dict[key] = result;\n return result;\n }\n\n public DictionaryNode Get(char value)\n {","\n // Get individual child node.\n ","if (this._dict == null)\n {\n return null;\n }\n int key = value % 26;\n return this._dict[key];\n }\n\n public void SetWord(string word)\n {\n this._word = word;\n }\n\n public string GetWord()\n {\n return this._word;\n }\n}\n\nclass Program\n{\n static void Main()\n {\n ","DictionaryNodeRoot"," tree = new DictionaryNodeRoot();\n tree.AddWord(\"cat\");\n tree.AddWord(\"car\");\n tree.AddWord(\"cap\");\n\n var t1 = System.Diagnostics.Stopwatch.StartNew();\n for (int i = 0; i < 10000000; i++)\n {\n if (!tree.ContainsWord(","\"car\"","))\n {\n return;\n }\n }","\n // ... Elapsed time.\n ","Console.WriteLine($","\"Time: {t1.ElapsedMilliseconds}\"",");\n }\n}\n\n","C# program that uses Dictionary","\n\nusing System;\nusing System.Collections.Generic;\n\nclass Program\n{\n static void Main()\n {","\n // Use Dictionary.\n ","var dictionary = new ","Dictionary","<string, bool>();\n dictionary[\"cat\"] = true;\n dictionary[\"car\"] = true;\n dictionary[\"cap\"] = true;\n\n var t1 = System.Diagnostics.Stopwatch.StartNew();\n for (int i = 0; i < 10000000; i++)\n {\n if (!dictionary.ContainsKey(","\"car\"","))\n {\n return;\n }\n }","\n // ... Elapsed time.\n ","Console.WriteLine($","\"Time: {t1.ElapsedMilliseconds}\"",");\n }\n}\n\n","Results","\n\nTime: ","124"," DictionaryNodeRoot.ContainsWord (with array for nodes)\nTime: ","202"," Dictionary.ContainsKey","Notes, prefix search."," With a tree, we can search just for prefixes\u2014the entire key does not need to be known. This can avoid many substrings. It is critical for games like Scrabble.","Some research."," Directed acyclic graphs are important. They can be used to find optimal paths. This can vastly improve programs\u2014a DAG could replace most primitive maze algorithms. ","Maze ","maze","For example, it is possible to find shortest paths and longest paths from a given starting vertex in DAGs in linear time by processing the vertices in a topological order, and calculating the path length for each vertex to be the minimum or maximum length obtained via any of its incoming edges.","Directed Acyclic Graph: Wikipedia ","https://en.wikipedia.org/wiki/Directed_acyclic_graph","A summary."," This tree uses Dictionary fields for child nodes. It works with strings. Its design can be adapted to other value types. The nodes can be optimized with other representations. ","br","ins","class","adsbygoogle","data-ad-client","ca-pub-4712093147740724","data-ad-slot","3679700504","data-ad-format","link","br","ins","class","adsbygoogle","data-ad-client","ca-pub-4712093147740724","data-ad-slot","6227126509","data-ad-format","auto"],"url()","url()","url()"]

["url()","url()","url()","url()","url()","url()","url()","url()","url()","B","url()","url()"]