记一次.NET代码重构(下)

public override bool TryGetMember(GetMemberBinder binder, out object result){            if (!_dictionary.TryGetValue(binder.Name, out result))            {                result = null;                return true;            }            var dictionary = result as IDictionary;            if (dictionary != null)            {                result = new DynamicJsonObject(dictionary);                return true;            }            var arrayList = result as ArrayList;            if (arrayList != null && arrayList.Count > 0)            {                if (arrayList[0] is IDictionary)result = new List(arrayList.Cast<IDictionary>().Select(x => new DynamicJsonObject(x)));                else                    result = new List(arrayList.Cast());            }            return true;        }    }    #endregion}

接下来是getcontent方法,此方法的目的很简单,就是要根据客户传递的模板变量参数键值对和短信模板内容,拼装成最后的短信发送内容,之前此方法里面是硬编码的,现在我们需要变成动态获取。

短信模板的内容示例:

【一应生活】您有一件单号为expressNumbers company,已到communityName收发室,请打开一应生活APP“收发室”获取取件码进行取件。点击下载http://a.app.qq.com/o/simple.jsp?pkgname=com.ening.life

我发现这样的模板内容有问题,模板中的变量参数是直接用的英文单词表示的,而我们的短信内容中可能有时候也会存在英文单词,那么我就给所有的变量参数加上{}。修改后如下:

【一应生活】您有一件单号为{expressNumbers} {company},已到{communityName}收发室,请打开一应生活APP“收发室”获取取件码进行取件。点击下载http://a.app.qq.com/o/simple.jsp?pkgname=com.ening.life

我们需要根据客户传递过来的对象,将短信模板中的变量参数,替换成变量参数对应的值。那么我们首先就要解析这个对象中的键值对信息。

  /// 把object对象的属性反射获取到字典列表中        ///         /// object对象        /// 返回Dictionary(属性名,属性值)列表         static Dictionary GetProperties(object data)        {            Dictionary dict = new Dictionary();            Type type = data.GetType();            string[] propertyNames = type.GetProperties().Select(p => p.Name).ToArray();            foreach (var prop in propertyNames)            {                object propValue = type.GetProperty(prop).GetValue(data, null);                string value = (propValue != null) ? propValue.ToString() : "";                if (!dict.ContainsKey(prop))                {                    dict.Add(prop, value);                }            }            return dict;        }

接下来是通过正则表达式来匹配短信模板内容。

  /// 多个匹配内容        ///         /// 输入内容        /// 表达式字符串        /// 分组名, ""代表不分组        static List GetList(string sInput, string sRegex, string sGroupName)        {            List list = new List();            Regex re = new Regex(sRegex, RegexOptions.IgnoreCase | RegexOptions.IgnorePatternWhitespace | RegexOptions.Multiline);            MatchCollection mcs = re.Matches(sInput);            foreach (Match mc in mcs)            {                if (sGroupName != "")                {                    list.Add(mc.Groups[sGroupName].Value);                }                else                {                    list.Add(mc.Value);                }            }            return list;        }        public static string ReplaceTemplate(string template, object data)        {            var regex = @"{(?.*?)}";            List itemList = GetList(template, regex, "name"); //获取模板变量对象            Dictionary dict = GetProperties(data);            foreach (string item in itemList)            {                //如果属性存在,则替换模板,并修改模板值                if (dict.ContainsKey(item))                {                    template = template.Replace("{"+item+"}", dict.First(x => x.Key == item).Value);                }            }            return template;}

这样就讲客户传递的对象和我们的解析代码进行了解耦,客户传递的对象不再依赖于我们的代码实现,而是依赖于我们数据表中模板内容的配置。

这几个方法我是写好了,顺便弄个单元测试来验证一下是不是我要的效果,可怜的是,这个项目中根本就没用到单元测试,没办法,我自己创建一个单元测试

