模板工程标准的控制台程序要素

(1)完整明确文字提示用户(并且使用红,绿,黄三种颜色标识提示文字;红色为错误相关,绿色为成功相关,黄色为进行中提示)

(2)告知用户运行进度(完成任务的一部分即显示进度条)

(3)如果需要用户做出选择,应该出现文本选择组件(通常为让用户选择yes还是no,或者使用1,2,3让用户做出选择)

(4)支持空执行(即使用事务回滚的方法,做出模拟执行,以避免在真正执行时出现错误,真正执行前会发现错误)

(5)支持无限次运行(使用缓存记录ID号的方法,若已经执行过则不再执行)

(6)命令行可以支持加参数

(7)核心业务逻辑应该被封装

例如:

/**
     * 山西永济基地的成品均改为标准件
     * 运行命令为./yii wms-handle/handle-piece-type-for-shanxi
     */
    public function actionHandlePieceTypeForShanxi(){
        //提示用户选择模拟更新还是真实更新
        if(Console::confirm("请选择真实更新还是模拟更新;模拟更新不会真正更新数据;[yes]为真实更新[no]为模拟更新")){
            $tip = Console::ansiFormat(PHP_EOL."正在轮询真实更新中",[Console::FG_GREEN]);
            Console::output("{$tip}");
            unset($tip);
            $is_update=true;
        }else{
            $tip = Console::ansiFormat(PHP_EOL."正在模拟更新中,不会真正更新",[Console::FG_RED]);
            Console::output("{$tip}");
            unset($tip);
            $is_update=false;
        }

        $wmsProductInSheetModels = \core\models\WmsProductInSheet::find()->where(['common_producer_info_id'=>14, 'piece_type'=>2])->orderBy(['id'=>SORT_ASC])->all();
        //设置进度条
        Console::startProgress(0, count($wmsProductInSheetModels));
        foreach ($wmsProductInSheetModels as $key => $wmsProductInSheetModel){
            $transtion = \Yii::$app->db->beginTransaction();
            try {
                //提示文字
                $tip = Console::ansiFormat(PHP_EOL . "正在处理成品入库单{$wmsProductInSheetModel->wms_product_in_sheet_number}", [Console::FG_YELLOW]);
                Console::output("{$tip}");
                unset($tip);
                //已经处理过则无需再次处理
                if (\Yii::$app->redis->sismember("in_sheet_number_updated_piece_type_for_shanxi", $wmsProductInSheetModel->wms_product_in_sheet_number)) {
                    throw new \yii\db\Exception(PHP_EOL . "成品入库单{$wmsProductInSheetModel->wms_product_in_sheet_number}已经更新过,无需再次更新");
                }

                //核心处理逻辑
                $updateHandler = $wmsProductInSheetModel->updatePieceTypeFrom2To1();
                if (!$updateHandler['status']){
                    throw new \yii\db\Exception($updateHandler['errmsg']);
                }else{
                    //提示文字
                    $tip = Console::ansiFormat(PHP_EOL . "正在处理成品入库单{$wmsProductInSheetModel->wms_product_in_sheet_number}成功", [Console::FG_GREEN]);
                    Console::output("{$tip}");
                    unset($tip);
                }

                //更新进度条
                Console::updateProgress($key + 1, count($wmsProductInSheetModels));
                //如果是模拟更新则回滚
                if (!$is_update) {
                    throw new \yii\db\Exception(PHP_EOL . "此次是模拟更新,不会真正更新,正在回滚到初始状态!");
                }
                $transtion->commit();
                //将已更新的入库单号添加到已更新队列中
                \Yii::$app->redis->sadd("in_sheet_number_updated_piece_type_for_shanxi", $wmsProductInSheetModel->wms_product_in_sheet_number);
            } catch (\yii\base\Exception $e) {
                $transtion->rollBack();
                $tip = Console::ansiFormat(PHP_EOL . $e->getMessage(), [Console::FG_RED]);
                Console::output("{$tip}");
                unset($tip);
            } catch (\yii\base\ErrorException $e) {
                $transtion->rollBack();
                $tip = Console::ansiFormat(PHP_EOL . $e->getMessage(), [Console::FG_RED]);
                Console::output("{$tip}");
                unset($tip);
            }
            unset($wmsProductInSheetModel);
        }
        unset($wmsProductInSheetModels);
        //设置进度条
        Console::endProgress();

        //成品出库单
        $wmsProductOutSheetModels = \core\models\WmsProductOutSheet::find()->where(['common_producer_info_id'=>14, 'piece_type'=>2])->orderBy(['id'=>SORT_ASC])->all();
        //设置进度条
        Console::startProgress(0, count($wmsProductOutSheetModels));
        foreach ($wmsProductOutSheetModels as $key => $wmsProductOutSheetModel){
            $transtion = \Yii::$app->db->beginTransaction();
            try {
                //提示文字
                $tip = Console::ansiFormat(PHP_EOL . "正在处理成品出库单{$wmsProductOutSheetModel->wms_product_out_sheet_number}", [Console::FG_YELLOW]);
                Console::output("{$tip}");
                unset($tip);
                //已经处理过则无需再次处理
                if (\Yii::$app->redis->sismember("in_sheet_number_updated_piece_type_for_shanxi", $wmsProductOutSheetModel->wms_product_out_sheet_number)) {
                    throw new \yii\db\Exception(PHP_EOL . "成品出库单{$wmsProductOutSheetModel->wms_product_out_sheet_number}已经更新过,无需再次更新");
                }

                //核心处理逻辑
                $updateHandler = $wmsProductOutSheetModel->updatePieceTypeFrom2To1();
                if (!$updateHandler['status']){
                    throw new \yii\db\Exception($updateHandler['errmsg']);
                }else{
                    //提示文字
                    $tip = Console::ansiFormat(PHP_EOL . "正在处理成品出库单{$wmsProductOutSheetModel->wms_product_out_sheet_number}成功", [Console::FG_GREEN]);
                    Console::output("{$tip}");
                    unset($tip);
                }

                //更新进度条
                Console::updateProgress($key + 1, count($wmsProductOutSheetModels));
                //如果是模拟更新则回滚
                if (!$is_update) {
                    throw new \yii\db\Exception(PHP_EOL . "此次是模拟更新,不会真正更新,正在回滚到初始状态!");
                }
                $transtion->commit();
                //将已更新的入库单号添加到已更新队列中
                \Yii::$app->redis->sadd("in_sheet_number_updated_piece_type_for_shanxi", $wmsProductOutSheetModel->wms_product_out_sheet_number);
            } catch (\yii\base\Exception $e) {
                $transtion->rollBack();
                $tip = Console::ansiFormat(PHP_EOL . $e->getMessage(), [Console::FG_RED]);
                Console::output("{$tip}");
                unset($tip);
            } catch (\yii\base\ErrorException $e) {
                $transtion->rollBack();
                $tip = Console::ansiFormat(PHP_EOL . $e->getMessage(), [Console::FG_RED]);
                Console::output("{$tip}");
                unset($tip);
            }
            unset($wmsProductOutSheetModel);
        }
        unset($wmsProductOutSheetModels);
        //设置进度条
        Console::endProgress();
    }

