精华内容
下载资源
问答
  • 提高效率一直是个永恒的话题,编程中有一也是可以提到效率的,那就是专注做一件事情,让其它没有强紧密联系的与之分开。这里分享下我们做CRUD时遇到的常见数据处理场景: 数据库表字段全部设计为非空,即使这个...

     

    提高效率一直是个永恒的话题,编程中有一项也是可以提到效率的,那就是专注做一件事情,让其它没有强紧密联系的与之分开。这里分享下我们做CRUD时遇到的常见数据处理场景:

    • 数据库表字段全部设计为非空,即使这个字段在业务上是可以为空的,之所以将数据库表字段全部设计为非空,这里有优点也有缺点,我们认为优点大于缺点,所以选择了它

         优点:

    1. 获取值时,不用判断这个字段是否为null,直接可用于逻辑运算。
    2. mysql DBA推荐此方案,可能是有利于性能,这里我并非求证过。

         缺点:

    1. 业务含义没有null清楚,比如int字段默认值设置成0,0就没有null语义清晰。
    2. 在使用ORM插入数据时,需要处理非空字段值为null的问题。
    •   系统字段的赋值,比如创建人,创建人id,创建时间,编辑人,编辑人id,编辑时间等,这些都需要在实际插入数据库前赋值给Model。这些系统字段与具体的业务一般没有太大的关联关系,只是起到标注数据被什么人在什么时间处理的,当这些非业务相关的代码充斥在代码中时,就显得有些多余,而且这类代码多了也会显示冗余,最后带来的结果就是非关键代码比例大。


      上面关于默认值与null语义问题不需要解决,因为我们认为具有默认值带来的优点远大于可空字段带来的烦恼,我们来看默认值与系统字段一般情况下如何处理:

    • 在操作ORM时,将模型所有可空的字段都手动赋值成默认值,int的赋值为0等。
    • 在设计数据库时,将非空字段加上默认值,让数据库来处理这些未插入值的字段,如果使用mybatis的话,mapper中提到的插入操作有两个:insert,insertSelective,后面这个insertSelective就是处理非空字段的,即插入的模型对于不需要赋值的字段就保持null值,数据库在插入时生成的sql语句也不会包含这些字段,这样就可以利用上数据库的默认值了。如果正巧数据库的结构当初设计时没有设计默认值,又不能改的情况就比较糟糕了,情况回到上面手动赋值,可能会出现类似如下的代码:编写一个函数通过反射来解析每个字段,如果为null就修改为默认值:
     public static <T> void emptyNullValue(final T model) {
            Class<?> tClass = model.getClass();
            List<Field> fields = Arrays.asList(tClass.getDeclaredFields());
            for (Field field : fields) {
                Type t = field.getType();
                field.setAccessible(true);
                try {
                    if (t == String.class && field.get(model) == null) {
                        field.set(model, "");
                    } else if (t == BigDecimal.class && field.get(model) == null) {
                        field.set(model, new BigDecimal(0));
                    } else if (t == Long.class && field.get(model) == null) {
                        field.set(model, new Long(0));
                    } else if (t == Integer.class && field.get(model) == null) {
                        field.set(model, new Integer(0));
                    } else if (t == Date.class && field.get(model) == null) {
                        field.set(model, TimeHelper.LocalDateTimeToDate(java.time.LocalDateTime.of(1990, 1, 1, 0, 0, 0, 0)));
                    }
                } catch (IllegalAccessException e) {
                    e.printStackTrace();
                }
            }
        }


      然后在代码调用insert前调用函数来解决:

    ModelHelper.emptyNullValue(request);

      如何处理系统字段呢,在创建编辑数据时,需要获取当前用户,然后根据逻辑分别更新创建人信息以及编辑人信息,我们专门编写一个反射机制的函数来处理系统字段:

      注:下面的系统字段的识别,是靠系统约定实现的,比如creator约定为创建人等,可根据不同的情况做数据兼容,如果系统设计的好,一般在一个系统下所有表的风格应该是相同的。

    public static <T> void buildCreateAndModify(T model,ModifyModel modifyModel,boolean isCreate){
    
            Class<?> tClass = model.getClass();
            List<Field> fields = Arrays.asList(tClass.getDeclaredFields());
            for (Field field : fields) {
                Type t = field.getType();
                field.setAccessible(true);
                try {
                    if(isCreate){
                        if (field.getName().equals(modifyModel.getcId())) {
                            field.set(model, modifyModel.getUserId());
                        }
                        if (field.getName().equals(modifyModel.getcName())) {
                            field.set(model, modifyModel.getUserName());
                        }
                        if (field.getName().equals(modifyModel.getcTime())) {
                            field.set(model, new Date());
                        }
                    }
                    if (field.getName().equals(modifyModel.getmId())) {
                        field.set(model, modifyModel.getUserId());
                    }
                    if (field.getName().equals(modifyModel.getmName())) {
                        field.set(model, modifyModel.getUserName());
                    }
                    if (field.getName().equals(modifyModel.getmTime())) {
                        field.set(model, new Date());
                    }
                } catch (IllegalAccessException e) {
                    e.printStackTrace();
                }
            }
        }

      最后在数据处理前,根据创建或者编辑去调用函数来给系统字段赋值,这类代码都混杂在业务代码中。

    ModifyModel modifyModel = new ModifyModel();
            modifyModel.setUserId(getCurrentEmployee().getId());
            modifyModel.setUserName(getCurrentEmployee().getName());
            if (request.getId() == 0) {
                ModelHelper.buildCreateAndModify(request, modifyModel, true);
                deptService.insert(request);
            } else {
                ModelHelper.buildCreateAndModify(request, modifyModel, false);
                deptService.updateByPrimaryKey(request);
            }


      我们可以利用参数注入来解决。参数注入的理念就是在spring mvc接收到前台请求的参数后,进一步对接收到的参数做处理以达到预期的效果。我们来创建ManageModelConfigMethodArgumentResolver,它需要实现HandlerMethodArgumentResolver,这个接口看起来比较简单,包含两个核心方法:

    •  判断是否是需要注入的参数,一般通过判断参数上是否有特殊的注解来实现,也可以增加一个其它的参数判断,可根据具体的业务做调整,我这里只以是否有特殊注释来判定是否需要参数注入。
    @Override
        public boolean supportsParameter(MethodParameter parameter) {
            return parameter.hasParameterAnnotation(ManageModelConfig.class);
        }
    •  参数注入,它提供了一个扩展入口,让我们有机会对接收到的参数做进一步的处理。
    @Override
        public Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer,
                NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception {
    
            Object manageModel =getRequestResponseBodyMethodProcessor().resolveArgument(parameter, mavContainer, webRequest, binderFactory);
    
            ServletRequest servletRequest = webRequest.getNativeRequest(ServletRequest.class);
            Employee currentUser = (Employee) servletRequest.getAttribute(DEFAULT_ATTRIBUTE_GET_USER_FROM_REQUEST);
            if (null == currentUser)
            {
                return manageModel;
            }
            ManageModelConfig parameterAnnotation = parameter.getParameterAnnotation(ManageModelConfig.class);
            ModelHelper.setDefaultAndSystemFieldsValue(manageModel, currentUser,parameterAnnotation.isSetDefaultFieldsValue());
            return manageModel;
        }

        这段函数有几处核心逻辑:

    • 取得参数对象,因为我们处理的是ajax请求的参数,最简单的注入方法就是得到实际参数通过反射去处理默认字段以及系统的值。ajax请求与form表单post提交的数据绑定略有不同,可参考之前文章分享的列表页动态搜索的参数注入(列表页的动态条件搜索)。获取当前请求参数对象,我们可以借助如下两个对象配合来完成:
      • RequestMappingHandlerAdapter
      • RequestResponseBodyMethodProcessor
    private RequestMappingHandlerAdapter requestMappingHandlerAdapter=null;
        private RequestResponseBodyMethodProcessor requestResponseBodyMethodProcessor = null;
    
        private RequestResponseBodyMethodProcessor getRequestResponseBodyMethodProcessor() {
            if(null==requestMappingHandlerAdapter)
            {
                requestMappingHandlerAdapter=new RequestMappingHandlerAdapter();
            }
    
            if (null==requestResponseBodyMethodProcessor) {
                List<HttpMessageConverter<?>> messageConverters = requestMappingHandlerAdapter.getMessageConverters();
                messageConverters.add(new MappingJackson2HttpMessageConverter());
                requestResponseBodyMethodProcessor = new RequestResponseBodyMethodProcessor(messageConverters);
            }
            return requestResponseBodyMethodProcessor;
        }

        通过如下代码就可以取到参数对象了,其实就是让spring mvc重新解析了一遍参数。

    Object manageModel =getRequestResponseBodyMethodProcessor().resolveArgument(parameter, mavContainer, webRequest, binderFactory);
    • 如何获取当前用户,我们在成功登录系统后,将当前用户的信息存储在request中,然后就可以在函数中获取当前用户,也可以采用其它方案,比如ThreadLocal,缓存等等。
    ServletRequest servletRequest = webRequest.getNativeRequest(ServletRequest.class);
            Employee currentUser = (Employee) servletRequest.getAttribute(DEFAULT_ATTRIBUTE_GET_USER_FROM_REQUEST);
    • 调用处理函数解决默认字段以及系统的赋值,可以根据配置来决定是否处理字段默认值。
    ManageModelConfig parameterAnnotation = parameter.getParameterAnnotation(ManageModelConfig.class);
            ModelHelper.setDefaultAndSystemFieldsValue(manageModel, currentUser,parameterAnnotation.isSetDefaultFieldsValue());

      最后将我们的参数注入逻辑启动起来,这里选择在xml中配置:

     <mvc:annotation-driven content-negotiation-manager="contentNegotiationManager">
            <mvc:argument-resolvers>
                <bean class="cn.wanmei.party.management.common.mvc.method.annotation.ManageModelConfigMethodArgumentResolver"/>
            </mvc:argument-resolvers>
        </mvc:annotation-driven>


        再看action中的调用:只需要在参数前面增加注解@ManageModelConfig,如果需要处理默认值,则将启用默认值的选项设置成true即可,下面的实现部分完全看不到任何与业务无关的代码。

    @RequestMapping(value = "/addOrUpdateUser")
        @ResponseBody
        public Map<String, Object> addOrUpdateUser(@ManageModelConfig(isSetDefaultFieldsValue=true) EmployeeDto request) {
            Map<String, Object> ret = new HashMap<>();
            ValidateUtil.ValidateResult result= new ValidateUtil().ValidateModel(request);
            boolean isCreate=request.getId() == 0;
    
           try {
               if (isCreate)
               {
                   employeeService.insert(request);
               }
               else
               {
                   employeeService.updateByPrimaryKey(request);
    
               }
               ret.put("data", "ok");
           }catch (Exception e){
               ret.put("err", e.getMessage());
           }
    
            return ret;
        }

    通过自定义实现HandlerMethodArgumentResolver,来捕获ajax请求的参数,利用反射机制动态的将系统字段以及需要处理默认值的字段自动赋值,避免人工干预,起到了代码精简,逻辑干净,问题统一处理的目的。需要注意的是这些实现都是结合当前系统设计的,比如我们认为id字段>0就代表是更新操作,为空或者等于小于0就代表是创建,系统字段也是约定名称的等等。

     

    转载于:https://www.cnblogs.com/ASPNET2008/p/5393391.html

    展开全文
  • 1、如何由数列的前面几写出一个通公式.  利用观察、分析、归纳的思想方法得出通公式,本身也是一个重要的数学思想. 例5、写出数列的一个通公式,使它的前5分别是下列各数:  (1)  (2) ; (3)5...
  • GetCharacterPlacement 该函数用于了解如何用一个给定的字符显示一个字串 GetCharWidth 调查字体中一个或多个字符的宽度 GetFontData 接收一种可缩放字体文件的数据 GetFontLanguageInfo 返回目前选入指定设备...
  • 判断及取得控制 动态新增控制 产生前端处理语法 3-4-4表单流程设计 浏览另一页 传递条件 第四章以Visual Studio.NET开发Web应用系统 4-1 Visual Studio.NET与Web 4-1-1有关Web的...
  • 功能方面:砍掉了很多无关紧要的功能,比如贴内回复,友情链接(可以作为插件存在),让代码变得更加精炼,目前打包后只有600k 。   正式版在以下几个方面做了改进: 一级版块分类 多维主题分类 随着帖子的增多...
  • 每章小结的最后一“测验”包含了一系列关于本章内容的判断题和选择题。在附录E中可以找到这些问题、可能的答案、解决方案以及相关解释。  第Ⅰ部分:PL/SQL基础  ·第1章介绍了Oracle开发架构以及连接机制。 ...
  • 每章小结的最后一“测验”包含了一系列关于本章内容的判断题和选择题。在附录E中可以找到这些问题、可能的答案、解决方案以及相关解释。  第Ⅰ部分:PL/SQL基础  ·第1章介绍了Oracle开发架构以及连接机制。 ...
  • asp.net知识库

    2015-06-18 08:45:45
    如何判断ArrayList,Hashtable,SortedList 这类对象是否相等 帮助解决网页和JS文件中的中文编码问题的小工具 慎用const关键字 装箱,拆箱以及反射 动态调用对象的属性和方法——性能和灵活性兼备的方法 消除由try/...
  • JAVA面试题最全集

    2010-03-13 13:09:10
    判断一个文件或目录是否存在 如何读写文件 7.Java多态的实现(继承、重载、覆盖) 8.编码转换,怎样实现将GB2312编码的字符串转换为ISO-8859-1编码的字符串。 9.Java中访问数据库的步骤,Statement和...
  • 2.1.1 设备无关的编程和OpenGL 2.1.2 窗口的编程 2.1.3 如何打开一个窗口画图 2.2 OpenGL的基本图形元素 2.2.1 几个点丛绘制的例子 2.3 OpenGL中的直线绘制 2.3.1 绘制折线和多边形 2.3.2 使用moveTo()和lineTo...
  • 五:Java如何做到让机器理解我们想要做的东西 用·个图来描述这个过程会比较谷易理解: PDF文件使用" pdfFactory Pro"试用版本创建ww, fineprint,cn Java私塾跟我学系列JAⅥ篇网址:htp:/www.lavass.Cn电话:...
  • (我觉得学习如何把人们的积极性调动起来,以及如何配合协同的团队精神,是我行政工作的主要目标。) A:I have refined my management style by using an open-door policy. (我以开放式的政策,改进我的行政管理...
  • 思科网络技术学院教程CCNA1

    热门讨论 2013-05-28 06:37:53
    而一些章开始的“附加的感兴趣的主题”部分列出的是与核心课程无关但作为网络工程师需要理解的主题。  此外,作者也提供了大量真实世界的信息。包括不同的真实网络的例子和附加的主题。我们鼓励你去学习这些扩充的...
  • 《数据结构 1800题》

    热门讨论 2012-12-27 16:52:03
    5.抽象数据类型的定义仅取决于它的一组__(1)_,而与_(2)_无关,即不论其内部结构如何变化,只要它的_(3)_不变,都不影响其外部使用。【山东大学 2001 三、3(2分)】 6.数据结构中评价算法的两个重要指标是...
  • 并生成一依赖配置,最后将所有更新、或者被依赖的组件统一升级版本号加入发布队列。 完整代码仓库:https://github.com/fex-team/fit/blob/master/scripts/module-manage/utils/version.js</p> <h3>...
  • 机房与Internet相连,需要为每台计算机分配IP地址、计算机名,设置DNS、网关等,这是一既麻 烦又耗时的事情。噢易机房BOSS系统修改IP地址的功能前所未有的方便,可以通过发射端收集所有网卡的 MAC地址,在发射...
  • 内容简介回到顶部↑ASP.NET具有开发效率高、平台无关性、安全性强等方面的优势,广泛用于创建动态Web站点和基于Web的分布式应用程序,是广大Windows程序员开发Web应用程序的有效工具。ASP.NET 2.0在Web应用程序...
  • modbus通信协议

    热门讨论 2010-08-06 15:26:22
    在其它网络上(象MAP和Modbus Plus)Modbus消息被转成与串行传输无关的帧。 1、ASCII模式 当控制器设为在Modbus网络上以ASCII(美国标准信息交换代码)模式通信,在消息中的每个8Bit字节都作为两个ASCII字符发送。...
  • 其次列为综合课程,因为它把各门职能课程结合在一起,不仅阐明了各种职能在企业争取达到经营目标的过程中如何相互作用,而且指出了企业与其所处的环境怎样互相影响。 表2.1.2 企业管理学的学科分类 ────...

空空如也

空空如也

1 2
收藏数 40
精华内容 16
关键字:

如何判断无关项