常见的工作流框架:activiti, JBPM, OSWorkflow

activiti框架基于23张基础的表数据, 基于Mybatis操作数据库.

JBPM框架基于18张基础的表数据, 基于hibernate操作数据库.

一. activiti基础数据表的创建

1.通过sql创建

导入activiti-5.13\database\create目录下的sql建表语句, 如果数据库是mysql, 则需要导入activiti.mysql.create.engine.sql, activiti.mysql.create.history.sql, activiti.mysql.create.identity.sql三张表;

2.通过java代码创建

2.1 在不提供xml配置文件的情况下

  //通过在java中创建流程引擎配置对象来创activiti的基础表数据
    @Test
    public void demo1 () {
        //创建流程引擎配置对象
        ProcessEngineConfiguration configuration =
                ProcessEngineConfiguration.createStandaloneProcessEngineConfiguration();
        configuration.setJdbcDriver("com.mysql.jdbc.Driver");
        configuration.setJdbcUrl("jdbc:mysql://localhost:3306/activiti_demo");
        configuration.setJdbcUsername("root");
        configuration.setJdbcPassword("123");
        //设置自动建表
        configuration.setDatabaseSchemaUpdate("true");
        //创建流程引擎对象, 在创建流程引擎对象时会自动建表
        ProcessEngine buildProcessEngine = configuration.buildProcessEngine();
    }

2.2 提供xml配置文件

1.配置文件位于src根路径下, 名称为activiti-context.xml

<!-- 配置一个流程引擎配置对象 -->
    <bean id="processEngineConfiguration"
        class="org.activiti.engine.impl.cfg.StandaloneProcessEngineConfiguration">
        <property name="jdbcDriver" value="com.mysql.jdbc.Driver"></property>
        <property name="jdbcUrl" value="jdbc:mysql://localhost:3306/activiti_demo"></property>
        <property name="jdbcUsername" value="root"></property>
        <property name="jdbcPassword" value="luoji1025"></property>
        <property name="databaseSchemaUpdate" value="true"></property>
    </bean>
    
    <!-- 配置一个流程引擎工厂bean用于创建流程引擎配置对象 -->
    <bean id="processEngine" class="org.activiti.spring.ProcessEngineFactoryBean">
        <property name="processEngineConfiguration" ref="processEngineConfiguration"></property>
    </bean>
</beans>

2.Java代码

//通过配置文件结合java代码实现activiti基础表的创建
    @Test
    public void demo2() {
        //通过配置文件,创建流程引擎配置对象
        ProcessEngineConfiguration configuration = ProcessEngineConfiguration.
                createProcessEngineConfigurationFromResource("activiti-context.xml", "processEngineConfiguration");
        ProcessEngine processEngine = configuration.buildProcessEngine();
    }

2.3 使用默认的配置文件(推荐使用)

  要求配置文件必须在类路径的根路径下,配置文件的名称必须为activiti-context.xml或者为activiti.cfg.xml,xml配置文件中必须配置流程引擎配置对象,id必须为processEngineConfiguration,必须配置流程引擎工厂bean,id必须为processEngine(具体配置与上面的xml一致)

//通过默认的配置文件创建activiti基础表
    @Test
    public void demo3 () {
        ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
    }

二. activiti中23张表的介绍

Activiti的后台是有数据库的支持,所有的表都以ACT_开头。 第二部分是表示表的用途的两个字母标识。 用途也和服务的API对应。

1)      ACT_RE_*: 'RE'表示repository。 这个前缀的表包含了流程定义和流程静态资源 (图片,规则,等等)。

2)      ACT_RU_*: 'RU'表示runtime。 这些运行时的表,包含流程实例,任务,变量,异步任务,等运行中的数据。 Activiti只在流程实例执行过程中保存这些数据, 在流程结束时就会删除这些记录。 这样运行时表可以一直很小速度很快。

3)      ACT_ID_*: 'ID'表示identity。 这些表包含身份信息,比如用户,组等等。

