एक सफेद स्थान के साथ कई सफेद रिक्त स्थान कैसे बदलें


108

मान लीजिए कि मेरे पास एक स्ट्रिंग है जैसे:

"Hello     how are   you           doing?"

मैं एक ऐसा कार्य करना चाहता हूँ जो कई स्थानों को एक स्थान पर बदल दे।

तो मुझे मिलेगा:

"Hello how are you doing?"

मुझे पता है कि मैं regex या कॉल का उपयोग कर सकता हूं

string s = "Hello     how are   you           doing?".replace("  "," ");

लेकिन मुझे यह सुनिश्चित करने के लिए कई बार कॉल करना होगा कि सभी अनुक्रमिक व्हाट्सएप केवल एक के साथ बदल दिए जाते हैं।

क्या पहले से ही इसके लिए एक निर्मित विधि है?


क्या आप स्पष्ट कर सकते हैं: क्या आप केवल रिक्त स्थान, या "सभी" व्हाट्सएप के साथ काम कर रहे हैं?
जॉन स्कीट

और क्या आप चाहते हैं कि कोई भी गैर-स्पेस व्हॉट्सएप स्पेस में बदल जाए?
जॉन स्केट

मैं सिर्फ श्रृंखला में सभी व्हाट्सएप का मतलब अधिकतम 1
मैट

1
के संभावित डुप्लिकेट stackoverflow.com/questions/206717/...
माइकल Freidgeim

2 बातों पर विचार करने के लिए: 1. char.IsWhiteSpace गाड़ी-रिटर्न, शामिल linefeed आदि 2. 'खाली स्थान के' शायद अधिक सही Char.GetUnicodeCategory (ch) के साथ परीक्षण किया जाता है = Globalization.UnicodeCategory.SpaceSeparator
smirkingman

जवाबों:


196
string cleanedString = System.Text.RegularExpressions.Regex.Replace(dirtyString,@"\s+"," ");

40
Imo, regex से बचना अगर आपके साथ उनका आराम समय से पहले होने वाला अनुकूलन है
Tim Hoolihan

8
यदि आप आवेदन करने में समय नहीं लगाते हैं, तो यह ओवरहेड प्रसंस्करण के 1 माइक्रोसेकंड का खर्च उठा सकता है।
डैनियल

16
ध्यान दें कि 's' न केवल श्वेत रिक्त स्थान को प्रतिस्थापित करता है, बल्कि नई पंक्ति वर्ण भी है।
बार्ट किर्स

12
अच्छी पकड़, यदि आप चाहते हैं कि रिक्त स्थान पैटर्न को "[] +" पर ले जाएँ
टिम होलिहान

9
क्या आपको सिंगल व्हाट्सएप को बदलने से बचने के लिए '+' के बजाय '{2,}' का उपयोग नहीं करना चाहिए?
18

52

यह प्रश्न उतना सरल नहीं है जितना कि अन्य पोस्टरों ने इसे बनाया है (और जैसा कि मैं मूल रूप से इसे मानता था) - क्योंकि यह प्रश्न बहुत सटीक नहीं है क्योंकि इसे होने की आवश्यकता है।

"स्पेस" और "व्हाट्सएप" के बीच अंतर है। यदि आप केवल रिक्त स्थान का मतलब रखते हैं, तो आपको एक regex का उपयोग करना चाहिए " {2,}"। अगर आपका मतलब किसी व्हाट्सएप से है तो यह अलग बात है। क्या सभी व्हाट्सएप को स्पेस में बदलना चाहिए? शुरू और अंत में अंतरिक्ष का क्या होना चाहिए?

नीचे दिए गए मानदंड के लिए, मैंने मान लिया है कि आप केवल रिक्त स्थान की परवाह करते हैं, और आप शुरुआत और अंत में भी सिंगल स्पेस के लिए कुछ भी नहीं करना चाहते हैं।

ध्यान दें कि शुद्धता प्रदर्शन की तुलना में लगभग हमेशा महत्वपूर्ण होती है। यह तथ्य कि स्प्लिट / ज्वाइन सॉल्यूशन किसी भी अग्रणी / अनुगामी व्हाट्सएप (यहां तक ​​कि केवल एक रिक्त स्थान) को हटा देता है, जहां तक ​​आपकी निर्दिष्ट आवश्यकताओं (जो कि अपूर्ण हो सकती है) के रूप में गलत है।

बेंचमार्क MiniBench का उपयोग करता है ।

using System;
using System.Text.RegularExpressions;
using MiniBench;

