NC业务插件模型源码分析

 

NC业务插件模型源码分析

1. Summary

为了增加NC程序事件响应的可扩展性,在Java事件响应的前后添加一些必要的操作。

2. NC事件监听器数据库设计

NC业务插件模型源码分析

表中字段的含义如下表:

pub_eventlistener

pk_eventlistener主键1001ZF1000000000099E
name名称同步经销商云平台商家
implcalssname实现类的全限定名,需实现IBusinessListener接口OrgInsertAfterListener
pk_eventtype外键,连接pub_eventtype表1018Z01000000000LLB9
owner外键,连接dap_dapsystemEC99

pub_eventtype

pk_eventtype主键1018Z01000000000LLB9
eventtypename事件类型名称新增后
sourceid元数据ID945f38b6-48ec-43e6-bb09-77ec89a3728f
owner?1010

dap_dapsystem

moduleid主键,与pub_eventlistener表的owner关联EC99
devmodule模块名称eschain

3 基本代码分析

3.1 代码配置组合

为了说明业务插件代码的运行原理,我们以品牌(bd_branddoc)插入后业务插件为例进行说明。VC6.0中大部分单据界面都需要一个Spring.xml 文件来配置界面,对界面进行布局。这个xml文件必须放在client文件夹下,路径和注册功能节点是注册的“BeanConfigPath”参数一致。

NC业务插件模型源码分析

uiuapbd_brand.jar文件中找到对应的spring配置文件:branddoc_globe.xml,主要配置如下:

<beans>
    <!--
        界面布局总装###########################################################
    -->
    <bean id="container" class="nc.ui.uif2.TangramContainer"
        init-method="initUI">
        <property name="tangramLayoutRoot">
            <bean class="nc.ui.uif2.tangramlayout.node.CNode">
                <property name="component" ref="brandlist"></property>
            </bean>
        </property>
        <property name="editActions">
            <list>
                <ref bean="saveAction" />
                ......
            </list>
        </property>
        <property name="model" ref="batchBillTableModel"></property>
    </bean>

    <bean id="saveAction" class="nc.ui.uif2.actions.batch.BatchSaveAction">
        <property name="model" ref="batchBillTableModel" />
        <property name="editor" ref="brandlist" />
        <property name="validationService" ref="validatorServicer"></property>
    </bean>

    <!-- 批量操作应用模型 -->
    <bean id="batchBillTableModel" class="nc.ui.uif2.model.BatchBillTableModel">
        <property name="service" ref="batchModelService"></property>
        <property name="businessObjectAdapterFactory" ref="objectadapterfactory"></property>
        <property name="context" ref="context"></property>
    </bean>

    <!-- 应用服务类,负责进行模型操作的处理 -->
    <bean id="batchModelService" class="nc.ui.bd.material.branddoc.model.BrandDocModelServicer" />
    ......
</beans>

3.2 代码调用过程

当点击保存按钮时,会调用nc.ui.uif2.actions.batch.BatchSaveAction类(父类为:javax.swing.Action)的doAction方法。基本的类调用关系如下图:

NC业务插件模型源码分析

nc.ui.uif2.actions.batch.BatchSaveAction#doAction()
↓
nc.ui.uif2.model.BatchBillTableModel#save()
↓
nc.ui.bd.material.branddoc.model.BrandDocModelServicer#batchSave
↓
nc.impl.bd.material.branddoc.BrandDocManagerImpl#batchSaveBrandDoc
↓
nc.impl.bd.material.branddoc.BrandDocManagerImpl(父类:nc.bs.bd.baseservice.md.BatchBaseService)#batchSave
↓
insertVO
↓
fireBeforeInsertEvent
fireAfterInsertEvent
↓
nc.bs.businessevent.bd.BDCommonEventUtil#dispatchInsertBeforeEvent
nc.bs.businessevent.bd.BDCommonEventUtil#dispatchInsertAfterEvent
↓
nc.bs.businessevent.EventDispatcher#fireEvent() // 静态方法

3.4 EventDispatcher事件派发过程

业务插件的事件派发过程主要有nc.bs.businessevent.EventDispatcher类的静态方法fireEvent完成,核心代码如下:

public static void fireEvent(IBusinessEvent event) throws BusinessException {

        writeDebugLog(event);
        // 从数据库中获取event listener 数据映射关系
        Map<String, UnionVO> classNameMap = getListenersInfo(event.getSourceID(), event.getEventType());
        if (classNameMap == null || classNameMap.isEmpty())
            return;

        String className = null, devModuleCode = null;
        try {
            String currgroup_localtype = getCurrGroupLocalType();
            String currgroup_industrytype = getCurrGroupIndustryType();

            for (Iterator<String> iter = classNameMap.keySet().iterator(); iter
                    .hasNext();) {
              // 获取event listener 的实现类名称
                className = iter.next();
                UnionVO unionvo = classNameMap.get(className);
                devModuleCode = unionvo.getDevmodulecode();
                String classInfoMsg = "Plugin class [" + className + "] modulecode [" + devModuleCode + "] ";
                /*
                 * 执行优先级为1/2/3/4的插件并集: 
                 * 行业为水平+本地化为国际化
                 * 行业为水平+ 本地化(如果本地化能匹配)
                 * 匹配行业(如果行业能匹配)+ 本地化为国际化 
                 * 匹配行业(如果行业匹配) + 本地化(如果本地化能匹配)
                 * 
                 * 插件中,行业为空或者为默认值综合控股集团按水平算,本地化为空或为中国按国际化算
                 */
                String localtype = StringUtil.isEmpty(unionvo.getLocaltype()) ? "" : unionvo.getLocaltype();
                String industrytype = StringUtil.isEmpty(unionvo.getIndustrytype()) ? "" : unionvo.getIndustrytype();
                int level = LocalAndIndustryLevelQueryUtil.getLevelByLocalAndIndustry(
                        localtype, industrytype, currgroup_localtype, currgroup_industrytype);
                if (level == 1 || level == 2 || level == 3 || level == 4) {
                    Logger.debug(classInfoMsg + " begin.");
                    Long start = System.currentTimeMillis();
                    // 利用反射获取IBusinessListener实例
                    IBusinessListener instance = getInstanceByClassName(className, devModuleCode);
                  // 执行Event listener 的doAction方法
                    instance.doAction(event);
                    Logger.debug(classInfoMsg + " end successfully.cost time : " + (System.currentTimeMillis() - start));
                }
            }
        } catch (BusinessException e) {
            Logger.error(getErrorMsg(event, className), e);
            throw e;
        } catch (RuntimeException e) {
            Logger.error(getErrorMsg(event, className), e);
            throw e;
        }
    }

fireEvent方法的主要任务是说先从表:pub_eventlistener,pub_eventtype与dap_dapsystem获取相应的数据并保存到Map中

nc.bs.businessevent.EventDispatcher#getListenersInfo() // 返回值:Map<String, UnionVO>
↓
getEventType_ClassNameMap() // 返回值为: Map<String, Map<String, UnionVO>>
↓
getAllListenersMap() 
↓
getAllListenersByOrder()
↓
nc.impl.businessevent.EventListenerQryServiceImpl#getAllListeners()

//getAllListeners的代码实现如下:
public UnionVO[] getAllListeners() throws BusinessException {
        String sql = "select T1.sourceid as sourceid,T1.eventtypecode as eventtype," +
                    "T0.implclassname as implclassname,T0.operindex as operindex," +
                    "M.devmodule as devmodulecode," +
                    "T0.localtype as localtype,T0.industrytype as industrytype " +
                     "from pub_eventlistener T0 " +
                        "left outer join pub_eventtype T1 on T0.pk_eventtype =T1.pk_eventtype " +
                        "left join dap_dapsystem  M on M.moduleid=T0.owner "+
                     "where T0.enabled = 'Y' or T0.enabled = 'y'";
        List<UnionVO> vos = (List<UnionVO>) getBaseDAO().executeQuery(sql,
                new BeanListProcessor(UnionVO.class));
        return vos.toArray(new UnionVO[0]);
    }

getEventType_ClassNameMap返回Map>的映射关系,他们的含义如下:

KeySourceId+'@'+eventtytpe3ee53558-6398-4096-a91f-c7aa00e93701@新增后
Valueimplclassname与UnionVo的映射 

其中SourceId的获取是在nc.impl.bd.material.branddoc.BrandDocManagerImpl实例创建时指定的:

public BrandDocManagerImpl() {
        super(IBDMetaDataIDConst.BRANDDOC);
    }

nc.itf.bd.pub.IBDMetaDataIDConst中包含了所有模块对应的SourceId常量值。

以上。

相关推荐