4)      ACT_HI_*: 'HI'表示history。 这些表包含历史数据,比如历史流程实例, 变量,任务等等。

ACT_GE_*: 通用数据, 用于不同场景下。

  业务流程建模与标注(Business Process Model and Notation,BPMN) ,描述流程的基本符号,包括这些图元如何组合成一个业务流程图(Business Process Diagram)

三. activiti中常用的对象

3.1 几个和流程相关的对象

Deployment:部署对象,和部署表对应act_re_deployment

ProcessDefinition:流程定义对象,和流程定义表对应act_re_procdef

ProcessInstance:流程实例对象,和流程实例表对应act_ru_execution

Task:任务对象,和任务表对应act_ru_task

3.2 几个Service对象

RepositoryService:操作部署、流程定义等静态资源信息

RuntimeService:操作流程实例,启动流程实例、查询流程实例、删除流程实例等动态信息

TaskService:操作任务,查询任务、办理任务等和任务相关的信息

HistoryService:操作历史信息的,查询历史信息

IdentityService:操作用户和组

3.3  几个Query对象

DeploymentQuery:对应查询部署表act_re_deployment

ProcessDefinitionQuery:对应查询流程定义表act_re_procdef

ProcessInstanceQuery:对应查询流程实例表act_ru_execution

TaskQuery:对应查询任务表act_ru_task

四. Activiti流程定义框架中常用的API

4.1 部署与流程定义相关的API

4.1.1 部署流程定义

1>基于元数据(bpmn和png)文件创建

/*
     * 部署流程定义
     * 影响的表
     * 1.act_re_deployment:部署表
     * 2.act_re_procdef:流程定义表(流程部署后,会在流程定义表中创建一个流程定义对象)
     * 3.act_ge_bytearray:二进制表(保存bpmn和png文件)
     */

 private ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
    @Test
    public void demo4 () {
        //创建一个部署构建器对象, 用于加载制定的文件
        DeploymentBuilder deployment = processEngine.getRepositoryService().createDeployment();
        deployment.addClasspathResource("qjlc.bpmn");
        deployment.addClasspathResource("qjlc.png");
        //部署,返回部署对象
        Deployment deploy = deployment.deploy();
        System.out.println(deploy.getId());
    }

2> 基于bpmn和png文件的压缩文件创建

@Test
    public void demo1 () {
        //获取文件部署构建器对象
        DeploymentBuilder deploymentBuilder = processEngine.getRepositoryService().createDeployment();
        //2.通过zip文件部署(常用)
        ZipInputStream zipInputStream = new ZipInputStream(this.getClass().getClassLoader()
                .getResourceAsStream("qjlc.zip"));
        deploymentBuilder.addZipInputStream(zipInputStream);
        Deployment deploy = deploymentBuilder.deploy();
        System.out.println(deploy.getId());
    }

4.1.2 查询部署信息

@Test
    public void demo2 () {
        DeploymentQuery query = processEngine.getRepositoryService().createDeploymentQuery();
//        query.deploymentId("701");
        List<Deployment> list = query.list();
        for (Deployment deployment : list) {
            System.out.println(deployment.getId() + " " + deployment.getName());
        }
    }

4.1.3 删除部署信息

/*
     * 删除部署信息
     */
    @Test
    public void demo3 () {
        String deploymentId = "601";
        boolean cascade = false;
        processEngine.getRepositoryService().deleteDeployment(deploymentId, cascade);
    }

4.1.4 查询流程定义

 /*
     * 查询流程定义
     * 操作的数据表:act_re_procdef流程定义表
     */
    @Test
    public void demo5 () {
        //获取流程定义查询对象
        ProcessDefinitionQuery query = processEngine.getRepositoryService().createProcessDefinitionQuery();
        //根据流程定义的key进行过滤
        query.processDefinitionKey("qjlc");
        //添加排序条件
        query.orderByProcessDefinitionVersion().desc();

   //查询最新的版本

   query.latestVersion();
        //查询
        List<ProcessDefinition> list = query.list();
        for (ProcessDefinition definition : list) {
            System.out.println(definition.getId() + " " + definition.getKey() + " " + definition.getName());
        }
    }