模拟执行的过程如下所示:

真实更新的过程如下:

Yii2控制台程序最佳实践的更多相关文章

  1. YII2 随笔 视图最佳实践

    yii\base\Controller::render(): 渲染一个 视图名 并使用一个 布局 返回到渲染结果. yii\base\Controller::renderPartial(): 渲染一个 ...

  2. 【转】优化Web程序的最佳实践

    自动排版有点乱,看着蛋疼,建议下载中文PDF版阅读或阅读英文原文. Yahoo!的Exceptional Performance团队为改善Web性能带来最佳实践.他们为此进行了 一系列的实验.开发了各 ...

  3. paip.java win程序迁移linux的最佳实践

    paip.java win程序迁移linux的最佳实践 1.class load路径的问题... windows哈第一的从calsses目录加载,,而linux优先从jar加载.. 特别的是修理了ja ...

  4. Windows Azure 安全最佳实践 - 第 6 部分:Azure 服务如何扩展应用程序安全性

    多种Windows Azure服务可以帮助您将应用程序安全性扩展到云. 有三种服务可提供多个提供程序之间的身份标识映射.内部部署数据中心间的连接和相互发送消息的应用程序功能(无论应用程序位于何处). ...

  5. Asp.NetCore程序发布到CentOs(含安装部署netcore)--最佳实践(二)

    Asp.NetCore程序发布到CentOs(含安装部署netcore)--最佳实践(一) 接上一篇 3. Nginx配置反向代理 3.1 cnetos 安装nginx 首先,我们需要在服务器上安装N ...

  6. Asp.NetCore程序发布到CentOs(含安装部署netcore)--最佳实践(一)

    环境 本地 win7 服务器:Virtual Box 上的Centos ssh工具: Xshell 文件传输: xftp 1.在本地创建asp.net core应用发布 1.1 使用Vs2017 新建 ...

  7. mpvue最佳实践 , 美团出的一个小程序框架

    看手机微信,看到说美团出了1个小程序框架,  mpvue 搜下来试试,看了网上的一个对比 ----------------- 以下为引用 我们对微信小程序.mpvue.WePY 这三个开发框架的主要能 ...

  8. Atitit. Gui控件and面板----程序快速启动区--最佳实践Launchy ObjectDock-o0g

    Atitit.  Gui控件and面板----程序快速启动区--最佳实践Launchy ObjectDock-o0g 两个方式::: 键盘式::先用热键呼叫出QS,然后开始输入程序中的部分字母,按En ...

  9. Javascript模块化编程(一)模块的写法最佳实践六、输入全局变量 独立性是模块的重要特点,模块内部最好不与程序的其他部分直接交互。 为了在模块内部调用全局变量,必须显式地将其他变量输入模块。

    Javascript模块化编程,已经成为一个迫切的需求.理想情况下,开发者只需要实现核心的业务逻辑,其他都可以加载别人已经写好的模块但是,Javascript不是一种模块化编程语言,它不支持类clas ...

