Dot Net Perls
Top 37 C# Example Pages...

[";sywttsrkrytthtdrtkrrt.t.bZ(CEST~~}T~~YF6666666666+BCCXB#CPXCCP4AXCEP69#CCC(CCXCCCP75666656(CsBBCP95468FXBBBC*BaBCCXS}T~~}T~~","IEqualityComparer."," This interface introduces a custom GetHashCode method. We implement this interface in the C# language. We also measure performance.","String keys."," We compare GetHashCode methods on custom string keys. We use it with the Dictionary type. You can implement the IEqualityComparer interface for the Dictionary collection.","First example."," First we see the custom IEqualityComparer I defined. It implements the IEqualityComparer interface, and its name is StringIndexKeyComparer. ","Note: ","Your comparer can have any name, but it must use the interface inheritance syntax as shown.","Interface ","interface","ins","class","adsbygoogle","data-ad-client","ca-pub-4712093147740724","data-ad-slot","6227126509","data-ad-format","auto","ins","class","adsbygoogle","data-ad-client","ca-pub-4712093147740724","data-ad-slot","6227126509","data-ad-format","auto","Based on:"," .NET 4.6\n\n","Class that implements IEqualityComparer: C#","\n\npublic class StringIndexKeyComparer : IEqualityComparer<string>\n{","\n /// <summary>\n /// Has a good distribution.\n /// </summary>\n ","const int _multiplier = 89;","\n\n /// <summary>\n /// Whether the two strings are equal\n /// </summary>\n ","public bool Equals(string x, string y)\n {\n return x == y;\n }","\n\n /// <summary>\n /// Return the hash code for this string.\n /// </summary>\n ","public int GetHashCode(string obj)\n {","\n // Stores the result.\n ","int result = 0;","\n\n // Don't compute hash code on null object.\n ","if (obj == null)\n {\n return 0;\n }","\n\n // Get length.\n ","int length = obj.Length;","\n\n // Return default code for zero-length strings [valid, nothing to hash with].\n ","if (length > 0)\n {","\n // Compute hash for strings with length greater than 1\n ","char let1 = obj[0];"," // First char of string we use\n ","char let2 = obj[length - 1];"," // Final char\n\n // Compute hash code from two characters\n ","int part1 = let1 + length;\n result = (_multiplier * part1) + let2 + length;\n }\n return result;\n }\n}","Class, notes."," This class implements IEqualityComparer. To implement that interface, it must define Equals and GetHashCode. The class uses a new GetHashCode implementation. ","IEqualityComparer Interface: MSDN ","https://msdn.microsoft.com/en-us/library/ms132151.aspx","Tip: ","The GetHashCode method above is optimal on the string keys I had. You will need to write your own GetHashCode method.","Tip 2: ","You should compute the result from the most significant parts of your keys, which provide the best distribution of the return value.","GetHashCode."," This method computes its result from the key[1] char, and from the final character in key. It applies a multiplier to the first character, and also adds the length of the key. ","Multiply ","multiply","Keys."," A Dictionary can have any type keys and values. But the hashing method used for those types is the default, unless you pass in an IEqualityComparer in the Dictionary constructor. ","Note: ","For my project, I have a set of about 650 string keys. The string keys are in a certain format, with a pattern to their characters.","String keys:","\n\n22D-Array-IEnumerable\n22D-Array-Use\n27-Zip\n27-Zip-DEFLATE-Ratio\n27-Zip-Examples\n2About\n2Action-Dictionary\n2Adobe-CS3-Rounded\n2Adobe-Fireworks-CS3-Resampling","Keys, analysis."," The string keys have the same first character, which is a digit. Because this first character is monotonous, it isn't useful for hashing because its distribution is poor. ","And: ","I decided that the best distribution of hash codes would use the second character, the length, and another character.","So: ","To reduce collisions, one character would need to be multiplied by a constant.","Benchmark: tested on 499 string keys","\n\nDefault string IEqualityComparer: ","4043 ms","\nCustom IEqualityComparer: ","2736 ms","Example, use IEqualityComparer."," We pass an instance of the IEqualityComparer to the Dictionary constructor. It is a good idea to use a static instance of your IEqualityComparer. ","Next: ","The code uses the same custom IEqualityComparer shown earlier in this document. It is passed to the Dictionary in the constructor.","Constructor ","constructor","C# program that uses custom class","\n\nusing System;\nusing System.Collections.Generic;","\n\n/// <summary>\n/// [see above]\n/// </summary>\n","public class StringIndexKeyComparer : IEqualityComparer<string>\n{","\n // [see above]\n","}\n\nclass Program\n{\n static void Main()\n {\n var custom = new Dictionary<string, bool>(","new StringIndexKeyComparer()",");\n custom.Add(\"22D-Array-IEnumerable\", true);\n custom.Add(\"22D-Array-Use\", true);\n custom.Add(\"27-Zip\", true);\n custom.Add(\"27-Zip-DEFLATE-Ratio\", true);\n custom.Add(\"27-Zip-Examples\", true);\n custom.Add(\"2About\", true);\n custom.Add(\"2Action-Dictionary\", true);\n custom.Add(\"2Adobe-CS3-Rounded\", true);\n custom.Add(\"2Adobe-Fireworks-CS3-Resampling\", true);\n }\n}","A discussion."," An excellent hash code can improve your Dictionary's performance. There are excellent books on this topic, such as Robert Sedgewick's Algorithms in C++, which I used. ","Tip: ","To better distribute the hash code computations in your program, using a multiplier is useful.","Tip 2: ","A hashtable has an internal array of buckets, and when you use a multiplier, you can spread out the indexes to those buckets.","And: ","This reduces the number of hash collisions and improves lookup performance.","Multiplier."," The constant 89 provides a good hashing distribution in many string data sets. There is a custom constructor I defined on the StringIndexKeyComparer. ","Tip: ","Any GetHashCode method will have to be custom to your keys. Therefore, just delete the contents of GetHashCode and start over.","But: ","It is not worth implementing on most projects, unless there is a bottleneck on your Dictionary, which rarely occurs.","Fuzzy lookups."," A Dictionary by default uses exact lookups. It computes a hash code based on a key and if another value matches it exactly, the two keys match. But this can be changed. ","Next: ","We implement a custom, fuzzy lookup that ignores characters. This is the custom IEqualityComparer.","Tip: ","It computes hashes (in GetHashCode) and compares keys (in Equals) ignoring all hyphen characters.","So: ","The string \"cat-1\" is considered equal to the string \"cat1.\" The hyphen (dash) character is simply ignored.","Example custom IEqualityComparer: C#","\n\nusing System.Collections.Generic;\n\nclass CustomComparer : ","IEqualityComparer","<string>\n{\n public bool ","Equals","(string x, string y)\n {\n int xPos = 0;\n int yPos = 0;\n while (true)\n {","\n // ... Fail if past end.\n ","if (xPos >= x.Length)\n {\n return false;\n }\n if (yPos >= y.Length)\n {\n return false;\n }","\n // ... Skip past hyphens.\n ","if (x[xPos] == '-')\n {\n xPos++;\n continue;\n }\n if (y[yPos] == '-')\n {\n yPos++;\n continue;\n }","\n // ... Fail if different.\n ","if (x[xPos] != y[yPos])\n {\n return false;\n }","\n // ... If we have traversed both strings with no error, we match.\n ","if (xPos == x.Length - 1 &&\n yPos == y.Length - 1)\n {\n return true;\n }\n // ... Increment both places.\n xPos++;\n yPos++;\n }\n }\n\n public int ","GetHashCode","(string obj)\n {\n int code = 0;","\n // ... Add together all chars.\n ","for (int i = 0; i < obj.Length; i++)\n {\n if (obj[i] != '-')\n {\n code += obj[i];\n }\n }\n return code;\n }\n}","A warning."," The code has some problems. Its implementation of GetHashCode is slow and would become slower on many keys. But the code shows the concept. ","And: ","It avoids string allocations. I have found this is often a key to good performance.","Fuzzy, continued."," Next, we use the CustomComparer we created. This Main method shows how hyphens are ignored in the keys of a Dictionary that uses this comparer. ","Main ","main","Var ","var","Here: ","All three lookups return the correct value. Some unusual cases, not shown, may not work correctly (such as trailing hyphens).","C# program that uses Dictionary, CustomComparer","\n\nusing System;\nusing System.Collections.Generic;\n\nclass Program\n{\n static void Main()\n {","\n // ... Add data to dictionary.\n ","var dictionary = new ","Dictionary","<string, int>(new ","CustomComparer","());\n dictionary[","\"cat-1\"","] = 1;\n dictionary[\"cat-2\"] = 2;\n dictionary[\"dog-bark\"] = 10;\n dictionary[\"dog-woof\"] = 20;","\n\n // ... Lookup values, ignoring hyphens.\n ","Console.WriteLine(dictionary[","\"cat1\"","]);\n Console.WriteLine(dictionary[","\"cat-1\"","]);\n Console.WriteLine(dictionary[\"dog--bark\"]);\n }\n}\n\n","Output","\n\n1\n1\n10","Fuzzy, uses."," This code has many possible uses. It can potentially eliminate many Substring, Replace or ToLower calls. You can simply use a hashing function, with no new strings. ","Substring ","substring","Replace ","replace","ToLower ","tolower","Tip: ","You could even implement a simple language using this hashing mechanism. It could process strings without modifying them.","In my experiments,"," avoiding string allocations (caused by Substring or similar methods) is usually a performance win. And this style of IEqualityComparer tends to benefit performance. ","Optimization ","optimization","GetHashCode."," This method can be optimized. Try multiplying its result by an important character in the string. Also, there is an easy way to test how selective your hash numbers are. ","GetHashCode ","gethashcode","Test: ","Increment a static int field each time Equals is called. With a better hash code, Equals is called fewer times.","Tip: ","With a more selective hash, the internal Dictionary methods need to search fewer values with Equals, so it is called less.","With IEqualityComparer,"," we look up strings based on the results of a method. We apply fuzzy matching, or any other transformation before comparing keys. We need not modify those keys. ","ins","class","adsbygoogle","data-ad-client","ca-pub-4712093147740724","data-ad-slot","3679700504","data-ad-format","link","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()","J","url()","K","url()","url()","url()"]