4.1.5 查询一次部署对应的流程部署文件名称和输入流

@Test
    public void demo5 () {
        String deploymentId = "1";
        List<String> names = processEngine.getRepositoryService().getDeploymentResourceNames(deploymentId);
        for (String name : names) {
            System.out.println(name);
            InputStream inputStream = processEngine.getRepositoryService().getResourceAsStream(deploymentId, name);
            try {
                FileUtils.copyInputStreamToFile(inputStream, new File("E:\\"+name));
                inputStream.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

4.1.6 获取文件名称和输入流

@Test
    public void demo6 () throws IOException {
        //获取流程定义的id
        String processDefinitionId = "qjlc:4:904";
        //获取流程定义查询对象
        ProcessDefinitionQuery query = processEngine.getRepositoryService().createProcessDefinitionQuery();
        //根据id进行查询
        query.processDefinitionId(processDefinitionId);
        //调整查询结果集
        ProcessDefinition processDefinition = query.singleResult();
        //根据流程定义对象获取png图片的名称
        String name = processDefinition.getDiagramResourceName();
        //获取png图片对应的输入流
        InputStream inputStream = processEngine.getRepositoryService().getProcessDiagram(processDefinitionId);
        //调用fileutils工具类查询数据
        FileUtils.copyInputStreamToFile(inputStream, new File("e:\\"+name));
        inputStream.close();
    }

4.2 流程定义实例和任务相关的API

4.2.1 启动流程定义实例

  /*
     * 启动流程实例
     * 操作的数据表:
     * 1.act_ru_execution:流程实例表
     * 2.act_ru_task:任务表(启动流程实例时,会在任务表中创建一个任务)
     */
    @Test
    public void demo6 () {
        //流程定义的id
        String processId = "qjlc:1:4";
        //通过runtimeservice启动流程实例
        ProcessInstance processInstance = processEngine.getRuntimeService().startProcessInstanceById(processId);
        System.out.println(processInstance.getId() + "  " + processInstance.getBusinessKey() + "  " + processInstance.getParentId());    
    }

4.2.2 查询流程实例

@Test
    public void demo8 () {
        ProcessInstanceQuery query = processEngine.getRuntimeService().createProcessInstanceQuery();
        List<ProcessInstance> list = query.list();
        for (ProcessInstance processInstance : list) {
            System.out.println(processInstance.getId() + " " + processInstance.getActivityId());
        }
        //查询任务
        TaskQuery query2 = processEngine.getTaskService().createTaskQuery();
        List<Task> list2 = query2.list();
        for (Task task : list2) {
            System.out.println(task.getId() + "  " + task.getName());
        }
    }

4.2.3 查询任务

/*
     * 查询任务
     */
    @Test
    public void demo7 () {
        TaskQuery taskQuery = processEngine.getTaskService().createTaskQuery();
        taskQuery.taskAssignee("王五");
        List<Task> list = taskQuery.list();
        for (Task task : list) {
            System.out.println(task.getId() + "  " + task.getName());
        }
    }

4.2.4 办理任务

/*
     * 办理任务
     */
    @Test
    public void demo8 () {
        String taskId = "402";
        processEngine.getTaskService().complete(taskId);
    }

五. 流程变量

5.1流程变量的创建

流程变量的创建有四种方式:创建流程实例时创建, 办理任务时创建, 基于runtimeservice创建, 基于taskservice创建

5.1.1 当流程实例启动时设置流程变量

@Test
    public void demo2 () {
        String processDefinitionId = "qjlc:5:1604";
        Map<String, Object> variables = new HashMap<String, Object>();
        variables.put("key1", "value1");
        variables.put("key2", "value2");
        ProcessInstance  processInstance = processEngine.getRuntimeService().startProcessInstanceById(processDefinitionId, variables);
        System.out.println(processInstance.getId());
    }

5.1.2 在办理任务时创建

@Test
    public void demo3 () {
        TaskService taskService = processEngine.getTaskService();
        TaskQuery taskQuery = taskService.createTaskQuery();
        List<Task> list = taskQuery.list();
        for (Task task : list) {
            System.out.println(task.getId());
            String taskId = task.getId();
            Map<String, Object> variables = new HashMap<String, Object>();
            variables.put("k1", "v1");
            variables.put("k2", "v2");
            taskService.complete(taskId, variables);
        }
        
    }

5.1.3 利用runtimeservice创建流程变量

@Test
    public void demo4 () {
        String executionId = "1701";
        String variableName = "qjyy";
        Object value = "想请假了!";
        //创建单个流程变量
        processEngine.getRuntimeService().setVariable(executionId, variableName, value);
        //创建多个流程变量
        Map<String, Object> variables = new HashMap<>();
        variables.put("k3", "v3");
        variables.put("k4", "v4");
        processEngine.getRuntimeService().setVariables(executionId, variables);
    }

5.1.4 利用taskservice创建流程变量

@Test
    public void demo5 () {
        String taskId = "1804";
        String variableName = "qjts";
        Object value = 7;
        //创建单个流程变量
        processEngine.getTaskService().setVariable(taskId, variableName, value);
        //创建多个流程对象
        Map<String, Object> variables = new HashMap<>();
        variables.put("k5", "v5");
        variables.put("k6", "v6");
        processEngine.getTaskService().setVariables(taskId, variables);
    }

5.1.5 流程变量中存放自定义对象(自定义对象要实现Serializable接口)

@Test
    public void demo5_1 () {
        //创建流程实例
        /*String processDefinitionKey = "qjlc";
        ProcessInstance processInstance = processEngine.getRuntimeService().startProcessInstanceByKey(processDefinitionKey );
        System.out.println(processInstance.getId());*/
        //创建流程变量
        String executionId = "2501";
        Map<String, Object> variables = new HashMap<>();
        User user = new User();
        user.setUsername("老王");
        user.setPassword("123");
        variables.put("user", user );
        processEngine.getRuntimeService().setVariables(executionId , variables);
    }

自定义对象User

package cn.rodge.activiti.variable;
import java.io.Serializable;
public class User implements Serializable {
    private static final long serialVersionUID = 1L;
    private String username;
    private String password;
    public String getUsername() {
        return username;
    }
    public void setUsername(String username) {
        this.username = username;
    }
    public String getPassword() {
        return password;
    }
    public void setPassword(String password) {
        this.password = password;
    }
}

5.2 流程变量的获取

5.2.1 基于RuntimeService获取

@Test
    public void demo6 () {
        String executionId = "1701";
        String variableName = "qjts";
        //获取单个流程变量
        Object variable = processEngine.getRuntimeService().getVariable(executionId, variableName);
        System.out.println(variable);
        //获取多个流程变量
        Map<String, Object> variables = processEngine.getRuntimeService().getVariables(executionId);
        System.out.println(variables);
    }

5.2.2 基于TaskService获取

@Test
    public void demo7 () {
        String taskId = "1804";
        String variableName = "qjyy";
        //获取单个流程变量
        Object variable = processEngine.getTaskService().getVariable(taskId, variableName);
        System.out.println(variable);
        Object getVariables;
        //获取多个流程变量, 获取当前流程所在流程实例中的所有流程变量
        Map<String, Object> variables = processEngine.getTaskService().getVariables(taskId);
        System.out.println(variables);
    }

六. 组任务

6.1 候选人组任务

6.1.1 查询组任务

@Test
    public void demo5 () {
        TaskQuery taskQuery = processEngine.getTaskService().createTaskQuery();
        taskQuery.taskCandidateUser("李四");
        List<Task> list = taskQuery.list();
        for (Task task : list) {
            System.out.println(task.getId() + "  " + task.getName());
        }
    }

6.1.2 拾取组任务

@Test
    public void demo6 () {
        String taskId = "3802";
        String userId = "李四";
        processEngine.getTaskService().claim(taskId, userId);
    }

6.1.3 退回组任务

@Test
    public void demo7 () {
        String taskId = "3802";
        String userId = null;
        processEngine.getTaskService().setAssignee(taskId, userId);
    }

6.2 候选组组任务

6.2.1 创建组

@Test
        public void demo3 () {
            Group group = new GroupEntity();
            group.setId("财务人员组");
            group.setName("财务人员组");
            processEngine.getIdentityService().saveGroup(group);
        }

6.2.2 创建用户

@Test
    public void demo4 () {
        User user = new UserEntity();
        user.setId("002");
        user.setFirstName("小明");
        processEngine.getIdentityService().saveUser(user);
    }

6.2.3 将用户添加到组

@Test
    public void demo5 () {
        String groupId = "财务人员组";
        String userId = "002";
        processEngine.getIdentityService().createMembership(userId, groupId);
    }

6.2.4 查询组任务

@Test
    public void demo8 () {
        TaskQuery taskQuery = processEngine.getTaskService().createTaskQuery();
        taskQuery.taskCandidateUser("002");
        List<Task> list = taskQuery.list();
        for (Task task : list) {
            System.out.println(task.getId() + " " + task.getName());
        }
    }

6.2.5 拾取组任务

@Test
    public void demo9 () {
        String userId = "002";
        String taskId = "4702";
        processEngine.getTaskService().claim(taskId, userId);
    }

拾取完组任务后,就可以按照正常的任务办理流程办理任务了

6.2.6 退回组任务

@Test
    public void demo10 () {
        String userId = null;
        String taskId = "4702";
        processEngine.getTaskService().setAssignee(taskId, userId);
    }

七. 网关(排他网关)

private ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

    /*
     * 部署流程
     */
    @Test
    public void demo1() {
        DeploymentBuilder deploymentBuilder = processEngine.getRepositoryService().createDeployment();
        deploymentBuilder.addClasspathResource("cn/rodge/activiti/gatewall/bxlcnew.bpmn");
        deploymentBuilder.addClasspathResource("cn/rodge/activiti/gatewall/bxlcnew.png");
        Deployment deploy = deploymentBuilder.deploy();
        System.out.println(deploy.getId());
    }

    /*
     * 创建流程实例
     */
    @Test
    public void demo2() {
        String processDefinitionKey = "cwlc";
        ProcessInstance processInstance = processEngine.getRuntimeService()
                .startProcessInstanceByKey(processDefinitionKey);
        System.out.println(processInstance.getId());
    }

    /*
     * 查询流程实例
     */
    @Test
    public void demo3() {
        ProcessInstanceQuery processInstanceQuery = processEngine.getRuntimeService().createProcessInstanceQuery();
        List<ProcessInstance> list = processInstanceQuery.list();
        for (ProcessInstance processInstance : list) {
            System.out.println(processInstance.getId() + "  " + processInstance.getActivityId());
        }
    }
    /*
     * 查询任务
     */
    @Test
    public void demo4 () {
        TaskQuery taskQuery = processEngine.getTaskService().createTaskQuery();
        taskQuery.taskAssignee("王五");
        List<Task> list = taskQuery.list();
        for (Task task : list) {
            System.out.println(task.getId() + "  " + task.getAssignee());
        }
    }
    /*
     * 执行任务同时创建变量bxje
     */
    @Test
    public void demo5 () {
        String taskId = "6004";
        Map<String, Object> variables = new HashMap<>();
        variables.put("bxje", 2000);
        processEngine.getTaskService().complete(taskId, variables);
    }

当定义的流程变量bxje>1000时, 就会执行"财务主管审批"的分支

八. spring整合activiti

第一步:提供spring配置文件,配置数据源、事务管理器、spring提供的流程引擎配置对象、流程引擎工厂bean

<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
                        http://www.springframework.org/schema/beans/spring-beans.xsd
                        http://www.springframework.org/schema/context
                        http://www.springframework.org/schema/context/spring-context-2.5.xsd
                        http://www.springframework.org/schema/tx
                        http://www.springframework.org/schema/tx/spring-tx-3.0.xsd">
    <!-- 配置数据源 -->
    <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"></property>
        <property name="url" value="jdbc:mysql://localhost:3306/activiti_demo"></property>
        <property name="username" value="root"></property>
        <property name="password" value="luoji1025"></property>
    </bean>
    <!-- 配置事务管理器 -->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    <!-- 配置一个Spring整合流程引擎配置对象 -->
    <bean id="processEngineConfiguration"
        class="org.activiti.spring.SpringProcessEngineConfiguration">
        <!-- 配置数据源 -->
        <property name="dataSource" ref="dataSource"></property>
        <!-- 配置事务 -->
        <property name="transactionManager" ref="transactionManager"></property>
        <!-- 配置自动activiti的自动建表 -->
        <property name="databaseSchemaUpdate" value="true"></property>
    </bean>
    
    <!-- 配置一个流程引擎工厂bean用于创建流程引擎配置对象 -->
    <bean id="processEngine" class="org.activiti.spring.ProcessEngineFactoryBean">
        <property name="processEngineConfiguration" ref="processEngineConfiguration"></property>
    </bean>
</beans>

java代码

public class ActivitiSpring {
    public static void main(String[] args) {
        // 获取配置文件
        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
        // 从配置文件中获取流程引擎对象
        ProcessEngine processEngine = (ProcessEngine) context.getBean("processEngine");
        ProcessDefinitionQuery definitionQuery = processEngine.getRepositoryService().createProcessDefinitionQuery();
        List<ProcessDefinition> list = definitionQuery.list();
        for (ProcessDefinition processDefinition : list) {
            System.out.println(processDefinition.getId() + "  " + processDefinition.getName());
        }
    }
}

activiti工作流框架简介的更多相关文章

  1. Activiti工作流引擎简介

    Activiti工作流引擎简介 一.概述 Activiti是由Alfresco软件在2010年5月17日发布的业务流程管理(BPM)框架,它是覆盖了业务流程管理,工作流,服务协作等领域的一个开源,灵活 ...

  2. 项目结合activiti工作流框架使用

    项目结合activiti工作流框架使用: 1.项目与工作流框架的结合. 2.状态:草稿(待审批).审批中.审批通过.审批失败 3. 提交审批: 0 草稿(待审批),记录绑定工作流执行id,审批状态设置 ...

  3. Activiti工作流框架学习(一)——环境的搭建和数据表的了解

    一.什么是工作流 工作流(Workflow),就是“业务过程的部分或整体在计算机应用环境下的自动化”,它主要解决的是“使在多个参与者之间按照某种预定义的规则传递文档.信息或任务的过程自动进行,从而实现 ...

  4. Activiti工作流框架学习(一)之通用数据表详细介绍

    文/朱季谦 Activiti工作流引擎自带了一套数据库表,这里面有一个需要注意的地方: 低于5.6.4的MySQL版本不支持时间戳或毫秒级的日期.更糟糕的是,某些版本在尝试创建此类列时将引发异常,而其 ...

  5. Activiti工作流框架学习笔记(二)之springboot2.0整合工作流Activiti6.0

    以前在工作当中做过不少与工作流Activiti有关的工作,当时都是spring集成activiti5.22的项目,现在回过头去看,其实版本已经稍微老了,因此,基于先前的工作经验,决定用较新版本的技术来 ...

  6. Activiti工作流框架——快速上手

        一.前言 最近在做公司的OA,里面有用到工作流,公司用的是 jbpm4,感觉比较老,资料有点少,就先学学 新一点的 activiti  ㄟ(▔▽▔)ㄏ 首先工作流的概念是:工作流(Workfl ...

  7. Activiti工作流框架学习(二)——使用Activiti提供的API完成流程操作

    可以在项目中加入log4j,将logj4.properties文件拷入到src目录下,这样框架执行的sql就可以输出到到控制台,log4j提供的日志级别有以下几种: Fatal  error  war ...

  8. 项目实践之工作流引擎基本文档!Activiti工作流框架中流程引擎API和服务详解

    流程引擎的API和服务 流程引擎API(ProcessEngine API)是与Activiti打交道的最常用方式 Activiti从ProcessEngine开始.在ProcessEngine中,可 ...

  9. Activiti工作流小序曲

    一般涉及到OA.ERP等公司办公系统都必须有一套办公流程,这时候使用activiti工作流框架会大大减轻我们的工作量,提高我们的开发效率. Activiti工作流简单介绍: 工作流(workflow) ...

随机推荐

  1. 02_JNI中Java代码调用C代码,Android中使用log库打印日志,javah命令的使用,Android.mk文件的编写,交叉编译

     1  编写以下案例(下面的三个按钮都调用了底层的C语言): 项目案例的代码结构如下: 2 编写DataProvider的代码: package com.example.ndkpassdata; ...

  2. 并发服务器--02(基于I/O复用——运用epoll技术)

    本文承接自上一博文I/O复用——运用Select函数. epoll介绍 epoll是在2.6内核中提出的.和select类似,它也是一种I/O复用技术,是之前的select和poll的增强版本. Li ...

  3. 中国象棋游戏Chess(3) - 实现走棋规则

    棋盘的绘制和走棋参看博文:中国象棋游戏Chess(1) - 棋盘绘制以及棋子的绘制,中国象棋游戏Chess(2) - 走棋 现在重新整理之前写的代码,并且对于每个棋子的走棋规则都进行了限制,不像之前那 ...

  4. "《算法导论》之‘队列’":队列的三种实现(静态数组、动态数组及指针)

    本文有关栈的介绍部分参考自网站数据结构. 1. 队列  1.1 队列的定义 队列(Queue)是只允许在一端进行插入,而在另一端进行删除的运算受限的线性表. (1)允许删除的一端称为队头(Front) ...

  5. Linux打包命令 - tar

    上一篇文章谈到的命令大多仅能针对单一文件来进行压缩,虽然 gzip 与 bzip2 也能够针对目录来进行压缩, 不过,这两个命令对目录的压缩指的是『将目录内的所有文件 "分别" 进 ...

  6. 前端开发我为什么选择cordova

    cordova与phonegap有什么关系? phoengap 官方网址:http://phonegap.com 如果能了解一个框架的兴起还是一件比较有趣的事.08年一次ios开发者大会上来自Nito ...

  7. ASI与AFN网络请求的的比较

    对比 ASI AFN 更新状态 2012年10月份,已经停止更新 持续更新中,目前已更新至3.0版 介绍 ASI的直接操作对象ASIHTTPRequest,是一个实现了了NSCopying协议的NSO ...

  8. 《转》xcode创建一个工程的多个taget,便于测试和发布多个版本

    背景:很多时候,我们需要在一个工程中创立多个target,也就是说我们希望同一份代码可以创建两个应用,放到模拟器或者真机上,或者是,我们平时有N多人合作开发,当测试的时候,在A这里装了一遍测A写的那块 ...

  9. Java深入了解TreeSet

    Java中的TreeSet是Set的一个子类,TreeSet集合是用来对象元素进行排序的,同样他也可以保证元素的唯一.那TreeSet为什么能保证元素唯一,它是怎样排序的呢?先看一段代码: publi ...

  10. Razor视图

    @{ string name="jerry";} <div> @name </div>     //显示jerry @{ string js="& ...