एक पुराने प्रश्न को पुनर्जीवित करने के लिए नहीं, लेकिन लगा कि मैं कम से कम थोड़ा आसान तरीका पेश कर सकता हूं, अगर कोई सेटअप अधिक जटिल हो।
इसलिए यदि हम एक नया कस्टम फॉर्मेटर बनाते हैं तो हम string.Format
बिना अपना फोन नंबर बदले ए को सरल स्वरूपण का उपयोग कर सकते हैंlong
तो पहले कस्टम फॉर्मेटर बनाने की सुविधा देता है:
using System;
using System.Globalization;
using System.Text;
namespace System
{
/// <summary>
/// A formatter that will apply a format to a string of numeric values.
/// </summary>
/// <example>
/// The following example converts a string of numbers and inserts dashes between them.
/// <code>
/// public class Example
/// {
/// public static void Main()
/// {
/// string stringValue = "123456789";
///
/// Console.WriteLine(String.Format(new NumericStringFormatter(),
/// "{0} (formatted: {0:###-##-####})",stringValue));
/// }
/// }
/// // The example displays the following output:
/// // 123456789 (formatted: 123-45-6789)
/// </code>
/// </example>
public class NumericStringFormatter : IFormatProvider, ICustomFormatter
{
/// <summary>
/// Converts the value of a specified object to an equivalent string representation using specified format and
/// culture-specific formatting information.
/// </summary>
/// <param name="format">A format string containing formatting specifications.</param>
/// <param name="arg">An object to format.</param>
/// <param name="formatProvider">An object that supplies format information about the current instance.</param>
/// <returns>
/// The string representation of the value of <paramref name="arg" />, formatted as specified by
/// <paramref name="format" /> and <paramref name="formatProvider" />.
/// </returns>
/// <exception cref="System.NotImplementedException"></exception>
public string Format(string format, object arg, IFormatProvider formatProvider)
{
var strArg = arg as string;
// If the arg is not a string then determine if it can be handled by another formatter
if (strArg == null)
{
try
{
return HandleOtherFormats(format, arg);
}
catch (FormatException e)
{
throw new FormatException(string.Format("The format of '{0}' is invalid.", format), e);
}
}
// If the format is not set then determine if it can be handled by another formatter
if (string.IsNullOrEmpty(format))
{
try
{
return HandleOtherFormats(format, arg);
}
catch (FormatException e)
{
throw new FormatException(string.Format("The format of '{0}' is invalid.", format), e);
}
}
var sb = new StringBuilder();
var i = 0;
foreach (var c in format)
{
if (c == '#')
{
if (i < strArg.Length)
{
sb.Append(strArg[i]);
}
i++;
}
else
{
sb.Append(c);
}
}
return sb.ToString();
}
/// <summary>
/// Returns an object that provides formatting services for the specified type.
/// </summary>
/// <param name="formatType">An object that specifies the type of format object to return.</param>
/// <returns>
/// An instance of the object specified by <paramref name="formatType" />, if the
/// <see cref="T:System.IFormatProvider" /> implementation can supply that type of object; otherwise, null.
/// </returns>
public object GetFormat(Type formatType)
{
// Determine whether custom formatting object is requested.
return formatType == typeof(ICustomFormatter) ? this : null;
}
private string HandleOtherFormats(string format, object arg)
{
if (arg is IFormattable)
return ((IFormattable)arg).ToString(format, CultureInfo.CurrentCulture);
else if (arg != null)
return arg.ToString();
else
return string.Empty;
}
}
}
तो अगर आप इसका उपयोग करना चाहते हैं तो आप कुछ ऐसा करेंगे:
String.Format(new NumericStringFormatter(),"{0:###-###-####}", i["MyPhone"].ToString());
सोचने के लिए कुछ अन्य बातें:
अभी अगर आपने एक लंबा फॉर्मैटर निर्दिष्ट किया है, तो आपने एक स्ट्रिंग को प्रारूपित करने के लिए किया है, यह सिर्फ अतिरिक्त # संकेतों को अनदेखा करेगा। उदाहरण के लिएString.Format(new NumericStringFormatter(),"{0:###-###-####}", "12345");
परिणाम 123-45 होगा - इसलिए आप चाहते हैं कि यह निर्माणकर्ता में किसी प्रकार के संभावित भराव चरित्र को ले जाए।
इसके अलावा, मैंने एक # चिह्न से बचने का एक तरीका प्रदान नहीं किया है, यदि आप इसे अपने आउटपुट स्ट्रिंग में शामिल करना चाहते हैं तो आप उस तरह से नहीं कर पाएंगे जैसे अभी है।
कारण यह है कि मैं रेगेक्स पर इस पद्धति को पसंद करता हूं, मुझे अक्सर उपयोगकर्ताओं को प्रारूप को निर्दिष्ट करने की अनुमति देने की आवश्यकता होती है और उपयोगकर्ता रेगेक्स को पढ़ाने की कोशिश करने की तुलना में इस प्रारूप का उपयोग करने की व्याख्या करना मेरे लिए काफी आसान है।
साथ ही वर्ग का नाम थोड़ा गलत है क्योंकि यह वास्तव में किसी भी स्ट्रिंग को प्रारूपित करने के लिए काम करता है जब तक आप इसे उसी क्रम में रखना चाहते हैं और इसके अंदर वर्णों को इंजेक्ट करते हैं।