1. 程式人生 > >micro-mvc框架支持mvc各層業務代碼熱部署

micro-mvc框架支持mvc各層業務代碼熱部署

template man type repo cor map agen 參數名稱 component

micro-mvc框架,可以與springmvc和springcloud整合,使所有的controller、servicebean、dao和sql業務邏輯代碼都支持熱部署方便開發人員調式和生產部署。
源碼與demo地址:https://github.com/jeffreyning/micro-mvc

與springmvc整合過程

編寫Controller接口

整合後Springmvc的controller只編寫接口,參數名稱必須用RequestParam註解。
使用InjectGroovy註解在接口中聲明對應的groovy實現名稱。
其他與傳統springmvc的controller無異。

package foo.web;

import java.util.Map;
import javax.servlet.http.HttpServletRequest;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.ResponseBody;
import com.nh.micro.service.InjectGroovy;

@Controller
@RequestMapping("test")
@InjectGroovy(name="TestController")
public interface TestController {
@RequestMapping("echo")
@ResponseBody
public  Map echo(@RequestParam(value="str") String str,HttpServletRequest httpRequest);

}

編寫Controller層的實現groovy

package groovy;

import javax.annotation.Resource;
import javax.servlet.http.HttpServletRequest;

import com.nh.micro.rule.engine.core.plugin.MicroAop;
import com.nh.micro.rule.engine.core.plugin.MicroDefaultLogProxy;
import com.nh.micro.service.InjectGroovy;
import com.nh.micro.template.MicroDbProxy;
import com.nh.micro.template.MicroTMProxy;

import foo.service.TestService;

@MicroAop(name=[MicroDefaultLogProxy.class,MicroTMProxy.class,MicroDbProxy.class], property=["","",""])
class TestController  {  
    @Resource
    public TestService testService;

    public Map echo(String str,HttpServletRequest httpRequest) {
        System.out.println("this is controller proxy");
        testService.test("111");
        Map retMap=new HashMap();
        retMap.put("status", "0");
        return retMap;
    }

}

配置controller層包掃描

使用GroovyBeanScannerConfigurer代替context:component-scan對controller進行掃描。

    <bean class="com.nh.micro.service.GroovyBeanScannerConfigurer">
        <property name="scanPath" value="foo.web"></property>
    </bean>
<!-- <context:component-scan base-package="foo.web" /> -->

Service層

編寫ServiceBean接口

並進行包掃描,使controller層能夠是resource加載到ServiceBean代理對象。註意在接口中使用@InjectGroovy註解聲明對應的實現業務邏輯的groovy文件名字
編寫ServiceBean接口

package foo.service;
import com.nh.micro.service.InjectGroovy;
@InjectGroovy(name="TestService")
public interface TestService {
    public void test(String id);
}

Service層groovy實現

package groovy.service;
import javax.annotation.Resource;
import foo.dao.TestDao;
import foo.dto.MicroTestDto;
import foo.repository.*;
import groovy.json.*;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import org.springframework.transaction.annotation.Transactional;
import com.nh.micro.dao.mapper.DefaultPageInfo;
import com.nh.micro.dao.mapper.InjectDao;
import com.nh.micro.rule.engine.core.plugin.MicroAop;
import com.nh.micro.rule.engine.core.plugin.MicroDefaultLogProxy;
import com.nh.micro.template.MicroDbProxy;
import com.nh.micro.template.MicroServiceTemplateSupport;
import com.nh.micro.template.MicroTMProxy;

@MicroAop(name=[MicroDefaultLogProxy.class,MicroTMProxy.class,MicroDbProxy.class], property=["","",""])
class TestService  {  

    @Resource
    public TestDao testDao;

    public void test(String id){

        Map paramMap=new HashMap();
        paramMap.put("id", id);
        MicroTestDto microTestDto=testDao.queryInfoById(paramMap);

        List<MicroTestDto> list=testDao.getInfoListAllMapper(microTestDto, ""); 

        DefaultPageInfo pageInfo=new DefaultPageInfo();
        pageInfo.setPageNo(1);
        List<MicroTestDto> retList=testDao.queryInfosByPage(paramMap, pageInfo);
        Long total=pageInfo.getTotal();
        System.out.println("total="+total);

    }

}

Service層包掃描配置

    <bean class="com.nh.micro.service.GroovyBeanScannerConfigurer">
        <property name="scanPath" value="foo.service"></property>
    </bean>

Dao層

使用micro-dao模塊,需要編寫dto類,dao接口和sql文件。通過掃描sql文件和dao接口,使service層能夠使用Resource註解加載dao代理對象。

編寫dto對象類

使用@MicroTableName映射數據中表名,使用@MicroMappingAnno映射表中列名。

package foo.dto;

import java.util.Date;
import com.nh.micro.orm.MicroMappingAnno;
import com.nh.micro.orm.MicroTableName;

@MicroTableName(name="micro_test")
public class MicroTestDto {
    @MicroMappingAnno(name="id")
    private String id;
    @MicroMappingAnno(name="meta_key")
    private String metaKey;
    @MicroMappingAnno(name="meta_name")
    private String metaName;
    @MicroMappingAnno(name="meta_type")
    private String metaType;
    @MicroMappingAnno(name="create_time")
    private Date createTime;

    @MicroMappingAnno(name="meta_num")
    private Integer metaNum;

    public Integer getMetaNum() {
        return metaNum;
    }

    public void setMetaNum(Integer metaNum) {
        this.metaNum = metaNum;
    }

    public Date getCreateTime() {
        return createTime;
    }

    public void setCreateTime(Date createTime) {
        this.createTime = createTime;
    }

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getMetaKey() {
        return metaKey;
    }

