.NET Core 2.1 से शुरू करके स्ट्रिंग का उपयोग करके रिवर्स करने का एक नया तरीका है string.Create
विधि ।
ध्यान दें कि यह समाधान यूनिकोड संयोजन वर्ण आदि को सही तरीके से नहीं संभालता है, क्योंकि "लेस मिसे \ u0301rables" को "selbarésiM seL" में बदल दिया जाएगा। अन्य उत्तर एक बेहतर समाधान के लिए।
public static string Reverse(string input)
{
return string.Create<string>(input.Length, input, (chars, state) =>
{
state.AsSpan().CopyTo(chars);
chars.Reverse();
});
}
यह अनिवार्य रूप input
से एक नए स्ट्रिंग के पात्रों को कॉपी करता है और नए स्ट्रिंग को इन-प्लेस में उलट देता है।
क्यों string.Create
उपयोगी है?
जब हम मौजूदा सरणी से एक स्ट्रिंग बनाते हैं, तो एक नया आंतरिक सरणी आवंटित किया जाता है और मूल्यों की प्रतिलिपि बनाई जाती है। अन्यथा, इसके निर्माण (सुरक्षित वातावरण में) के बाद एक स्ट्रिंग को म्यूट करना संभव होगा। अर्थात्, निम्नलिखित स्निपेट में हमें 10 की लंबाई को दो बार आवंटित करना होगा, एक को बफर के रूप में और एक को स्ट्रिंग के आंतरिक सरणी के रूप में।
var chars = new char[10];
// set array values
var str = new string(chars);
string.Create
अनिवार्य रूप से हमें स्ट्रिंग के निर्माण समय के दौरान आंतरिक सरणी में हेरफेर करने की अनुमति देता है। यह है, हमें अब बफर की आवश्यकता नहीं है और इसलिए उस एक चार सरणी को आवंटित करने से बच सकते हैं।
स्टीव गॉर्डन ने इसके बारे में अधिक विस्तार से यहां लिखा है । MSDN पर एक लेख भी है ।
कैसे करें इस्तेमाल string.Create
?
public static string Create<TState>(int length, TState state, SpanAction<char, TState> action);
विधि तीन पैरामीटर लेता है:
- बनाने के लिए स्ट्रिंग की लंबाई,
- वह डेटा जिसे आप नए स्ट्रिंग को गतिशील रूप से बनाने के लिए उपयोग करना चाहते हैं,
- और एक प्रतिनिधि जो डेटा से अंतिम स्ट्रिंग बनाता है, जहां पहला पैरामीटर
char
नए स्ट्रिंग के आंतरिक सरणी को इंगित करता है और दूसरा वह डेटा (स्थिति) है जिसे आपने पास किया था string.Create
।
प्रतिनिधि के अंदर हम निर्दिष्ट कर सकते हैं कि डेटा से नया स्ट्रिंग कैसे बनाया जाता है। हमारे मामले में, हम Span
नए स्ट्रिंग द्वारा प्रयुक्त इनपुट स्ट्रिंग के अक्षरों को कॉपी करते हैं । फिर हम उल्टा करते हैं Span
और इसलिए पूरी स्ट्रिंग उलट जाती है।
मानक
स्वीकृत उत्तर के साथ एक स्ट्रिंग को उलटने के मेरे प्रस्तावित तरीके की तुलना करने के लिए, मैंने बेंचमार्कडॉटनेट का उपयोग करते हुए दो बेंचमार्क लिखे हैं।
public class StringExtensions
{
public static string ReverseWithArray(string input)
{
var charArray = input.ToCharArray();
Array.Reverse(charArray);
return new string(charArray);
}
public static string ReverseWithStringCreate(string input)
{
return string.Create(input.Length, input, (chars, state) =>
{
state.AsSpan().CopyTo(chars);
chars.Reverse();
});
}
}
[MemoryDiagnoser]
public class StringReverseBenchmarks
{
private string input;
[Params(10, 100, 1000)]
public int InputLength { get; set; }
[GlobalSetup]
public void SetInput()
{
// Creates a random string of the given length
this.input = RandomStringGenerator.GetString(InputLength);
}
[Benchmark(Baseline = true)]
public string WithReverseArray() => StringExtensions.ReverseWithArray(input);
[Benchmark]
public string WithStringCreate() => StringExtensions.ReverseWithStringCreate(input);
}
यहाँ मेरी मशीन पर परिणाम हैं:
| Method | InputLength | Mean | Error | StdDev | Gen 0 | Allocated |
| ---------------- | ----------- | -----------: | ---------: | --------: | -----: | --------: |
| WithReverseArray | 10 | 45.464 ns | 0.4836 ns | 0.4524 ns | 0.0610 | 96 B |
| WithStringCreate | 10 | 39.749 ns | 0.3206 ns | 0.2842 ns | 0.0305 | 48 B |
| | | | | | | |
| WithReverseArray | 100 | 175.162 ns | 2.8766 ns | 2.2458 ns | 0.2897 | 456 B |
| WithStringCreate | 100 | 125.284 ns | 2.4657 ns | 2.0590 ns | 0.1473 | 232 B |
| | | | | | | |
| WithReverseArray | 1000 | 1,523.544 ns | 9.8808 ns | 8.7591 ns | 2.5768 | 4056 B |
| WithStringCreate | 1000 | 1,078.957 ns | 10.2948 ns | 9.6298 ns | 1.2894 | 2032 B |
जैसा कि आप देख सकते हैं, ReverseWithStringCreate
हम ReverseWithArray
विधि द्वारा उपयोग की जाने वाली स्मृति का केवल आधा हिस्सा आवंटित करते हैं ।