internal class Program
{
    public static void Main(string[] args)
    {

        int size = int.Parse(args[0]);
        int gapBetweenExtraSpaces = int.Parse(args[1]);

        char[] chars = new char[size];
        for (int i=0; i < size/2; i += 2)
        {
            // Make sure there actually *is* something to do
            chars[i*2] = (i % gapBetweenExtraSpaces == 1) ? ' ' : 'x';
            chars[i*2 + 1] = ' ';
        }
        // Just to make sure we don't have a \0 at the end
        // for odd sizes
        chars[chars.Length-1] = 'y';

        string bigString = new string(chars);
        // Assume that one form works :)
        string normalized = NormalizeWithSplitAndJoin(bigString);


        var suite = new TestSuite<string, string>("Normalize")
            .Plus(NormalizeWithSplitAndJoin)
            .Plus(NormalizeWithRegex)
            .RunTests(bigString, normalized);

        suite.Display(ResultColumns.All, suite.FindBest());
    }

    private static readonly Regex MultipleSpaces = 
        new Regex(@" {2,}", RegexOptions.Compiled);

    static string NormalizeWithRegex(string input)
    {
        return MultipleSpaces.Replace(input, " ");
    }

    // Guessing as the post doesn't specify what to use
    private static readonly char[] Whitespace =
        new char[] { ' ' };

    static string NormalizeWithSplitAndJoin(string input)
    {
        string[] split = input.Split
            (Whitespace, StringSplitOptions.RemoveEmptyEntries);
        return string.Join(" ", split);
    }
}

कुछ परीक्षण चलता है:

c:\Users\Jon\Test>test 1000 50
============ Normalize ============
NormalizeWithSplitAndJoin  1159091 0:30.258 22.93
NormalizeWithRegex        26378882 0:30.025  1.00

c:\Users\Jon\Test>test 1000 5
============ Normalize ============
NormalizeWithSplitAndJoin  947540 0:30.013 1.07
NormalizeWithRegex        1003862 0:29.610 1.00


c:\Users\Jon\Test>test 1000 1001
============ Normalize ============
NormalizeWithSplitAndJoin  1156299 0:29.898 21.99
NormalizeWithRegex        23243802 0:27.335  1.00

यहां पहला नंबर पुनरावृत्तियों की संख्या है, दूसरा समय लिया गया है, और तीसरा स्केल स्कोर है जिसमें 1.0 सर्वश्रेष्ठ है।

यह दर्शाता है कि कम से कम कुछ मामलों में (इस सहित) एक नियमित अभिव्यक्ति स्प्लिट / जॉइन समाधान को बेहतर बना सकती है , कभी-कभी बहुत महत्वपूर्ण मार्जिन से।

हालांकि, अगर आप एक "सभी खाली स्थान के" आवश्यकता है, तो विभाजन करने के लिए बदल / जुड़ें करता जीतने के लिए दिखाई देते हैं। जैसा कि अक्सर होता है, शैतान विस्तार में है ...


1
महान विश्लेषण। इसलिए ऐसा प्रतीत होता है कि हम दोनों अलग-अलग डिग्री के लिए सही थे। मेरे उत्तर में कोड एक बड़े फ़ंक्शन से लिया गया था, जिसमें एक स्ट्रिंग के भीतर और शुरुआत से और अंत तक सभी व्हाट्सएप और / या नियंत्रण वर्णों को सामान्य करने की क्षमता है।
स्कॉट डोरमैन

1
मेरे द्वारा बताए गए व्हाट्सएप पात्रों में से अधिकांश के साथ, रेगेक्स और स्प्लिट / जॉइन में अधिकांश समान थे - एस / जे को शुद्धता और जटिलता की कीमत पर एक छोटा, छोटा लाभ था। उन कारणों के लिए, मैं सामान्य रूप से रेगेक्स पसंद करूंगा। मुझे गलत मत समझो - मैं एक रेगेक्स फैनबॉय से बहुत दूर हूं, लेकिन मुझे वास्तव में पहले प्रदर्शन का परीक्षण किए बिना प्रदर्शन के लिए अधिक जटिल कोड लिखना पसंद नहीं है।
जॉन स्कीट

NormalizeWithSplitAndJoin बहुत अधिक कचरा पैदा करेगा, यह बताना मुश्किल है कि क्या वास्तविक समस्या अधिक GC समय और प्रतिबंध के बाद अधिक हिट होगी।
इयान रिंगरोज

@IanRingrose किस प्रकार का कचरा बनाया जा सकता है?
द्रोणज

18

एक नियमित एक्सप्रेस-वे सबसे आसान तरीका होगा। यदि आप रेगेक्स को सही तरीके से लिखते हैं, तो आपको कई कॉल की आवश्यकता नहीं होगी।

इसे इसमें बदलें:

string s = System.Text.RegularExpressions.Regex.Replace(s, @"\s{2,}", " "); 

इसके साथ मेरा एक मुद्दा @"\s{2,}"यह है कि यह एकल टैब और अन्य यूनिकोड अंतरिक्ष वर्णों को एक स्थान से बदलने में विफल रहता है। यदि आप एक स्थान के साथ 2 टैब को बदलने जा रहे हैं, तो आपको संभवतः 1 टैब को एक स्थान से बदलना चाहिए। @"\s+"आपके लिए वही करेंगे।
डेविड स्पीच