随机推荐

  1. 总结 Linux 下安装 PHP 扩展步骤

    总结一下 Linux 下安装 PHP 扩展步骤,这里以安装 PHP 的 redis 扩展为例. 一.拿到扩展包下载地址,下载扩展包 pecl 上搜索 redis wget http://pecl.ph ...

  2. svn解决冲突问题

    1.在linux的branchs下创建了一个新的分支branch_2后,commit.提示: 2.这是把/my_branch/index.html删除后svn update也不能解决问题 3.到myb ...

  3. Java多线程(三)—— synchronized关键字详解

    一.多线程的同步 1.为什么要引入同步机制 在多线程环境中,可能会有两个甚至更多的线程试图同时访问一个有限的资源.必须对这种潜在资源冲突进行预防. 解决方法:在线程使用一个资源时为其加锁即可. 访问资 ...

  4. 使用python进行utf9编码和解码

    在2005年4月1日(也就是愚人节),IEEE的rfc4042文件规定了utf9和utf18这2个所谓的Unicode的高效转换格式. 具体的格式说明,有兴趣的话点击上面的rfc4042链接去观看. ...

  5. [Spark][Python][DataFrame][RDD]从DataFrame得到RDD的例子

    [Spark][Python][DataFrame][RDD]从DataFrame得到RDD的例子 $ hdfs dfs -cat people.json {"name":&quo ...

  6. [C#] LINQ之SelectMany

    声明:本文为www.cnc6.cn原创,转载时请注明出处,谢谢! 一.第一种用法: public static IEnumerable<TResult> SelectMany<TSo ...

  7. abbix通过JMX监控Tomcat(被监控端安装Tomat的服务器防火墙策略iptables配置)

    原文地址:http://jaychang.iteye.com/blog/2214830 一.目前的环境 被监控端192.168.153.191 /usr/local/tomcat 下载了catalin ...

  8. Python常用内建方法:__init__,__new__,__class__的理解

    python中所有类都是继承自object, 而object提供了很多原始的内建属性和方法,所以用户自定义的类在Python中也会继承这些内建属性.可以使用dir()函数可以查看,虽然python提供 ...

  9. JVM原理分析

    1 什么是JVM? JVM是Java Virtual Machine(Java虚拟机)的缩写,是通过在实际的计算机上仿真模拟各种计算机功能来实现的.由一套字节码指令集.一组寄存器.一个栈.一个垃圾回收 ...

  10. java中流的简单小结

    1.分类 按字节流分: InputStream(输出流)     OutputStream(输入流) 按字符流分: Reader Writer  提示:输入.输出是站在程序的角度而言,所有输入流是“读 ...