[TestClass]    public class MatchHelperTest    {        [TestMethod]        public void ReplaceTemplate()        {            //模板文本            var template = "【一应生活】您有一件单号为{expressNumbers} {company},已到{communityName}收发室,            请打开一应生活APP“收发室”获取取件码进行取件。点击下载http://a.app.qq.com/o/simple.jsp?pkgname=com.ening.life";            //数据对象            var data = new { expressNumbers = "2016", company = "长城", communityName = "长怡花园"};            string str = "【一应生活】您有一件单号为2016 长城,已到长怡花园收发室,            请打开一应生活APP“收发室”获取取件码进行取件。点击下载http://a.app.qq.com/o/simple.jsp?pkgname=com.ening.life";            string str1=MatchHelper.ReplaceTemplate(template, data);            Assert.AreEqual(str1,str);            //重复标签的测试            template = "【一应生活】您有一件单号为{expressNumbers} {company},已到{communityName}收发室,单号:{expressNumbers}";            str = "【一应生活】您有一件单号为2016 长城,已到长怡花园收发室,单号:2016";            str1=MatchHelper.ReplaceTemplate(template, data);            Assert.AreEqual(str1, str);        }    }

说到单元测试,我相信在许多公司都没有用起来,理由太多。我也觉得如果业务简单的话,根本没必要写单元测试,国内太多创业型公司项目进度都非常赶,如果说写单元测试不费时间,那绝对是骗人的,至于说写单元测试能提高开发效率,减少返工率,个人感觉这个还真难说,因为即便不写单元测试也还是可以通过许多其它手段来弥补的,个人观点,勿喷。

接下来修改GetContent方法如下:

public string GetContent(dynamic messageContext){            string strMsg = "";            string TypeCode = string.IsNullOrEmpty(messageContext.serviceCode) ? "001" : messageContext.serviceCode;            string channel = messageContext.channel;            try{var Module = unitOfWork.MessageModule.Get(c => c.Type == channel && c.TypeNo == TypeCode).FirstOrDefault();                if (!string.IsNullOrEmpty(Module.Content))                {                    var content = Module.Content;                    strMsg = MatchHelper.ReplaceTemplate(content, messageContext);                }                return strMsg;            }            catch (Exception ex)            {                strMsg = ex.Message;            }            return strMsg;        }

(话外:先吐槽一下之前这个变量命名,MessageContext messageContext 和string messageContent,长得太像了,一开始我重构的时候害我弄错了,建议不要在同一个方法中使用相似的变量名称,以免弄混淆。妈蛋,老司机的我又被坑了,愤怒,无可忍受,果断重命名。)

原来控制器调用业务逻辑代码是直接这样的

MessageModuleBusiness messageModuleBusiness = new MessageModuleBusiness()

依赖于具体类的实现,而我们知道,具体是不稳定的,抽象才是稳定的,我们应该面向接口编程。今天是发送短信,明天可能就是发邮件,又或者要加日志记录等等等。

public interface IMessageModuleBusiness{        ///         /// 组装消息内容        ///         /// 动态参数对象        /// 组装后的消息内容        string GetContent(dynamic messageContext);}

然后调用的代码修改为:

private IMessageModuleBusiness messageModuleBusiness = new MessageModuleBusiness();

最终的externalMerchantSendMessage代码为:

    /// 外部商户发送信息        public ActionResult externalMerchantSendMessage()        {            try            {                dynamic param = null;                string json = Request.QueryString.ToString();                if (Request.QueryString.Count != 0) //ajax get请求                {                    //兼容旧的客户调用写法,暂时硬编了                    if (json.Contains("param."))                    {                        json = json.Replace("param.", "");                    }                    json = "{" + json.Replace("=", ":'").Replace("&", "',") + "'}";                }                else  //ajax Post请求                {Request.InputStream.Position = 0;//切记这里必须设置流的起始位置为0,否则无法读取到数据                    json = new StreamReader(Request.InputStream).ReadToEnd();                }                var serializer = new JavaScriptSerializer();                serializer.RegisterConverters(new[] { new DynamicJsonConverter() });                param = serializer.Deserialize(json, typeof(object));                logger.Info("[externalMerchantSendMessage]param:" + param);                bool isAuth = authModelBusiness.isAuth(param.channel, param.phone, param.sign);                if (!isAuth)                {                    return Json(new Result()                    {                        resultCode = ((int)ResultCode.NoPermission).ToString(),                        resultMsg = "签名或无权限访问"                    }, JsonRequestBehavior.AllowGet);                }                var meaage = messageModuleBusiness.GetContent(param);                if (string.IsNullOrEmpty(meaage))                {                    return Json(new Result()                    {                        resultCode = ((int)ResultCode.failure).ToString(),                        resultMsg = "发送失败"                    }, JsonRequestBehavior.AllowGet);                }                SMSHelper helper = new SMSHelper();                helper.SendSMS(meaage, param.phone); //发送短信                return Json(new Result()                {                    resultCode = ((int)ResultCode.success).ToString(),                    resultMsg = "发送成功"                }, JsonRequestBehavior.AllowGet);            }            catch (Exception ex)            {                return Json(new Result()                {                    resultCode = ((int)ResultCode.failure).ToString(),                    resultMsg = "发送失败"+ex.Message                }, JsonRequestBehavior.AllowGet);            }        }

这样的话,即便日后通过反射或者IOC来再次解耦也方便。

好了,通过这样一步一步的重构,在不修改原有表结构和不影响客户调用的情况下,我已经将变化点进行了封装,当客户的模板参数变量变化的时候,再也不需要变更代码,只需要修改表中的模板内容就可以了。

901.jpg

重构时,画类图是一个非常好的习惯,代码结构一目了然,这里我附上类图。

以上就是记一次.NET代码重构(下)的内容,更多相关内容请关注PHP中文网(www.php.cn)!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1432064.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月17日 06:10:23
下一篇 2025年12月17日 06:10:37

相关推荐

  • 为 Jenkins 配置 .NET 持续集成环境

    去年年底,得益于公司引入 jenkins,让我们在持续集成方面迈出了第一步,本文不赘述如何安装 jenkins,主要关注点在于配置 .net 环境。另外本文是在 windows 环境下安装的 jenkins 进行操作。 一、安装环境 首先我们需要先准备几个安装包,将它们安装到 Windows 上: …

    2025年12月17日
    000
  • c#.net中const和readonly的区别

    (1) readonly和const都是用来标示常量的。(2) 初始化赋值不同。const修饰的常量必须在声明的同时赋值。例如: public class Class1 { public const int MaxValue = 10; //正确声明 public const MInValue; /…

    好文分享 2025年12月17日
    000
  • XML中如何生成XML报表模板_XML生成XML报表模板的方法与示例

    利用XSLT、编程语言或模板引擎可生成XML报表模板:1. XSLT将源XML转换为结构化报表;2. Python等语言通过DOM操作动态构建XML;3. Jinja2等模板引擎支持变量与逻辑控制,实现灵活输出。 在XML中生成XML报表模板,实际上是指利用XML的结构化特性设计一个可复用的数据模板…

    2025年12月17日
    000
  • XML中如何删除指定节点_XML删除指定节点的方法与技巧

    使用DOM、XPath、SAX/StAX或工具库可删除XML指定节点。DOM适合中小文件,通过removeChild()删除目标节点;XPath支持复杂条件精准定位;SAX/StAX流式处理适用于大文件;工具库如ElementTree提供简洁API。选择方法需考虑文件大小与性能需求。 在处理XML文…

    2025年12月17日
    000
  • XML中如何校验XML节点顺序_XML校验XML节点顺序的方法与技巧

    答案:使用XSD的xs:sequence可严格校验XML节点顺序,如FirstName→LastName→Age;若顺序错乱则校验失败。 在XML处理过程中,校验节点顺序是确保数据结构符合预期的重要环节。特别是在与外部系统交互、接口对接或数据导入导出时,严格的节点顺序可能影响解析结果或业务逻辑。虽然…

    2025年12月17日
    000
  • XML中如何处理空白节点_XML处理空白节点的操作步骤

    正确处理XML空白节点需根据解析器设置或编程逻辑过滤非重要空白。例如Java DOM可设setIgnoringElementContentWhitespace(true),Python可预处理移除,.NET可通过PreserveWhitespace=false控制,默认保留空白;也可通过DTD/XS…

    2025年12月17日
    000
  • XML中如何合并XML片段_XML合并XML片段的操作方法与技巧

    正确合并XML片段需先创建统一根节点,再通过编程语言的XML库或XSLT将各片段导入,确保编码、命名空间和属性唯一性,避免字符串拼接以防止结构错误。 在处理XML数据时,经常需要将多个XML片段合并成一个完整的文档。这种操作常见于配置文件整合、数据聚合或服务间通信场景。正确地合并XML片段不仅能保证…

    2025年12月17日
    000
  • XML中如何生成带CDATA节点的XML_XML生成带CDATA节点XML的方法与示例

    使用lxml、Java DOM和C# XmlDocument可生成带CDATA的XML,分别通过etree.CDATA、createCDATASection和CreateCDataSection方法实现,注意避免嵌套及编码问题。 在XML中,CDATA(Character Data)节点用于包裹文本…

    2025年12月17日
    000
  • XML中如何解析嵌套列表属性_XML解析嵌套列表属性的方法与步骤

    解析XML嵌套列表属性需结合DOM遍历、XPath查询与数据封装。首先使用ElementTree或lxml加载XML,通过findall或XPath定位item节点,提取id、type等属性及name、quantity等子元素文本,逐层解析后将结果存为字典列表,便于后续操作。 在处理XML数据时,经…

    2025年12月17日
    000
  • XML中如何解析带DTD的XML_XML解析带DTD的XML的操作方法

    解析带DTD的XML需根据需求启用或关闭验证:Java中通过setValidating(true)开启,Python和.NET需配置支持DTD的解析器;处理外部DTD时应确保路径可访问或使用EntityResolver映射本地文件,注意安全风险;若仅解析结构可关闭验证以提升性能。 解析带有 DTD(…

    2025年12月17日
    000
  • 什么是XML Encryption

    XML Encryption通过加密XML数据保障机密性,支持细粒度加密,利用CEK和KEK双重加密机制,结合和结构实现安全封装,并常与XML Signature协同使用以同时确保机密性、完整性和认证。 XML Encryption 是一种由万维网联盟(W3C)定义的技术标准,它允许我们对整个 XM…

    2025年12月17日
    000
  • XML中如何转化为对象_XML将XML转化为对象的方法与技巧

    使用JAXB将XML转Java对象需添加@XmlRootElement和@XmlElement注解,通过JAXBContext和Unmarshaller解析;2. C#中用XmlSerializer反序列化,类标记[XmlRoot]或[Serializable],调用Deserialize方法读取流…

    2025年12月17日
    000
  • XML注入攻击是什么?如何防范?

    XML注入发生在用户输入被直接拼接进XML文档且未转义特殊字符时,例如输入true可篡改权限结构。防范措施包括:对&等字符进行转义为&;使用DOM、XmlWriter等安全库生成XML避免手动拼接;严格验证输入格式与长度;禁用DTD和外部实体防止XXE攻击;在开发中始终净化所有不可信…

    2025年12月17日
    000
  • XML中如何处理空白节点_XML处理空白节点的技巧与步骤

    正确处理XML空白节点需根据场景选择策略:解析时可通过设置忽略空白、使用XPath精准定位或预处理清洗文本,避免格式化空白影响数据准确性。 在处理XML文档时,空白节点(如换行、空格、制表符等)常常会影响数据解析的准确性。尤其在使用DOM或XPath解析时,这些看似无害的空白可能会被识别为文本节点,…

    2025年12月17日
    000
  • XML配置文件如何设计?常见应用场景?

    设计XML配置文件需遵循语义化命名、合理使用属性与子元素、支持注释和Schema验证等原则,适用于Spring框架、Tomcat配置、Maven构建、Logback日志等场景,强调结构清晰、可扩展性和可维护性,尤其在企业级应用中仍具优势。 XML配置文件的设计核心在于结构清晰、可读性强、易于扩展。它…

    2025年12月17日
    000
  • XML与配置文件热重载如何实现?监听文件变化。

    实现XML配置热重载需监听文件变化、重新解析并安全替换配置。首先利用WatchService等工具监听文件修改事件;检测到变更后,异步重新解析XML,校验语法并对比新旧配置;通过原子引用或双缓冲机制更新内存配置,避免阻塞主线程和频繁抖动;最后通知相关组件同步状态。结合Spring Boot或配置中心…

    2025年12月17日
    000
  • 如何设计XML的国际化方案

    答案:设计XML国际化方案需分离可翻译内容与结构,推荐外部化资源文件并使用UTF-8编码、清晰翻译键、本地化格式处理及自动化工具链,以应对字符编码、上下文丢失、多语言同步等挑战,确保可维护性与扩展性。 设计XML的国际化方案,核心在于将可翻译内容与结构、逻辑分离,并为不同语言提供明确的标识或独立的存…

    2025年12月17日
    000
  • XML中如何处理非法字符_XML处理非法字符的技巧与方法

    XML解析失败常因非法字符导致,需清理控制字符并保留合法范围#x9、#xA、#xD及#x20-#xD7FF、#xE000-#xFFFD,可通过正则预处理或CDATA包裹已清洗内容,结合XML库容错机制有效避免异常。 在处理XML数据时,经常会遇到非法字符导致解析失败的问题。XML对可接受的字符有严格…

    2025年12月17日
    000
  • 什么是XML Swiss Army Knife

    “XML瑞士军刀”指的是一套多功能、集成化的工具集,用于应对XML数据处理的多样性与复杂性。它涵盖解析(DOM/SAX/StAX)、验证(DTD/XSD)、查询(XPath/XQuery)、转换(XSLT)及编辑工具(如Oxygen XML Editor),需根据项目需求、技术栈和成本灵活组合使用,…

    2025年12月17日
    000
  • XML在数字孪生中的应用

    XML为数字孪生提供结构化数据建模、跨平台互操作性及配置版本管理支持,通过层级标签描述孪生体属性与关系,利用XSD保障数据规范,作为通用文本格式实现系统间数据交换,并兼容Git等工具实现模型变更追踪。 XML在数字孪生中的应用,核心在于其作为一种强大的数据描述和交换语言,为数字孪生复杂的结构化信息提…

    2025年12月17日
    000

发表回复

登录后才能评论
关注微信