17

हालांकि मौजूदा उत्तर ठीक हैं, मैं एक दृष्टिकोण बताना चाहूंगा जो काम नहीं करता है:

public static string DontUseThisToCollapseSpaces(string text)
{
    while (text.IndexOf("  ") != -1)
    {
        text = text.Replace("  ", " ");
    }
    return text;
}

यह हमेशा के लिए पा सकते हैं। किसी को भी यह अनुमान लगाने की परवाह है कि क्यों? (मैं केवल इस पार आया था जब यह कुछ साल पहले एक समाचार समूह प्रश्न के रूप में पूछा गया था ... कोई वास्तव में एक समस्या के रूप में भाग गया।)


मुझे लगता है कि मुझे यह सवाल याद है कि एसओ पर थोड़ी देर के लिए पूछा गया था। IndexOf कुछ वर्णों को अनदेखा करता है जो प्रतिस्थापित नहीं करता है। तो डबल स्पेस हमेशा था, बस कभी हटा नहीं।
ब्रैंडन

19
ऐसा इसलिए है क्योंकि IndexOf कुछ यूनिकोड वर्णों की उपेक्षा करता है, इस मामले में विशिष्ट अपराधी कुछ एशियाई चरित्र iirc है। हम्म, Google के अनुसार शून्य-चौड़ाई वाला गैर-योजक।
अहाकर

