码迷,mamicode.com
首页 > 其他好文 > 详细

使用泛型编写通用的C#预处理类型转换方法

时间:2014-07-20 21:39:44      阅读:299      评论:0      收藏:0      [点我收藏+]

标签:style   blog   http   color   使用   strong   

废话

从.net3.5发布后,就很多前辈利用泛型创造出了很多很有趣的代码,通用的转换方法也被写的烂了,小弟不才,今天又来写一遍,只是为了做个人知识的管理和追赶大牛的步伐,请各位路过的大大多多批评指正。

思路

1、基本类型都实现了IConvertible这个接口

2、基本类型都实现了TryParse方法

实现

public static class Converter
    {
        /// <summary>
        /// 转换为其他继承IConvertible的类型
        /// </summary>
        /// <typeparam name="T">转换的类型</typeparam>
        /// <param name="value">要转换的值</param>
        /// <param name="success">是否成功</param>
        /// <returns></returns>
        public static T To<T>(this IConvertible value, out bool success) where T : IConvertible
        {
            if (value == null)
            {
                success = true;

                return default(T);
            }

            Type tResult = typeof(T);

            if (tResult == typeof(string))
            {
                success = true;

                return (T)(object)value.ToString();
            }

            MethodInfo mTryParse = tResult.GetMethod("TryParse", BindingFlags.Public | BindingFlags.Static, Type.DefaultBinder, new Type[] { typeof(string), tResult.MakeByRefType() }, new ParameterModifier[] { new ParameterModifier(2) });
            
            var parameters = new object[]{value.ToString(), default(T)};

            success = (bool)mTryParse.Invoke(null, parameters);

            return success ? (T)parameters[1] : default(T);
        }

        /// <summary>
        /// 转换为其他继承IConvertible的类型
        /// </summary>
        /// <typeparam name="T">转换的类型</typeparam>
        /// <param name="value">要转换的值</param>
        /// <returns></returns>
        public static T To<T>(this IConvertible value) where T : IConvertible
        {
            bool success;
            return To<T>(value, out success);
        }
    }

单元测试

[TestClass]
    public class UnitTessConverter
    {
        [TestMethod]
        public void TestTo()
        {
            int i = 1;
            double dResult = i.To<double>();
            Assert.AreEqual(i, 1d);
            Assert.AreEqual(1, i.To<char>());

            double d = 1.1d;
            int iResult = d.To<int>();
            Assert.AreEqual(0, iResult);
            float fResult = d.To<float>();
            Assert.AreEqual(1.1f, fResult);

            d = 1d;
            Assert.AreEqual(1, d.To<int>());

            float f = 1.1f;
            iResult = f.To<int>();
            Assert.AreEqual(0, iResult);

            string str = "1.1";
            Assert.AreEqual(1.1f, str.To<float>());
            Assert.AreEqual(1.1d, str.To<double>());
            Assert.AreEqual((decimal)1.1, str.To<decimal>());



            str = "1990-10-1 12:00";
            Assert.AreEqual(new DateTime(1990, 10, 1, 12, 0, 0), str.To<DateTime>());

            str = "100dd";
            bool success;
            Assert.AreEqual(DateTime.MinValue, str.To<DateTime>(out success));
            Assert.IsFalse(success);
            Assert.AreEqual(0, str.To<int>(out success));
            Assert.IsFalse(success);
            Assert.AreEqual(0, str.To<double>(out success));
            Assert.IsFalse(success);
            Assert.AreEqual(\0, str.To<char>(out success));
            Assert.IsFalse(success);


            str = null;
            fResult = str.To<float>();
            Assert.AreEqual(0f, fResult);

            Assert.AreEqual("Hibernating", MachineState.Hibernating.To<string>());
            Assert.AreEqual(0, MachineState.PowerOff.To<int>());
        }

        enum MachineState
        {
            PowerOff = 0,
            Running = 5,
            Sleeping = 10,
            Hibernating = Sleeping + 5
        }
    }

测试通过

bubuko.com,布布扣

运行效率测试

电脑配置:

bubuko.com,布布扣

效率测试代码:

class Program
    {
        static void Main(string[] args)
        {
            System.Diagnostics.Stopwatch st = new System.Diagnostics.Stopwatch();
            st.Start();
            for (int i = 0; i < 1000000; i++)
            {
                i.To<string>().To<double>().To<float>();
            }
            st.Stop();
            Console.WriteLine(st.ElapsedMilliseconds);
            Console.Read();

        }
    }

第一次:19639

第二次:19414

第三次:19262

 

优化

在上面的To方法中,用上了反射,反射是性能杀手,要尽量避免使用,所以我想到了把反射得到的“TryParse”的MethodInfo对象保存起来。

优化后的代码:

public static class Converter
    {
        /// <summary>
        /// 转换为其他继承IConvertible的类型
        /// </summary>
        /// <typeparam name="T">转换的类型</typeparam>
        /// <param name="value">要转换的值</param>
        /// <param name="success">是否成功</param>
        /// <returns></returns>
        public static T To<T>(this IConvertible value, out bool success) where T : IConvertible
        {
            if (value == null)
            {
                success = true;

                return default(T);
            }

            Type tResult = typeof(T);

            if (tResult == typeof(string))
            {
                success = true;

                return (T)(object)value.ToString();
            }

            MethodInfo mTryParse;

            if (_TryParse.ContainsKey(tResult.FullName))
            {
                mTryParse = _TryParse[tResult.FullName];
            }
            else
            {
                mTryParse = tResult.GetMethod("TryParse", BindingFlags.Public | BindingFlags.Static, Type.DefaultBinder, new Type[] { typeof(string), tResult.MakeByRefType() }, new ParameterModifier[] { new ParameterModifier(2) });

                _TryParse.Add(tResult.FullName, mTryParse);
            }

            var parameters = new object[]{value.ToString(), default(T)};

            success = (bool)mTryParse.Invoke(null, parameters);

            return success ? (T)parameters[1] : default(T);
        }

        /// <summary>
        /// 转换为其他继承IConvertible的类型
        /// </summary>
        /// <typeparam name="T">转换的类型</typeparam>
        /// <param name="value">要转换的值</param>
        /// <returns></returns>
        public static T To<T>(this IConvertible value) where T : IConvertible
        {
            bool success;
            return To<T>(value, out success);
        }

        private static Dictionary<string, MethodInfo> _TryParse = new Dictionary<string, MethodInfo>(); 
    }

重新运行单元测试,得到结果是通过

重新运行“效率测试代码”

第一次:11836

第二次:12170

第三次:11866

 

至此结束了这篇文章,望各位大大多多指点。

使用泛型编写通用的C#预处理类型转换方法,布布扣,bubuko.com

使用泛型编写通用的C#预处理类型转换方法

标签:style   blog   http   color   使用   strong   

原文地址:http://www.cnblogs.com/sheepswallow/p/3856388.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!