C# : Optimization

[".0s4*0|collections;datetime-format;optimization",["F@eBCDEV","KIRCGEGKQABEOGBEDCOCPKGAMGMAIAJASGRCLCBEKLMANEIAICOEMMPEMA","OOOCSTUUUUTTUUUUYFHDRDRDRDGDGDFOCOCBCOCCOBCCPGHDADGDGDGDFOCOCCBOCBPGDHDFOCOBOCCPHDFHDOCOBBBBOCOBCOCPRDRDRDRDFRDRDRDROCCPFOCCPADADADADOCOCOCCCCPGDGHDGDGDGDOBCCBPGDGHDOCCCOCBOBOWSTTUUUUTTUUUU",".wttk.srlt.sfsrls.stssejrfre.fr.tfj..",".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","Struct."," A struct stores its data in its type. It is not allocated separately on the managed heap. Structs often reside on the evaluation stack.","Consider this:"," every program uses simple structs. All value types (int, bool, char) are structs. We rarely benefit from creating custom structs.","Example."," A struct uses syntax similar to a class. It is a type definition. We describe a struct called Simple: this struct stores three values, including an int, a bool and a double value. ","Stack: ","Please notice how, in Main, the struct is created on the stack. No \"new\" keyword is used. It is used like a value type such as int.","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 declares struct","\n\nusing System;\n\nclass Program\n{\n ","struct"," Simple\n {\n public ","int"," Position;\n public ","bool"," Exists;\n public ","double"," LastValue;\n };\n\n static void Main()\n {","\n // ... Create struct on stack.\n ","Simple s;\n s.Position = 1;\n s.Exists = false;\n s.LastValue = 5.5;","\n\n // ... Write struct field.\n ","Console.WriteLine(s.Position);\n }\n}\n\n","Output","\n\n1","Debugger."," Next we explore a struct within the Visual Studio debugger. The struct has the composite name \"Program.Simple\": it is nested within the Program class. ","Tip: ","In the debugger, you can inspect variables like the struct in this program. Depending on your approach to coding, this may help.","Values."," Structs are custom value types that store the values in each field together. They do not store referenced data, such as the character array in a string. ","Tip: ","Heap allocation is used for the data of arrays and objects. MSDN says that structs \"do not require heap allocation.\"","Structs Tutorial: MSDN ","https://msdn.microsoft.com/en-us/library/aa288471(VS.71).aspx","Further: ","Variables \"directly contain the data of the struct.\" And \"a variable of a class type contains a reference to the data.\"","Values, continued."," With structs you avoid the overhead of objects in the C# language. You can combine multiple fields. This reduces memory pressure. And it (sometimes) improves performance. ","Value semantics: ","This term indicates whether the variable is being used like numbers and values are, or as an inherited class.","Note: ","MSDN states that \"complex numbers, points in a coordinate system, or key-value pairs in a dictionary\" are included.","Usage."," Structs should be considered in performance-sensitive parts of a program. Points containing coordinates and positions are excellent examples of structs, as is the DateTime struct. ","DateTime ","datetime","Here: ","We demonstrate that a DateTime is a struct. We create a DateTime and then copy it into a separate DateTime variable.","Tip: ","When the original changes the copy remains the same. The memory is kept separate.","C# program that uses DateTime struct","\n\nusing System;\n\nclass Program\n{\n static void Main()\n {","\n // DateTime is a struct.\n ","DateTime"," date = new DateTime(","2000",", 1, 1);","\n\n // When you assign a DateTime, a separate copy is created.\n ","DateTime dateCopy = date;","\n\n // The two structs have the same values.\n ","Console.WriteLine(date);\n Console.WriteLine(dateCopy);","\n\n // The copy is not affected when the original changes.\n ","date = DateTime.MinValue;\n Console.WriteLine(dateCopy);\n }\n}\n\n","Output","\n\n1/1/2000 12:00:00 AM\n1/1/2000 12:00:00 AM\n1/1/2000 12:00:00 AM","Usage, small classes."," Often in programs we have small classes that serve as collections of related variables stored in memory. Here we require no inheritance or polymorphism. ","Tip: ","These types often make ideal structs. But even here, structs may make things worse (often because of parameter passing).","Usage, offsets."," Structs are useful for storing coordinates of offsets in your files. These usually contain integers. It is also worthwhile to use structs for graphics. ","Points: ","When using graphics contexts, we use structs for points and coordinates.","Note: ","MSDN provides an example of a nullable integer used for databases. Use this if you don't want to use Nullable(T).","Database Integer Type: MSDN ","https://msdn.microsoft.com/en-us/library/aa664482(VS.71).aspx","Property."," Next we see an example of using properties with the struct type. Remember that your struct cannot inherit like classes or have complex constructors. ","But: ","You can provide properties for the struct that simplify access to its data.","Property ","property","C# program that uses property on struct","\n\nusing System;\n\nclass Program\n{\n static void Main()\n {","\n // Initialize to 0.\n ","S st = new S();\n st.X = 5;\n Console.WriteLine(st.X);\n }\n\n ","struct"," S\n {\n int _x;\n public int X\n {\n get { return _x; }\n set\n {\n if (value < 10)\n {\n _x = value;\n }\n }\n }\n };\n}\n\n","Output","\n\n5","Stack, heap."," Locals are allocated on the stack. This includes ints and object references. When we create an object from a class, it is allocated on the heap. The stack is normally faster. ","Note: ","The details of stacks and heaps are out of the scope here, but are worthwhile studying.","Parameters."," This is important. Structs are copied when you pass them as parameters to methods. Structs can therefore degrade performance when passed to methods. ","Struct vs. Class ","struct-versus-class","Class."," You can't easily change many classes (ones that inherit or implement interfaces) to structs. For simple classes, we can specify a struct to test performance. ","Also: ","You cannot use a custom default constructor on structs, as when they are constructed, all fields are assigned to zero.","Tip: ","You don't have to instantiate your struct with the new keyword. It instead works like an int\u2014you can directly access it.","C# program that uses class","\n\nclass Program\n{\n ","class"," C\n {\n public int X;\n public int Y;\n };\n\n static void Main()\n {\n C local = new C();\n local.X = 1;\n local.Y = 2;\n }\n}\n\n","C# program that uses struct","\n\nclass Program\n{\n ","struct"," C\n {\n public int X;\n public int Y;\n };\n\n static void Main()\n {\n C local;\n local.X = 1;\n local.Y = 2;\n }\n}","Null."," You cannot compare a struct instance to the null literal. If this is confusing, think of structs as ints or bools. You can't set your integer variable to null. ","Note: ","Nullable types are a generic type that can be null. In complex programs these can help simplify code.","Nullable types."," The C# language provides a nullable type wrapper for any object. With the nullable wrapper, you can assign the null literal to even values like integers. ","Nullable Bool ","nullable-bool","Nullable DateTime ","nullable-datetime","Nullable Int ","nullable-int","Nullable Memory ","nullable-memory","Memory."," We examine the memory usage of allocating many structs versus the memory usage of allocating many classes. I looked at the memory layout of the console program in the CLRProfiler. ","CLRProfiler: ","This is a free tool by Microsoft that visualizes the memory allocations of .NET programs.","Memory, classes."," This is the memory profile of the version that uses classes. The List took 512 KB and was one object. Internally the List stored 100000 objects and required 3.8 MB. ","List ","list","Note: ","Sorry for the hard-to-read image. You can just my word for the results (or try the benchmark yourself).","Memory, structs."," Using structs, CLRProfiler indicates that the List took 24 bytes and contained one object of 4.0 MB. That one object is an array of 100000 structures, all stored together. ","Tip: ","Structs are not stored as separate objects in arrays, but are grouped together. We see that structs consume less memory.","Class version:","\n\nSize of List: ","1 object","\n ","512 KB","\nSize of internal array: ","100000 objects","\n ","3.8 MB","\n\n","Struct version:","\n\nSize of List: ","1 object","\n ","24 bytes","\nSize of internal array: ","1 object","\n ","4.0 MB","Allocation speed."," We allocate lots of classes. Then we allocate lots of structs. We time the performance of the .NET Framework for each type. ","Note: ","It was easier to gather speed benchmarks of structs. I compared two classes to two structs.","Note 2: ","Both pairs of opposites, class S and struct S, use either eight ints or four strings.","Classes tested in benchmark: C#","\n\nclass S\n{\n public int A;\n public int B;\n public int C;\n public int D;\n public int E;\n public int F;\n public int G;\n public int H;\n}\n\nclass S\n{\n public string A;\n public string B;\n public string C;\n public string D;\n}\n\n","Structs tested in benchmark: C#","\n\nstruct S\n{\n public int A;\n public int B;\n public int C;\n public int D;\n public int E;\n public int F;\n public int G;\n public int H;\n}\n\nstruct S\n{\n public string A;\n public string B;\n public string C;\n public string D;\n}","Allocation speed, continued."," Strings are reference types. Their internal data is not embedded in the struct. Just the reference or pointer is. ","Thus: ","The performance benefits of structs with strings persists even when their referential data is accessed.","Notes: ","10 million loops of 100000 iterations. Tests allocation speed of the data structures. Structs were much faster in both tests.","Struct/class allocation performance timing","\n\nClass with 8 ints: ","2418 ms","\nStruct with 8 ints: "," 936 ms"," [faster]\n\nClass with 4 string references: ","2184 ms","\nStruct with 4 string references: "," 795 ms"," [faster]","Allocation speed, analysis."," In this test, we see substantial speedups of over two times when allocating the structs. Note they are stored in a List field. ","Also: ","I tried to ensure accuracy of the test by assigning each field and avoiding property accesses.","Strings."," You can use string fields in struct types. The struct will not store the string's data. Strings are objects, so the data will be stored externally, where the reference points. ","However: ","Using structs can improve performance with the string reference itself. References are also data.","ASP.NET."," In an old project, I recorded referrer data. These store two string fields and a DateTime field. By hovering over DateTime in Visual Studio, you see that it too is a struct. ","And: ","Because DateTime itself is a struct, it will be stored directly in the struct allocation on the stack.","Thus: ","In a struct with two strings and a DateTime, the struct will hold two references and one value together.","Example: ","We replaced the class with a struct. This should improve performance by about two times and reduce memory.","However: ","Some benefits depend on usage. If the type is often passed to methods, this optimization could reduce speed.","C# program that uses struct in website","\n\nusing System;\nusing System.Collections.Generic;\n\nclass Program\n{\n static void Main()\n {\n var _d = new Dictionary<string, ReferrerInfo>();","\n\n // New struct:\n ","ReferrerInfo i;\n i.OriginalString = \"cat\";\n i.Target = \"mat\";\n i.Time = DateTime.Now;\n\n _d.Add(\"info\", i);\n }","\n\n /// <summary>\n /// Contains information about referrers.\n /// </summary>\n ","struct"," ReferrerInfo\n {\n public string OriginalString;"," // Reference.\n ","public string Target;"," // Reference.\n ","public DateTime Time;"," // Value.\n ","};\n}","Offsets."," In a database system, file blobs are stored in large files together. I needed a way to store their offsets. Therefore we can use a struct with two members\u2014ints storing positions. ","Int ","int","And: ","Structs were ideal here. There were over 500 instances of the object. And they only had two fields (both value types).","KeyValuePair: ","The .NET Framework provides the KeyValuePair struct for storing a pair. It is used internally in the Dictionary.","KeyValuePair ","keyvaluepair","C# that uses Dictionary of structs","\n\nusing System.Collections.Generic;\n\nclass Program\n{\n static void Main()\n {","\n // Stores Dictionary of structs.\n ","var _d = new Dictionary<string, FileData>();\n FileData f;\n f.Start = 1000;\n f.Length = 200;\n _d.Add(\"key\", f);\n }","\n\n /// <summary>\n /// Stores where each blob is stored.\n /// </summary>\n ","struct"," FileData\n {\n public int Start;\n public int Length;\n }\n}","Pointers."," The C# language frees us from the complexity of pointers and memory allocation. But understanding pointers is important. ","Tip: ","Pointers, like references, are values that contain the addresses of data. When used correctly, pointers are fast.","But: ","Their syntax and lack of error checking causes problems. They are harder to use than references.","Note: ","The struct keyword gives us more power over references and how fields are stored.","Pointers, unsafe."," Pointers are used with structs in unsafe code. Unsafe code is just that: not safe, but still usually functional. It is sometimes faster. ","Fixed buffers: ","The struct type can be used with fixed buffers as a memory region.","Fixed Buffer Struct ","fixed-buffer","ValueType."," Values such as integers are stored in structs. The ValueType class is an abstract base class for these values. You can refer to them with a ValueType reference. ","ValueType Class ","valuetype","A summary."," Structs are an advanced topic. But every developer has used them in value types. By understanding value semantics, we learn an important performance optimization.","A final caution."," Structs often degrade performance, so should be typically avoided. More important optimizations, like using Dictionary or List correctly, are often more helpful. ","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()","url()","url()","url()","url()","url()","B","url()","F","url()","url()","url()","url()","url()","url()","url()","url()","url()"]