मैंने सीखा है कि कठिन तरीका :( stackoverflow.com/questions/9260693/…
एंटोनियो बकुला

मैने कठिनाइयों का सामना कर सीखा। दो शून्य चौड़ाई गैर योजक (\ u200C \ u200C) के साथ विशिष्ट। IndexOf इस "डबल स्पेस" का इंडेक्स देता है, लेकिन रिप्लेसमेंट इसकी जगह नहीं लेता है। मुझे लगता है कि ऐसा इसलिए है क्योंकि IndexOf के लिए, आपको StringComparsion (ऑर्डिनल) को प्रतिस्थापित करने के लिए समान व्यवहार करने की आवश्यकता है। इस तरह, इन दोनों में से कोई भी "डबल स्पेस" नहीं पाएगा। StringComparsion docs.microsoft.com/en-us/dotnet/api/… के
मार्टिन

4

जैसा कि पहले ही बताया गया है, यह आसानी से एक नियमित अभिव्यक्ति द्वारा किया जाता है। मैं सिर्फ इतना जोड़ूंगा कि आप एक .trim () जोड़ना चाहते हैं जिससे अग्रणी / अनुगामी व्हाट्सएप से छुटकारा पा सकें।


4

यहाँ समाधान मैं के साथ काम है। RegEx और स्ट्रिंग के बिना।

public static string TrimWhiteSpace(this string Value)
{
    StringBuilder sbOut = new StringBuilder();
    if (!string.IsNullOrEmpty(Value))
    {
        bool IsWhiteSpace = false;
        for (int i = 0; i < Value.Length; i++)
        {
            if (char.IsWhiteSpace(Value[i])) //Comparion with WhiteSpace
            {
                if (!IsWhiteSpace) //Comparison with previous Char
                {
                    sbOut.Append(Value[i]);
                    IsWhiteSpace = true;
                }
            }
            else
            {
                IsWhiteSpace = false;
                sbOut.Append(Value[i]);
            }
        }
    }
    return sbOut.ToString();
}

तो तुम कर सकते हो:

string cleanedString = dirtyString.TrimWhiteSpace();

4

एक तेज़ अतिरिक्त व्हाट्सएप रिमूवर ... यह सबसे तेज़ है और फेलिप मैकडो की इन-प्लेस कॉपी पर आधारित है।

static string InPlaceCharArray(string str)
{
    var len = str.Length;
    var src = str.ToCharArray();
    int dstIdx = 0;
    bool lastWasWS = false;
    for (int i = 0; i < len; i++)
    {
        var ch = src[i];
        if (src[i] == '\u0020')
        {
            if (lastWasWS == false)
            {
                src[dstIdx++] = ch;
                lastWasWS = true;
            }
        }
        else
        { 
            lastWasWS = false;
            src[dstIdx++] = ch;
        }
    }
    return new string(src, 0, dstIdx);
}

बेंचमार्क ...

कोडप्रोक 2015 पर फेलिप मैकडो द्वारा InPlaceCharArraySpaceOnly और मल्टी-स्पेस हटाने के लिए Sunsetquest द्वारा संशोधित। समय: 3.75 टिक्स

InPlaceCharArray द्वारा फेलिप मचाडो 2015 और थोड़ा बहु अंतरिक्ष हटाने के लिए Sunsetquest द्वारा संशोधित। समय 6.50 टिक्स (टैब का भी समर्थन करता है)

जॉन स्कीट द्वारा स्प्लिटएंडजॉयऑनस्पेस । समय: 13.25 टिक्स

StubBuilder द्वारा fubo समय: 13.5 टिक्स (टैब का भी समर्थन करता है)

जॉन स्कीट द्वारा संकलन के साथ रेगेक्स । समय: 17 टिक्स

StringBuilder डेविड एस 2013 समय द्वारा: 30.5 टिक्स

ब्रैंडन टाइम द्वारा गैर-संकलन के साथ रेगेक्स : 63.25 टिक्स

StringBuilder उपयोगकर्ता द्वारा 214147 समय: 77.125 टिक्स

नॉन-कंपाइल टिम हूलिहान टाइम के साथ रेगेक्स : 147.25 टिक्स

बेंचमार्क कोड ...

using System;
using System.Text.RegularExpressions;
using System.Diagnostics;
using System.Threading;
using System.Text;

static class Program
{
    public static void Main(string[] args)
    {
    long seed = ConfigProgramForBenchmarking();

    Stopwatch sw = new Stopwatch();

    string warmup = "This is   a Warm  up function for best   benchmark results." + seed;
    string input1 = "Hello World,    how are   you           doing?" + seed;
    string input2 = "It\twas\t \tso    nice  to\t\t see you \tin 1950.  \t" + seed;
    string correctOutput1 = "Hello World, how are you doing?" + seed;
    string correctOutput2 = "It\twas\tso nice to\tsee you in 1950. " + seed;
    string output1,output2;

    //warm-up timer function
    sw.Restart();
    sw.Stop();

    sw.Restart();
    sw.Stop();
    long baseVal = sw.ElapsedTicks;

    // InPlace Replace by Felipe Machado but modified by Ryan for multi-space removal (http://www.codeproject.com/Articles/1014073/Fastest-method-to-remove-all-whitespace-from-Strin)
    output1 = InPlaceCharArraySpaceOnly (warmup);
    sw.Restart();
    output1 = InPlaceCharArraySpaceOnly (input1);
    output2 = InPlaceCharArraySpaceOnly (input2);
    sw.Stop();
    Console.WriteLine("InPlaceCharArraySpaceOnly : " + (sw.ElapsedTicks - baseVal));
    Console.WriteLine("  Trial1:(spaces only) " + (output1 == correctOutput1 ? "PASS " : "FAIL "));
    Console.WriteLine("  Trial2:(spaces+tabs) " + (output2 == correctOutput2 ? "PASS " : "FAIL "));

    // InPlace Replace by Felipe R. Machado and slightly modified by Ryan for multi-space removal (http://www.codeproject.com/Articles/1014073/Fastest-method-to-remove-all-whitespace-from-Strin)
    output1 = InPlaceCharArray(warmup);
    sw.Restart();
    output1 = InPlaceCharArray(input1);
    output2 = InPlaceCharArray(input2);
    sw.Stop();
    Console.WriteLine("InPlaceCharArray: " + (sw.ElapsedTicks - baseVal));
    Console.WriteLine("  Trial1:(spaces only) " + (output1 == correctOutput1 ? "PASS " : "FAIL "));
    Console.WriteLine("  Trial2:(spaces+tabs) " + (output2 == correctOutput2 ? "PASS " : "FAIL "));

    //Regex with non-compile Tim Hoolihan (https://stackoverflow.com/a/1279874/2352507)
    string cleanedString = 
    output1 = Regex.Replace(warmup, @"\s+", " ");
    sw.Restart();
    output1 = Regex.Replace(input1, @"\s+", " ");
    output2 = Regex.Replace(input2, @"\s+", " ");
    sw.Stop();
    Console.WriteLine("Regex by Tim Hoolihan: " + (sw.ElapsedTicks - baseVal));
    Console.WriteLine("  Trial1:(spaces only) " + (output1 == correctOutput1 ? "PASS " : "FAIL "));
    Console.WriteLine("  Trial2:(spaces+tabs) " + (output2 == correctOutput2 ? "PASS " : "FAIL "));

    //Regex with compile by Jon Skeet (https://stackoverflow.com/a/1280227/2352507)
    output1 = MultipleSpaces.Replace(warmup, " ");
    sw.Restart();
    output1 = MultipleSpaces.Replace(input1, " ");
    output2 = MultipleSpaces.Replace(input2, " ");
    sw.Stop();
    Console.WriteLine("Regex with compile by Jon Skeet: " + (sw.ElapsedTicks - baseVal));
    Console.WriteLine("  Trial1:(spaces only) " + (output1 == correctOutput1 ? "PASS " : "FAIL "));
    Console.WriteLine("  Trial2:(spaces+tabs) " + (output2 == correctOutput2 ? "PASS " : "FAIL "));

    //Split And Join by Jon Skeet (https://stackoverflow.com/a/1280227/2352507)
    output1 = SplitAndJoinOnSpace(warmup);
    sw.Restart();
    output1 = SplitAndJoinOnSpace(input1);
    output2 = SplitAndJoinOnSpace(input2);
    sw.Stop();
    Console.WriteLine("Split And Join by Jon Skeet: " + (sw.ElapsedTicks - baseVal));
    Console.WriteLine("  Trial1:(spaces only) " + (output1 == correctOutput1 ? "PASS " : "FAIL "));
    Console.WriteLine("  Trial2:(spaces+tabs) " + (output2 == correctOutput2 ? "PASS " : "FAIL "));

    //Regex by Brandon (https://stackoverflow.com/a/1279878/2352507
    output1 = Regex.Replace(warmup, @"\s{2,}", " ");
    sw.Restart();
    output1 = Regex.Replace(input1, @"\s{2,}", " ");
    output2 = Regex.Replace(input2, @"\s{2,}", " ");
    sw.Stop();
    Console.WriteLine("Regex by Brandon: " + (sw.ElapsedTicks - baseVal));
    Console.WriteLine("  Trial1:(spaces only) " + (output1 == correctOutput1 ? "PASS " : "FAIL "));
    Console.WriteLine("  Trial2:(spaces+tabs) " + (output2 == correctOutput2 ? "PASS " : "FAIL "));

    //StringBuilder by user214147 (https://stackoverflow.com/a/2156660/2352507
    output1 = user214147(warmup);
    sw.Restart();
    output1 = user214147(input1);
    output2 = user214147(input2);
    sw.Stop();
    Console.WriteLine("StringBuilder by user214147: " + (sw.ElapsedTicks - baseVal));
    Console.WriteLine("  Trial1:(spaces only) " + (output1 == correctOutput1 ? "PASS " : "FAIL "));
    Console.WriteLine("  Trial2:(spaces+tabs) " + (output2 == correctOutput2 ? "PASS " : "FAIL "));

    //StringBuilder by fubo (https://stackoverflow.com/a/27502353/2352507
    output1 = fubo(warmup);
    sw.Restart();
    output1 = fubo(input1);
    output2 = fubo(input2);
    sw.Stop();
    Console.WriteLine("StringBuilder by fubo: " + (sw.ElapsedTicks - baseVal));
    Console.WriteLine("  Trial1:(spaces only) " + (output1 == correctOutput1 ? "PASS " : "FAIL "));
    Console.WriteLine("  Trial2:(spaces+tabs) " + (output2 == correctOutput2 ? "PASS " : "FAIL "));


    //StringBuilder by David S 2013 (https://stackoverflow.com/a/16035044/2352507)
    output1 = SingleSpacedTrim(warmup);
    sw.Restart();
    output1 = SingleSpacedTrim(input1);
    output2 = SingleSpacedTrim(input2);
    sw.Stop();
    Console.WriteLine("StringBuilder(SingleSpacedTrim) by David S: " + (sw.ElapsedTicks - baseVal));
    Console.WriteLine("  Trial1:(spaces only) " + (output1 == correctOutput1 ? "PASS " : "FAIL "));
    Console.WriteLine("  Trial2:(spaces+tabs) " + (output2 == correctOutput2 ? "PASS " : "FAIL "));
}

// InPlace Replace by Felipe Machado and slightly modified by Ryan for multi-space removal (http://www.codeproject.com/Articles/1014073/Fastest-method-to-remove-all-whitespace-from-Strin)
static string InPlaceCharArray(string str)
{
    var len = str.Length;
    var src = str.ToCharArray();
    int dstIdx = 0;
    bool lastWasWS = false;
    for (int i = 0; i < len; i++)
    {
        var ch = src[i];
        if (src[i] == '\u0020')
        {
            if (lastWasWS == false)
            {
                src[dstIdx++] = ch;
                lastWasWS = true;
            }
        }
        else
        { 
            lastWasWS = false;
            src[dstIdx++] = ch;
        }
    }
    return new string(src, 0, dstIdx);
}

// InPlace Replace by Felipe R. Machado but modified by Ryan for multi-space removal (http://www.codeproject.com/Articles/1014073/Fastest-method-to-remove-all-whitespace-from-Strin)
static string InPlaceCharArraySpaceOnly (string str)
{
    var len = str.Length;
    var src = str.ToCharArray();
    int dstIdx = 0;
    bool lastWasWS = false; //Added line
    for (int i = 0; i < len; i++)
    {
        var ch = src[i];
        switch (ch)
        {
            case '\u0020': //SPACE
            case '\u00A0': //NO-BREAK SPACE
            case '\u1680': //OGHAM SPACE MARK
            case '\u2000': // EN QUAD
            case '\u2001': //EM QUAD
            case '\u2002': //EN SPACE
            case '\u2003': //EM SPACE
            case '\u2004': //THREE-PER-EM SPACE
            case '\u2005': //FOUR-PER-EM SPACE
            case '\u2006': //SIX-PER-EM SPACE
            case '\u2007': //FIGURE SPACE
            case '\u2008': //PUNCTUATION SPACE
            case '\u2009': //THIN SPACE
            case '\u200A': //HAIR SPACE
            case '\u202F': //NARROW NO-BREAK SPACE
            case '\u205F': //MEDIUM MATHEMATICAL SPACE
            case '\u3000': //IDEOGRAPHIC SPACE
            case '\u2028': //LINE SEPARATOR
            case '\u2029': //PARAGRAPH SEPARATOR
            case '\u0009': //[ASCII Tab]
            case '\u000A': //[ASCII Line Feed]
            case '\u000B': //[ASCII Vertical Tab]
            case '\u000C': //[ASCII Form Feed]
            case '\u000D': //[ASCII Carriage Return]
            case '\u0085': //NEXT LINE
                if (lastWasWS == false) //Added line
                {
                    src[dstIdx++] = ch; //Added line
                    lastWasWS = true; //Added line
                }
            continue;
            default:
                lastWasWS = false; //Added line 
                src[dstIdx++] = ch;
                break;
        }
    }
    return new string(src, 0, dstIdx);
}

static readonly Regex MultipleSpaces =
    new Regex(@" {2,}", RegexOptions.Compiled);

//Split And Join by Jon Skeet (https://stackoverflow.com/a/1280227/2352507)
static string SplitAndJoinOnSpace(string input)
{
    string[] split = input.Split(new char[] { ' '}, StringSplitOptions.RemoveEmptyEntries);
    return string.Join(" ", split);
}

//StringBuilder by user214147 (https://stackoverflow.com/a/2156660/2352507
public static string user214147(string S)
{
    string s = S.Trim();
    bool iswhite = false;
    int iwhite;
    int sLength = s.Length;
    StringBuilder sb = new StringBuilder(sLength);
    foreach (char c in s.ToCharArray())
    {
        if (Char.IsWhiteSpace(c))
        {
            if (iswhite)
            {
                //Continuing whitespace ignore it.
                continue;
            }
            else
            {
                //New WhiteSpace

                //Replace whitespace with a single space.
                sb.Append(" ");
                //Set iswhite to True and any following whitespace will be ignored
                iswhite = true;
            }
        }
        else
        {
            sb.Append(c.ToString());
            //reset iswhitespace to false
            iswhite = false;
        }
    }
    return sb.ToString();
}

//StringBuilder by fubo (https://stackoverflow.com/a/27502353/2352507
public static string fubo(this string Value)
{
    StringBuilder sbOut = new StringBuilder();
    if (!string.IsNullOrEmpty(Value))
    {
        bool IsWhiteSpace = false;
        for (int i = 0; i < Value.Length; i++)
        {
            if (char.IsWhiteSpace(Value[i])) //Comparison with WhiteSpace
            {
                if (!IsWhiteSpace) //Comparison with previous Char
                {
                    sbOut.Append(Value[i]);
                    IsWhiteSpace = true;
                }
            }
            else
            {
                IsWhiteSpace = false;
                sbOut.Append(Value[i]);
            }
        }
    }
    return sbOut.ToString();
}

//David S. 2013 (https://stackoverflow.com/a/16035044/2352507)
public static String SingleSpacedTrim(String inString)
{
    StringBuilder sb = new StringBuilder();
    Boolean inBlanks = false;
    foreach (Char c in inString)
    {
        switch (c)
        {
            case '\r':
            case '\n':
            case '\t':
            case ' ':
                if (!inBlanks)
                {
                    inBlanks = true;
                    sb.Append(' ');
                }
                continue;
            default:
                inBlanks = false;
                sb.Append(c);
                break;
        }
    }
    return sb.ToString().Trim();
}

/// <summary>
/// We want to run this item with max priory to lower the odds of
/// the OS from doing program context switches in the middle of our code. 
/// source:https://stackoverflow.com/a/16157458 
/// </summary>
/// <returns>random seed</returns>
private static long ConfigProgramForBenchmarking()
{
    //prevent the JIT Compiler from optimizing Fkt calls away
    long seed = Environment.TickCount;
    //use the second Core/Processor for the test
    Process.GetCurrentProcess().ProcessorAffinity = new IntPtr(2);
    //prevent "Normal" Processes from interrupting Threads
    Process.GetCurrentProcess().PriorityClass = ProcessPriorityClass.High;
    //prevent "Normal" Threads from interrupting this thread
    Thread.CurrentThread.Priority = ThreadPriority.Highest;
    return seed;
}

}

बेंचमार्क नोट: रिलीज़ मोड, नो-डीबगर संलग्न, i7 प्रोसेसर, 4 रन का एवीजी, केवल छोटे तारों का परीक्षण किया गया


1
यहाँ संदर्भित मेरे लेख को देखकर अच्छा लगा! (मैं फेलिप मैकडो हूं) मैं बेंचमार्कडॉटनेट नामक एक उचित बेंचमार्क टूल का उपयोग करके इसे अपडेट करने वाला हूं! मैं सभी रनटाइम्स में रन बनाने की कोशिश करूँगा (अब जब हमारे पास नेट नेट और लाइक हैं ...
लाउडेनवियर

1
@ लॉडनवीयर - इस पर अच्छा काम। तुम्हारा लगभग 400% से तेज था! .Net कोर एक मुक्त 150-200% प्रदर्शन को बढ़ावा देने जैसा है। यह c ++ प्रदर्शन के करीब हो रहा है लेकिन कोड के लिए बहुत आसान है। टिप्पणी के लिए धन्यवाद।
सूर्यास्त के समय

यह केवल रिक्त स्थान करता है, अन्य श्वेत स्थान वर्ण नहीं। हो सकता है कि आप src [i] == '\ u0020' के बजाय char.IsWiteSpace (ch) चाहते हैं। मैंने देखा कि यह समुदाय द्वारा संपादित किया गया है। क्या उन्होंने इसे बोर किया?
ईविल कबूतर

3

मैं जो उपयोग कर रहा हूं उसे साझा कर रहा हूं, क्योंकि ऐसा प्रतीत होता है कि मैं कुछ अलग लेकर आया हूं। मैं इसे थोड़ी देर के लिए इस्तेमाल कर रहा हूं और यह मेरे लिए काफी तेज है। मुझे यकीन नहीं है कि यह दूसरों के खिलाफ कैसे ढेर हो जाता है। मैं इसे एक सीमांकित फ़ाइल लेखक में उपयोग करता हूं और इसके माध्यम से एक समय में बड़े डेटाटैबल्स को एक क्षेत्र में चलाता हूं।

    public static string NormalizeWhiteSpace(string S)
    {
        string s = S.Trim();
        bool iswhite = false;
        int iwhite;
        int sLength = s.Length;
        StringBuilder sb = new StringBuilder(sLength);
        foreach(char c in s.ToCharArray())
        {
            if(Char.IsWhiteSpace(c))
            {
                if (iswhite)
                {
                    //Continuing whitespace ignore it.
                    continue;
                }
                else
                {
                    //New WhiteSpace

                    //Replace whitespace with a single space.
                    sb.Append(" ");
                    //Set iswhite to True and any following whitespace will be ignored
                    iswhite = true;
                }  
            }
            else
            {
                sb.Append(c.ToString());
                //reset iswhitespace to false
                iswhite = false;
            }
        }
        return sb.ToString();
    }

2

जॉन स्कीट ने परीक्षण कार्यक्रम का उपयोग करते हुए, मैंने यह देखने की कोशिश की कि क्या मुझे हाथ से लिखा हुआ लूप तेज चलाने के लिए मिल सकता है।
मैं हर बार NormalizeWithSplitAndJoin को हरा सकता हूं, लेकिन 1000, 5 के इनपुट के साथ केवल NormalizeWithRegex को हरा सकता हूं।

static string NormalizeWithLoop(string input)
{
    StringBuilder output = new StringBuilder(input.Length);

    char lastChar = '*';  // anything other then space 
    for (int i = 0; i < input.Length; i++)
    {
        char thisChar = input[i];
        if (!(lastChar == ' ' && thisChar == ' '))
            output.Append(thisChar);

        lastChar = thisChar;
    }

    return output.ToString();
}

मैंने उस मशीन कोड को नहीं देखा है जिसमें घबराना पैदा होता है, हालांकि मुझे उम्मीद है कि समस्या कॉलिंग द्वारा StringBuilder.Append () और बहुत बेहतर करने के लिए असुरक्षित कोड के उपयोग की आवश्यकता होगी।

तो Regex.Replace () हरा करने के लिए बहुत तेज और कठिन है !!


2

VB.NET

Linha.Split(" ").ToList().Where(Function(x) x <> " ").ToArray

सी#

Linha.Split(" ").ToList().Where(x => x != " ").ToArray();

LINQ = D की शक्ति का आनंद लें


बिल्कुल सही! मेरे लिए यह सबसे सुरुचिपूर्ण दृष्टिकोण है। तो रिकॉर्ड के लिए, C # में वह होगा:string.Join(" ", myString.Split(' ').Where(s => s != " ").ToArray())
Efrain

1
Splitसभी व्हाट्सएप को पकड़ने और Whereक्लॉज को हटाने के लिए मामूली सुधार :myString.Split(null as char[], StringSplitOptions.RemoveEmptyEntries)
डेविड

1
Regex regex = new Regex(@"\W+");
string outputString = regex.Replace(inputString, " ");

यह अंतरिक्ष के साथ सभी गैर-शब्द वर्णों को प्रतिस्थापित करता है। तो यह कोष्ठक और उद्धरण आदि जैसी चीजों को भी बदल देगा, जो कि आप नहीं चाहते हैं।
हरमन

0

सबसे छोटा समाधान:

var regExp = / \ s + / g, newString = oldString.replace (regExp, '');


0

आप यह कोशिश कर सकते हैं:

    /// <summary>
    /// Remove all extra spaces and tabs between words in the specified string!
    /// </summary>
    /// <param name="str">The specified string.</param>
    public static string RemoveExtraSpaces(string str)
    {
        str = str.Trim();
        StringBuilder sb = new StringBuilder();
        bool space = false;
        foreach (char c in str)
        {
            if (char.IsWhiteSpace(c) || c == (char)9) { space = true; }
            else { if (space) { sb.Append(' '); }; sb.Append(c); space = false; };
        }
        return sb.ToString();
    }

0

रिप्लेसमेंट समूह एक ही के साथ कई श्वेत अंतरिक्ष वर्णों के प्रतिस्थापन का संकेत प्रदान करते हैं :

    public static void WhiteSpaceReduce()
    {
        string t1 = "a b   c d";
        string t2 = "a b\n\nc\nd";

        Regex whiteReduce = new Regex(@"(?<firstWS>\s)(?<repeatedWS>\k<firstWS>+)");
        Console.WriteLine("{0}", t1);
        //Console.WriteLine("{0}", whiteReduce.Replace(t1, x => x.Value.Substring(0, 1))); 
        Console.WriteLine("{0}", whiteReduce.Replace(t1, @"${firstWS}"));
        Console.WriteLine("\nNext example ---------");
        Console.WriteLine("{0}", t2);
        Console.WriteLine("{0}", whiteReduce.Replace(t2, @"${firstWS}"));
        Console.WriteLine();
    }

कृपया ध्यान दें कि दूसरा उदाहरण एकल रखता है \nजबकि स्वीकृत उत्तर अंतरिक्ष के साथ पंक्ति के अंत को प्रतिस्थापित करेगा।

यदि आपको पहले एक के साथ सफेद अंतरिक्ष वर्णों के किसी भी संयोजन को बदलने की आवश्यकता है , \kतो पैटर्न से केवल बैक-संदर्भ हटा दें ।


0

नियमित अभिव्यक्ति का उपयोग करना, एकल स्थान के साथ 2 या अधिक श्वेत रिक्त स्थान को बदलने के लिए भी एक अच्छा समाधान है।

हम regex पैटर्न का उपयोग “ s + ” के रूप में कर रहे हैं ।

  • \ n एक स्पेस, टैब, नई लाइन, कैरिज रिटर्न, फॉर्म फीड या वर्टिकल टैब से मेल खाता है।

  • '+' एक या अधिक घटनाएँ कहता है।

रेगेक्स उदाहरण

String blogName = "  Sourav .  Pal.   "

 String nameWithProperSpacing = blogName.replaceAll("\\s+", " ");   
System.out.println( nameWithProperSpacing );

-1

ऐसा करने के लिए कोई रास्ता नहीं बनाया गया है। आप यह कोशिश कर सकते हैं:

private static readonly char[] whitespace = new char[] { ' ', '\n', '\t', '\r', '\f', '\v' };
public static string Normalize(string source)
{
   return String.Join(" ", source.Split(whitespace, StringSplitOptions.RemoveEmptyEntries));
}

यह व्हाट्सएप के प्रमुख और अनुगामी व्हाट्सएप को हटा देगा और साथ ही किसी भी व्हॉट्सएप चरित्र को किसी भी आंतरिक व्हाट्सएप को ध्वस्त कर देगा। यदि आप वास्तव में केवल रिक्त स्थान को गिराना चाहते हैं, तो एक नियमित अभिव्यक्ति का उपयोग करने वाले समाधान बेहतर हैं; अन्यथा यह समाधान बेहतर है। ( जॉन स्कीट द्वारा किया गया विश्लेषण देखें ।)


7
यदि नियमित अभिव्यक्ति संकलित और कैश की गई है, तो मुझे यकीन नहीं है कि विभाजन और जुड़ने की तुलना में अधिक ओवरहेड है, जो मध्यवर्ती कचरे के तारों का भार पैदा कर सकता है । क्या आपने यह मानने से पहले कि आपके रास्ते में तेजी है, क्या आपने दोनों दृष्टिकोणों का सावधानीपूर्वक अध्ययन किया है?
जॉन स्कीट


3
ओवरहेड की बात करते हुए, पृथ्वी पर आप क्यों बुला रहे हैं source.ToCharArray()और फिर परिणाम को फेंक रहे हैं?
जॉन स्कीट

2
औरToCharArray() string.Join के परिणाम पर कॉल करना, केवल एक नया स्ट्रिंग बनाने के लिए ... वाह, इसके लिए एक पोस्ट में ओवरहेड की शिकायत करना केवल उल्लेखनीय है। -1।
जॉन स्कीट

1
ओह, और मान whitespaceलेना new char[] { ' ' }, यह गलत परिणाम देगा यदि इनपुट स्ट्रिंग शुरू होती है या एक स्थान के साथ समाप्त होती है।
जॉन स्कीट
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.