    public void setMetaKey(String metaKey) {
        this.metaKey = metaKey;
    }

    public String getMetaName() {
        return metaName;
    }

    public void setMetaName(String metaName) {
        this.metaName = metaName;
    }

    public String getMetaType() {
        return metaType;
    }

    public void setMetaType(String metaType) {
        this.metaType = metaType;
    }
}

編寫Dao接口

package foo.dao;
import java.util.List;
import java.util.Map;
import com.nh.micro.dao.mapper.ListInnerClass;
import com.nh.micro.dao.mapper.MicroCommonMapper;
import com.nh.micro.dao.mapper.MicroPageInfo;
import com.nh.micro.orm.MicroDbName;
import foo.dto.MicroTestDto;

@MicroDbName
public interface TestDao extends MicroCommonMapper<MicroTestDto> {
        public int updateInfo(Map paramMap);
        public int insertInfo(Map paramMap);
        @ListInnerClass(name=MicroTestDto.class)
    public List<MicroTestDto> queryInfosByPage(Map paramMap,MicroPageInfo pageInfo);    
    public MicroTestDto queryInfoById(Map paramMap);    
}

Dao層包掃描配置

    <bean id="daoScan" class="com.nh.micro.dao.mapper.scan.BeanScannerConfigurer">
        <property name="scanPath" value="foo.dao"></property>
    </bean>

編寫sql模板

類似jsp的語法編寫sql,支持#和$兩種替換符。不必區分select還是update,統一用<sql id=”xxx”>標簽,id與dao接口方法名一致,sql文件名與dao接口名一致。

<%! <sql id="queryInfoById"> %>
select * from micro_test where 1=1 
<% if(paramArray[0].get("id")!=null){ %>        
    and id = #{paramArray[0].id}            
<%  } %>
<%! </sql> %>

<%! <sql id="queryInfosByPage"> %>
select * from micro_test 
<%! </sql> %>

<%! <sql id="insertInfo"> %>
insert into micro_test(id,meta_key) values( 
?           
<% repList.add(paramArray[0].get("id"));%>
,?      
<% repList.add(paramArray[0].get("meta_key"));%>    
    )
<%! </sql> %>

<%! <sql id="updateInfo"> %>
update micro_test set  

<% if(paramArray[0].get("meta_key")!=null){%>
,meta_key=#{paramArray[0].get("meta_key")}  
<% } %> 
<% if(paramArray[0].get("meta_name")!=null){%>
,meta_name=#{paramArray[0].get("meta_name")}    
<% } %> 

where id=#{paramArray[0].get("id")}
<%! </sql> %>

Sql文件掃描配置

    <!-- micro-dao sql文件加載 -->
    <bean class="com.nh.micro.nhs.NhsInitUtil"
        init-method="initGroovyAndThread" lazy-init="false">
        <property name="fileList">
            <list>
                <bean class="com.nh.micro.rule.engine.core.GFileBean">
                    <property name="ruleStamp" value="true"></property>
                    <property name="jarFileFlag" value="true"></property>
                    <property name="dirFlag" value="true"></property>
                    <property name="rulePath" value="/groovy/"></property>
                </bean>
            </list>
        </property>
    </bean>

MicroDao說明

同時支持mysql和oracle
MicroDao相對mybatis的優點:
1,sql腳本支持修改後熱部署實時生效。
2,bean與數據庫字段映射關系,通過註解設置到bean中,不必在sql腳本中體現。
3,sql腳本支持類似jsp的寫法,且不必區分select、update使用不同標簽,更加靈活。
4,不需要使用插件,內置支持物理分頁。
5,不需要使用插件,內置支持針對bean的標準增刪改查功能。
6,不需要使用插件,內置支持讀寫分離,分庫分表。
7,針對mysql5.7支持動態字段。
支持mapper、template、非orm三種模式支撐業務系統
1,mapper指,通過掃描接口,運行時自動生成dao實例;
2,template指,通過繼承template標準父類,生成dao子類;
3,非orm指,直接使用microDao實例,可以執行非orm更靈活的數據庫操作。

關於事務

由於serviceBean接口對象通過包掃描發布為springbean,所以仍可以使用原有的spring事務機制。

    <!-- 配置platform transaction manager-->
    <bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"/>
    </bean> 

    <!-- 聲明式事物管理,配置事物管理advice-->
     <tx:advice id="txAdvice" transaction-manager="txManager">
        <tx:attributes>

            <tx:method name="get*" read-only="true"/>
<tx:method name="test*" propagation="REQUIRED"/>
            <tx:method name="add*" propagation="REQUIRED"/>
            <tx:method name="insert*" propagation="REQUIRED"/>
            <tx:method name="update*" propagation="REQUIRED"/>
            <tx:method name="*"/>
        </tx:attributes>
    </tx:advice> 

    <!-- 配置事物管理advice作用範圍與作用條件-->
    <aop:config>
        <aop:pointcut id="serviceLayerTransaction" expression="execution( * foo.service.*..*(..))"/>
        <aop:advisor pointcut-ref="serviceLayerTransaction" advice-ref="txAdvice"/>
    </aop:config>

SpringCloud整合

SpringCloud整合controller,service、dao層與springmvc整合均一致。
具體查看demo項目
Micro-springcloud-mvc

Nhmicro的Aop機制

加載groovy時提供aop代理機制,默認提供事務aop和數據源切換aop
事務aop,可在加載時,識別Transactional註解,實現事務控制。
可自行別寫特定功能代理。
開啟代理需要在groovy的類中設置MicroAop註解指定代理類

@MicroAop(name=[MicroDefaultLogProxy.class,MicroTMProxy.class,MicroDbProxy.class], property=["","",""])

micro-mvc框架支持mvc各層業務代碼熱部署