• 定时任务


    TOP 1:Timer

    Timer是 JDK 自带的定时任务执行类,无论任何项目都可以直接使用 Timer 来实现定时任务,所以 Timer 的优点就是使用方便

     但缺点是任务如果执行时间太长或者是任务执行异常,会影响其他任务调度,所以在生产环境下建议谨慎使用。

    public class MyTimerTask {
        public static void main(String[] args) {
            // 定义一个任务
            TimerTask timerTask = new TimerTask() {
                @Override
                public void run() {
                    System.out.println("Run timerTask:" + new Date());
                }
            };
            // 计时器
            Timer timer = new Timer();
            // 添加执行任务(延迟 1s 执行,每 3s 执行一次)
            timer.schedule(timerTask, 1000, 3000);
        }
    }

    程序执行结果如下:

    Run timerTask:Mon Aug 17 21:29:25 CST 2020

    Run timerTask:Mon Aug 17 21:29:28 CST 2020

    Run timerTask:Mon Aug 17 21:29:31 CST 2020

    问题 1:任务执行时间长影响其他任务

    当一个任务的执行时间过长时,会影响其他任务的调度,如下代码所示:

    public class MyTimerTask {
        public static void main(String[] args) {
            // 定义任务 1
            TimerTask timerTask = new TimerTask() {
                @Override
                public void run() {
                    System.out.println("进入 timerTask 1:" + new Date());
                    try {
                        // 休眠 5 秒
                        TimeUnit.SECONDS.sleep(5);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println("Run timerTask 1:" + new Date());
                }
            };
            // 定义任务 2
            TimerTask timerTask2 = new TimerTask() {
                @Override
                public void run() {
                    System.out.println("Run timerTask 2:" + new Date());
                }
            };
            // 计时器
            Timer timer = new Timer();
            // 添加执行任务(延迟 1s 执行,每 3s 执行一次)
            timer.schedule(timerTask, 1000, 3000);
            timer.schedule(timerTask2, 1000, 3000);
        }
    }

    程序执行结果如下:

    进入 timerTask 1:Mon Aug 17 21:44:08 CST 2020

    Run timerTask 1:Mon Aug 17 21:44:13 CST 2020

    Run timerTask 2:Mon Aug 17 21:44:13 CST 2020

    进入 timerTask 1:Mon Aug 17 21:44:13 CST 2020

    Run timerTask 1:Mon Aug 17 21:44:18 CST 2020

    进入 timerTask 1:Mon Aug 17 21:44:18 CST 2020

    Run timerTask 1:Mon Aug 17 21:44:23 CST 2020

    Run timerTask 2:Mon Aug 17 21:44:23 CST 2020

    进入 timerTask 1:Mon Aug 17 21:44:23 CST 2020

    从上述结果中可以看出,当任务 1 运行时间超过设定的间隔时间时,任务 2 也会延迟执行。 原本任务 1 和任务 2 的执行时间间隔都是 3s,但因为任务 1 执行了 5s,因此任务 2 的执行时间间隔也变成了 10s(和原定时间不符)。

    问题 2:任务异常影响其他任务

    使用 Timer 类实现定时任务时,当一个任务抛出异常,其他任务也会终止运行,如下代码所示:

    public class MyTimerTask {
        public static void main(String[] args) {
            // 定义任务 1
            TimerTask timerTask = new TimerTask() {
                @Override
                public void run() {
                    System.out.println("进入 timerTask 1:" + new Date());
                    // 模拟异常
                    int num = 8 / 0;
                    System.out.println("Run timerTask 1:" + new Date());
                }
            };
            // 定义任务 2
            TimerTask timerTask2 = new TimerTask() {
                @Override
                public void run() {
                    System.out.println("Run timerTask 2:" + new Date());
                }
            };
            // 计时器
            Timer timer = new Timer();
            // 添加执行任务(延迟 1s 执行,每 3s 执行一次)
            timer.schedule(timerTask, 1000, 3000);
            timer.schedule(timerTask2, 1000, 3000);
        }
    }

    程序执行结果如下:

    进入 timerTask 1:Mon Aug 17 22:02:37 CST 2020

    Exception in thread "Timer-0" java.lang.ArithmeticException: / by zero

    at com.example.MyTimerTask$1.run(MyTimerTask.java:21)

    at java.util.TimerThread.mainLoop(Timer.java:555)

    at java.util.TimerThread.run(Timer.java:505)

    Process finished with exit code 0

    TOP 2:ScheduledExecutorService

     ScheduledExecutorService 也是 JDK 1.5 自带的 API,我们可以使用它来实现定时任务的功能,也就是说 ScheduledExecutorService 可以实现 Timer 类具备的所有功能,并且它可以解决了 Timer 类存在的所有问题。

    public class MyScheduledExecutorService {
             public static void main(String[] args) {
             // 创建任务队列
             ScheduledExecutorService scheduledExecutorService =Executors.newScheduledThreadPool(10); // 10 为线程数量
             // 执行任务
             scheduledExecutorService.scheduleAtFixedRate(() -> {System.out.println("Run Schedule:" + new Date());}, 1, 3, TimeUnit.SECONDS); // 1s 后开始执行,每 3s 执行一次
             }
    }

    ScheduledExecutorService 可靠性测试

    ① 任务超时执行测试

    ScheduledExecutorService 可以解决 Timer 任务之间相应影响的缺点,首先我们来测试一个任务执行时间过长,会不会对其他任务造成影响,测试代码如下:

    public class MyScheduledExecutorService {
        public static void main(String[] args) {
            // 创建任务队列
            ScheduledExecutorService scheduledExecutorService =
                    Executors.newScheduledThreadPool(10);
            // 执行任务 1
            scheduledExecutorService.scheduleAtFixedRate(() -> {
                System.out.println("进入 Schedule:" + new Date());
                try {
                    // 休眠 5 秒
                    TimeUnit.SECONDS.sleep(5);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("Run Schedule:" + new Date());
            }, 1, 3, TimeUnit.SECONDS); // 1s 后开始执行,每 3s 执行一次
            // 执行任务 2
            scheduledExecutorService.scheduleAtFixedRate(() -> {
                System.out.println("Run Schedule2:" + new Date());
            }, 1, 3, TimeUnit.SECONDS); // 1s 后开始执行,每 3s 执行一次
        }
    }

    程序执行结果如下:

    Run Schedule2:Mon Aug 17 11:27:55 CST 2020

    进入 Schedule:Mon Aug 17 11:27:55 CST 2020

    Run Schedule2:Mon Aug 17 11:27:58 CST 2020

    Run Schedule:Mon Aug 17 11:28:00 CST 2020

    进入 Schedule:Mon Aug 17 11:28:00 CST 2020

    Run Schedule2:Mon Aug 17 11:28:01 CST 2020

    Run Schedule2:Mon Aug 17 11:28:04 CST 2020

    从上述结果可以看出,当任务 1 执行时间 5s 超过了执行频率 3s 时,并没有影响任务 2 的正常执行,因此使用 ScheduledExecutorService 可以避免任务执行时间过长对其他任务造成的影响。

    ② 任务异常测试

    接下来我们来测试一下 ScheduledExecutorService 在一个任务异常时,是否会对其他任务造成影响,测试代码如下:

    public class MyScheduledExecutorService {
        public static void main(String[] args) {
            // 创建任务队列
            ScheduledExecutorService scheduledExecutorService =
                    Executors.newScheduledThreadPool(10);
            // 执行任务 1
            scheduledExecutorService.scheduleAtFixedRate(() -> {
                System.out.println("进入 Schedule:" + new Date());
                // 模拟异常
                int num = 8 / 0;
                System.out.println("Run Schedule:" + new Date());
            }, 1, 3, TimeUnit.SECONDS); // 1s 后开始执行,每 3s 执行一次
            // 执行任务 2
            scheduledExecutorService.scheduleAtFixedRate(() -> {
                System.out.println("Run Schedule2:" + new Date());
            }, 1, 3, TimeUnit.SECONDS); // 1s 后开始执行,每 3s 执行一次
        }
    }

    程序执行结果如下:

    进入 Schedule:Mon Aug 17 22:17:37 CST 2020

    Run Schedule2:Mon Aug 17 22:17:37 CST 2020

    Run Schedule2:Mon Aug 17 22:17:40 CST 2020

    Run Schedule2:Mon Aug 17 22:17:43 CST 2020

    从上述结果可以看出,当任务 1 出现异常时,并不会影响任务 2 的执行。

    TOP 3:Spring Task

    如果使用的是 Spring 或 Spring Boot 框架,可以直接使用 Spring Framework 自带的定时任务,使用上面两种定时任务的实现方式,很难实现设定了具体时间的定时任务,比如当我们需要每周五来执行某项任务时,但如果使用 Spring Task 就可轻松的实现此需求。

    以 Spring Boot 为例,实现定时任务只需两步:开启定时任务;添加定时任务。

    具体实现步骤如下。

    ① 开启定时任务
    开启定时任务只需要在 Spring Boot 的启动类上声明 @EnableScheduling 即可,实现代码如下:

    @SpringBootApplication
    @EnableScheduling // 开启定时任务
    public class DemoApplication {
        // do someing
    }

    ② 添加定时任务

    定时任务的添加只需要使用 @Scheduled 注解标注即可,如果有多个定时任务可以创建多个 @Scheduled 注解标注的方法,示例代码如下:

    import org.springframework.scheduling.annotation.Scheduled;
    import org.springframework.stereotype.Component;
     
    @Component // 把此类托管给 Spring,不能省略
    public class TaskUtils {
        // 添加定时任务
        @Scheduled(cron = "59 59 23 0 0 5") // cron 表达式,每周五 23:59:59 执行
        public void doTask(){
            System.out.println("我是定时任务~");
        }
    }

    注意:定时任务是自动触发的无需手动干预,也就是说 Spring Boot 启动后会自动加载并执行定时任务。

    Spring Task 的实现需要使用 cron 表达式来声明执行的频率和规则,cron 表达式是由 6 位或者 7 位组成的(最后一位可以省略),每位之间以空格分隔,每位从左到右代表的含义如下:

     其中 * 和 ? 号都表示匹配所有的时间。附加cron 表达式在线生成地址:https://cron.qqe2.com/

    TOP4. 定时任务之-Quartz使用篇

    Quartz是OpenSymphony开源组织在Job scheduling领域又一个开源项目,它可以与J2EE与J2SE应用程序相结合也可以单独使用。Quartz可以用来创建简单或为运行十个,百个,甚至是好几万个Jobs这样复杂的日程序表。Jobs可以做成标准的Java组件或 EJBs。

    CronTrigger配置格式: [秒] [分] [小时] [日] [月] [周] [年]

    序号    说明    是否必填    允许填写的值             允许的通配符
    1       秒      是           0-59                , - * /
    2       分      是           0-59                , - * /
    3      小时     是           0-23                , - * /
    4       日      是           1-31                , - * ? / L W
    5       月      是         1-12 or JAN-DEC       , - * /
    6       周      是         1-7 or SUN-SAT        , - * ? / L #
    7       年      否         empty 或 1970-2099    , - * /

    通配符说明:
    * 表示所有值. 例如:在分的字段上设置 "*",表示每一分钟都会触发。
    ? 表示不指定值。使用的场景为不需要关心当前设置这个字段的值。例如:要在每月的10号触发一个操作,但不关心是周几,所以需要周位置的那个字段设置为"?" 具体设置为 0 0 0 10 * ?
    - 表示区间。例如 在小时上设置 "10-12",表示 10,11,12点都会触发。
    , 表示指定多个值,例如在周字段上设置 "MON,WED,FRI" 表示周一,周三和周五触发
    / 用于递增触发。如在秒上面设置"5/15" 表示从5秒开始,每增15秒触发(5,20,35,50)。在月字段上设置'1/3'所示每月1号开始,每隔三天触发一次。
    L 表示最后的意思。在日字段设置上,表示当月的最后一天(依据当前月份,如果是二月还会依据是否是润年[leap]), 在周字段上表示星期六,相当于"7"或"SAT"。如果在"L"前加上数字,则表示该数据的最后一个。例如在周字段上设置"6L"这样的格式,则表示“本月最后一个星期五"
    W 表示离指定日期的最近那个工作日(周一至周五). 例如在日字段上设置"15W",表示离每月15号最近的那个工作日触发。如果15号正好是周六,则找最近的周五(14号)触发, 如果15号是周未,则找最近的下周一(16号)触发.如果15号正好在工作日(周一至周五),则就在该天触发。如果指定格式为 "1W",它则表示每月1号往后最近的工作日触发。如果1号正是周六,则将在3号下周一触发。(注,"W"前只能设置具体的数字,不允许区间"-").

    # 序号(表示每月的第几个周几),例如在周字段上设置"6#3"表示在每月的第三个周六.注意如果指定"#5",正好第五周没有周六,则不会触发该配置(用在母亲节和父亲节再合适不过了)

    常用示例:

    0 0 12 * * ?	每天12点触发
    0 15 10 ? * *	每天10点15分触发
    0 15 10 * * ?	每天10点15分触发
    0 15 10 * * ? *	每天10点15分触发
    0 15 10 * * ? 2005	2005年每天10点15分触发
    0 * 14 * * ?	每天下午的 2点到2点59分每分触发
    0 0/5 14 * * ?	每天下午的 2点到2点59分(整点开始,每隔5分触发)
    0 0/5 14,18 * * ?	每天下午的 2点到2点59分(整点开始,每隔5分触发)每天下午的 18点到18点59分(整点开始,每隔5分触发)
    0 0-5 14 * * ?	每天下午的 2点到2点05分每分触发
    0 10,44 14 ? 3 WED	3月分每周三下午的 2点10分和2点44分触发
    0 15 10 ? * MON-FRI	从周一到周五每天上午的10点15分触发
    0 15 10 15 * ?	每月15号上午10点15分触发
    0 15 10 L * ?	每月最后一天的10点15分触发
    0 15 10 ? * 6L	每月最后一周的星期五的10点15分触发
    0 15 10 ? * 6L 2002-2005	从2002年到2005年每月最后一周的星期五的10点15分触发
    0 15 10 ? * 6#3	每月的第三周的星期五开始触发
    0 0 12 1/5 * ?	每月的第一个中午开始每隔5天触发一次
    0 11 11 11 11 ?	每年的11月11号 11点11分触发(光棍节)
    

      

    经过封装的管理类:

    import java.text.ParseException;  
      
    import org.quartz.CronTrigger;  
    import org.quartz.JobDetail;  
    import org.quartz.Scheduler;  
    import org.quartz.SchedulerException;  
    import org.quartz.SchedulerFactory;  
    import org.quartz.impl.StdSchedulerFactory;  
      
    /** 
     * 定时任务管理类 
     * 
     */  
    public class QuartzManager {  
        private static SchedulerFactory gSchedulerFactory = new StdSchedulerFactory();  
        private static String JOB_GROUP_NAME = "EXTJWEB_JOBGROUP_NAME";  
        private static String TRIGGER_GROUP_NAME = "EXTJWEB_TRIGGERGROUP_NAME";  
      
        /** 
         * 添加一个定时任务,使用默认的任务组名,触发器名,触发器组名 
         * 
         * @param jobName 
         *            任务名 
         * @param jobClass 
         *            任务 
         * @param time 
         *            时间设置,参考quartz说明文档 
         * @throws SchedulerException 
         * @throws ParseException 
         */  
        public static void addJob(String jobName, String jobClass, String time) {  
            try {  
                Scheduler sched = gSchedulerFactory.getScheduler();  
                JobDetail jobDetail = new JobDetail(jobName, JOB_GROUP_NAME, Class.forName(jobClass));// 任务名,任务组,任务执行类  
                // 触发器  
                CronTrigger trigger = new CronTrigger(jobName, TRIGGER_GROUP_NAME);// 触发器名,触发器组  
                trigger.setCronExpression(time);// 触发器时间设定  
                sched.scheduleJob(jobDetail, trigger);  
                // 启动  
                if (!sched.isShutdown()){  
                    sched.start();  
                }  
            } catch (Exception e) {  
                e.printStackTrace();  
                throw new RuntimeException(e);  
            }  
        }  
      
        /** 
         * 添加一个定时任务 
         * 
         * @param jobName 
         *            任务名 
         * @param jobGroupName 
         *            任务组名 
         * @param triggerName 
         *            触发器名 
         * @param triggerGroupName 
         *            触发器组名 
         * @param jobClass 
         *            任务 
         * @param time 
         *            时间设置,参考quartz说明文档 
         * @throws SchedulerException 
         * @throws ParseException 
         */  
        public static void addJob(String jobName, String jobGroupName,  
                String triggerName, String triggerGroupName, String jobClass, String time){  
            try {  
                Scheduler sched = gSchedulerFactory.getScheduler();  
                JobDetail jobDetail = new JobDetail(jobName, jobGroupName, Class.forName(jobClass));// 任务名,任务组,任务执行类  
                // 触发器  
                CronTrigger trigger = new CronTrigger(triggerName, triggerGroupName);// 触发器名,触发器组  
                trigger.setCronExpression(time);// 触发器时间设定  
                sched.scheduleJob(jobDetail, trigger);  
            } catch (Exception e) {  
                e.printStackTrace();  
                throw new RuntimeException(e);  
            }  
        }  
      
        /** 
         * 修改一个任务的触发时间(使用默认的任务组名,触发器名,触发器组名) 
         * 
         * @param jobName 
         * @param time 
         */  
        public static void modifyJobTime(String jobName, String time) {  
            try {  
                Scheduler sched = gSchedulerFactory.getScheduler();  
                CronTrigger trigger = (CronTrigger) sched.getTrigger(jobName, TRIGGER_GROUP_NAME);  
                if(trigger == null) {  
                    return;  
                }  
                String oldTime = trigger.getCronExpression();  
                if (!oldTime.equalsIgnoreCase(time)) {  
                    JobDetail jobDetail = sched.getJobDetail(jobName, JOB_GROUP_NAME);  
                    Class objJobClass = jobDetail.getJobClass();  
                    String jobClass = objJobClass.getName();  
                    removeJob(jobName);  
      
                    addJob(jobName, jobClass, time);  
                }  
            } catch (Exception e) {  
                e.printStackTrace();  
                throw new RuntimeException(e);  
            }  
        }  
      
        /** 
         * 修改一个任务的触发时间 
         * 
         * @param triggerName 
         * @param triggerGroupName 
         * @param time 
         */  
        public static void modifyJobTime(String triggerName,  
                String triggerGroupName, String time) {  
            try {  
                Scheduler sched = gSchedulerFactory.getScheduler();  
                CronTrigger trigger = (CronTrigger) sched.getTrigger(triggerName, triggerGroupName);  
                if(trigger == null) {  
                    return;  
                }  
                String oldTime = trigger.getCronExpression();  
                if (!oldTime.equalsIgnoreCase(time)) {  
                    CronTrigger ct = (CronTrigger) trigger;  
                    // 修改时间  
                    ct.setCronExpression(time);  
                    // 重启触发器  
                    sched.resumeTrigger(triggerName, triggerGroupName);  
                }  
            } catch (Exception e) {  
                e.printStackTrace();  
                throw new RuntimeException(e);  
            }  
        }  
      
        /** 
         * 移除一个任务(使用默认的任务组名,触发器名,触发器组名) 
         * 
         * @param jobName 
         */  
        public static void removeJob(String jobName) {  
            try {  
                Scheduler sched = gSchedulerFactory.getScheduler();  
                sched.pauseTrigger(jobName, TRIGGER_GROUP_NAME);// 停止触发器  
                sched.unscheduleJob(jobName, TRIGGER_GROUP_NAME);// 移除触发器  
                sched.deleteJob(jobName, JOB_GROUP_NAME);// 删除任务  
            } catch (Exception e) {  
                e.printStackTrace();  
                throw new RuntimeException(e);  
            }  
        }  
      
        /** 
         * 移除一个任务 
         * 
         * @param jobName 
         * @param jobGroupName 
         * @param triggerName 
         * @param triggerGroupName 
         */  
        public static void removeJob(String jobName, String jobGroupName,  
                String triggerName, String triggerGroupName) {  
            try {  
                Scheduler sched = gSchedulerFactory.getScheduler();  
                sched.pauseTrigger(triggerName, triggerGroupName);// 停止触发器  
                sched.unscheduleJob(triggerName, triggerGroupName);// 移除触发器  
                sched.deleteJob(jobName, jobGroupName);// 删除任务  
            } catch (Exception e) {  
                e.printStackTrace();  
                throw new RuntimeException(e);  
            }  
        }  
      
        /** 
         * 启动所有定时任务 
         */  
        public static void startJobs() {  
            try {  
                Scheduler sched = gSchedulerFactory.getScheduler();  
                sched.start();  
            } catch (Exception e) {  
                e.printStackTrace();  
                throw new RuntimeException(e);  
            }  
        }  
      
        /** 
         * 关闭所有定时任务 
         */  
        public static void shutdownJobs() {  
            try {  
                Scheduler sched = gSchedulerFactory.getScheduler();  
                if(!sched.isShutdown()) {  
                    sched.shutdown();  
                }  
            } catch (Exception e) {  
                e.printStackTrace();  
                throw new RuntimeException(e);  
            }  
        }  
    }  

    简单实现Schedule的Quartz的例子

    第一步:引包

      要使用Quartz,必须要引入以下这几个包:

      1、log4j-1.2.16

      2、quartz-2.1.7(版本过高,换版本低的)

      3、slf4j-api-1.6.1.jar

      4、slf4j-log4j12-1.6.1.jar

      这些包都在下载的Quartz包里面包含着,因此没有必要为寻找这几个包而头疼。

    第二步:创建要被定执行的任务类

      这一步也很简单,只需要创建一个实现了org.quartz.Job接口的类,并实现这个接口的唯一execute方法即可。如:

    import java.text.SimpleDateFormat; 
    import java.util.Date; 
    import org.quartz.Job; 
    import org.quartz.JobExecutionContext; 
    import org.quartz.JobExecutionException; 
     
    public class myJob implements Job { 
     
        @Override 
        public void execute(JobExecutionContext arg0) throws JobExecutionException { 
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss SSS"); 
            System.out.println(sdf.format(new Date())); 
        } 
     
    } 
     
    

    第三步:创建任务调度,并执行

    import java.text.SimpleDateFormat;
    import java.util.Date;
    import org.quartz.CronTrigger;
    import org.quartz.JobDetail;
    import org.quartz.Scheduler;
    import org.quartz.SchedulerFactory;
    import org.quartz.impl.StdSchedulerFactory;
     
    public class Test {
        public void go() throws Exception {
            // 首先,必需要取得一个Scheduler的引用
            SchedulerFactory sf = new StdSchedulerFactory();
            Scheduler sched = sf.getScheduler();
            String time="0 51 11 ? * *";
            // jobs可以在scheduled的sched.start()方法前被调用
     
            // job 1将每隔20秒执行一次
            JobDetail job = new JobDetail("job1", "group1", myJob.class);
            CronTrigger trigger = new CronTrigger("trigger1", "group1");
            trigger.setCronExpression(time);
            Date ft = sched.scheduleJob(job, trigger);
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss SSS");
            System.out.println(job.getKey() + " 已被安排执行于: " + sdf.format(ft) + ",并且以如下重复规则重复执行: " + trigger.getCronExpression());
     
            // job 2将每2分钟执行一次(在该分钟的第15秒)
            job = new JobDetail("job2", "group1",myJob.class);
            trigger = new CronTrigger("trigger2", "group1");
            trigger.setCronExpression(time);
            ft = sched.scheduleJob(job, trigger);
            System.out.println(job.getKey() + " 已被安排执行于: " + sdf.format(ft) + ",并且以如下重复规则重复执行: " + trigger.getCronExpression());
     
            // 开始执行,start()方法被调用后,计时器就开始工作,计时调度中允许放入N个Job
            sched.start();
            try {
                // 主线程等待一分钟
                Thread.sleep(60L * 1000L);
               } catch (Exception e) {
              }
            // 关闭定时调度,定时器不再工作
            sched.shutdown(true);
        }
     
        public static void main(String[] args) throws Exception {
     
            Test test = new Test();
            test.go();
        }
     
    }
    我话讲完!谁赞成?谁反对?
  • 相关阅读:
    zw版【转发·台湾nvp系列Delphi例程】Delphi 使用 HALCON库件COM控件数据格式转换
    zw版【转发·台湾nvp系列Delphi例程】HALCON AddNoiseWhite
    zw版【转发·台湾nvp系列Delphi例程】HALCON CheckDifference
    zw版【转发·台湾nvp系列Delphi例程】HALCON BinThreshold
    zw版【转发·台湾nvp系列Delphi例程】HALCON HighpassImage
    zw版【转发·台湾nvp系列Delphi例程】HALCON Histogram
    zw版【转发·台湾nvp系列Delphi例程】HALCON Component Histogram
    C 碎片五 数组
    C 碎片四 流程控制
    C 碎片三 运算符与表达式
  • 原文地址:https://www.cnblogs.com/wffzk/p/14806425.html
Copyright © 2020-2023  润新知