主頁 > 軟體設計 > 專案實戰:Qt+Ffmpeg+OpenCV相機程式(打開攝像頭、支持多種攝像頭、解析度調整、翻轉、旋轉、亮度調整、拍照、錄像、回放圖片、回放錄像)

專案實戰:Qt+Ffmpeg+OpenCV相機程式(打開攝像頭、支持多種攝像頭、解析度調整、翻轉、旋轉、亮度調整、拍照、錄像、回放圖片、回放錄像)

2020-09-10 17:06:31 軟體設計

若該文為原創文章,未經允許不得轉載
原博主博客地址:https://blog.csdn.net/qq21497936
原博主博客導航:https://blog.csdn.net/qq21497936/article/details/102478062
本文章博客地址:https://blog.csdn.net/qq21497936/article/details/108489004

紅胖子(紅模仿)的博文大全:開發技術集合(包含Qt實用技術、樹莓派、三維、OpenCV、OpenGL、ffmpeg、OSG、單片機、軟硬結合等等)持續更新中…(點擊傳送門)

開發專欄:專案實戰(點擊傳送門)

OpenCV開發專欄(點擊傳送門)


需求

??嵌入式windows設備上的相機程式,
??打開攝像頭,兼容多種攝像頭,攝像頭解析度切換(攝像頭管理),
??對影像進行翻轉、旋轉、亮度調整(影像基本演算法管理)
??對調整后的影像進行拍照、延時拍照,
??對調整后的影像進行錄像(編碼錄制),
??對照片和錄像進行回看(圖片瀏覽器、視頻播放器)
??長時間運行穩定,


Demo

??在這里插入圖片描述
??在這里插入圖片描述
??在這里插入圖片描述
??在這里插入圖片描述
??在這里插入圖片描述
??在這里插入圖片描述
??在這里插入圖片描述

體驗下載地址

??CSDN:https://download.csdn.net/download/qq21497936/12827160
??QQ群:1047134658(點擊“檔案”搜索“camera”,群內與博文同步更新)


原理

??使用ffmpeg處理攝像頭、使用OpenCV處理錄像和播放;


相關博客

《專案實戰:Qt+ffmpeg攝像頭檢測工具》
《專案實戰:Qt+OpenCV視頻播放器(支持播放器操作,如暫停、恢復、停止、時間、進度條拽托等)》
《OpenCV開發筆記(四):OpenCV圖片和視頻資料的讀取與存盤》
《FFmpeg開發筆記(一):ffmpeg介紹、windows開發環境搭建(mingw和msvc)》


v1.5.0功能

  • 打開攝像頭,兼容多種攝像頭,攝像頭解析度切換(攝像頭管理),
  • 對影像進行翻轉、旋轉、亮度調整(影像基本演算法管理)
  • 對調整后的影像進行拍照、延時拍照,
  • 對調整后的影像進行錄像(編碼錄制),
  • 對照片和錄像進行回看(圖片瀏覽器、視頻播放器)

Demo核心代碼

FfmpegCameraManager.h:攝像頭管理類

#ifndef FFMPEGCAMERAMANAGER_H
#define FFMPEGCAMERAMANAGER_H

/************************************************************\
 * 控制元件名稱: FfmpegCameraManager, ffmpeg管理類(用于攝像頭操作)
 * 控制元件描述:
 *          1.打開攝像頭
 *          2.支持動態切換解析度
 * 作者:紅模仿    聯系方式:QQ21497936
 * 博客地址:https://blog.csdn.net/qq21497936
 *       日期                版本               描述
 *    2018年09年14日     v1.0.0         ffmpeg模塊封裝空類
 *    2020年09年05日     v1.1.0         ffmpeg打開攝像頭,支持的動態解析度切換
 *    2020年09年08日     v1.2.0         兼容各種攝像頭,解決記憶體溢位bug,對最高幀率做了支持范圍內的限制
 *                                     限制幀率一般為25fps(除非最大小于25fps或者最小大于25fps)
\************************************************************/

#include <QObject>
#include <QString>
#include <QDebug>
#include <QTimer>
#include <QThread>
#include <QImage>
#include <QProcess>
#include <QMessageBox>
#include <QDateTime>

extern "C" {
    #include "libavcodec/avcodec.h"
    #include "libavformat/avformat.h"
    #include "libswscale/swscale.h"
    #include "libavdevice/avdevice.h"
    #include "libavformat/version.h"
    #include "libavutil/time.h"
    #include "libavutil/mathematics.h"
    #include "libavformat/avformat.h"
    #include "libswscale/swscale.h"
    #include "libswresample/swresample.h"
    #include "errno.h"
    #include "error.h"
}

#define LOG qDebug()<<__FILE__<<__LINE__<<QDateTime::currentDateTime().toString("hh:mm:ss:zzz")

class FfmpegCameraManager : public QObject
{
    Q_OBJECT
public:

public:
    explicit FfmpegCameraManager(QObject *parent = nullptr);

signals:
    void signal_captureOneFrame(QImage image);

public:
    static QString getAvcodecConfiguration();

public:
    bool init();
    bool openUsbCamera();
    QString getUsbCameraName();
    QList<QString> getUsbCameraInfo();
    int getCurrentFps();
    int getCurrentSizeFpsIndex();

    QList<QSize> getListSize() const;

public slots:
    void slot_start();
    void slot_stop();
    void slot_setSizeFps(int index);

protected slots:
    void slot_captureOneFrame();

signals:

public slots:


private:
    static bool _init;

    AVFormatContext *_pAVFormatContext;         // 全域背景關系
    AVInputFormat *_pAVInputFormat;
    AVDictionary* _pAVDictionary;               // 打開編碼器的配置

    AVCodecContext *_pAVCodecContextForAudio;   // 音頻解碼器背景關系
    AVCodecContext *_pAVCodecContextForVideo;   // 視頻解碼器背景關系(不帶音頻)

    AVCodec * _pAVCodecForAudio;                // 音頻解碼器
    AVCodec * _pAVCodecForVideo;                // 視頻解碼器(不帶音頻)

    int _streamIndexForAudio;                   // 音頻流序號
    int _streamIndexForVideo;                   // 視頻流序號

    SwrContext *_pSwrContextForAudio;           // 音頻轉換背景關系

    bool _running;
    bool _first;
    bool _opened;
    uint8_t *_pOutBuffer;

    AVFrame * _pFrame;
    AVFrame * _pFrameRGB;
    AVPacket *_pAVPacket;
    SwsContext *_pSwsContext;

    int _videoIndex;

    QString _cameraDescription;

    QList<QSize> _listSize;
    QList<int> _listFps;
    QList<QString> _listSizeFpsInfo;
    int _currentSizeFpsIndex;
};

#endif // FfmpegCameraManager_H

FfmpegCameraManager.cpp:攝像頭管理類

...
bool FfmpegCameraManager::openUsbCamera()
{
    if(!_init)
    {
        LOG << "未初始化";
        return true;
    }
    _pAVInputFormat = av_find_input_format("dshow");
    if(!_pAVInputFormat)
    {
        LOG << "Failed to av_find_input_format";
        return false;
    }
    if(_cameraDescription == "")
    {
        LOG << "無攝像頭";
        return false;
    }
    QString cameraDescription = QString("video=%1").arg(_cameraDescription);
    if(_listSizeFpsInfo.size() == 0)
    {
        LOG << "未獲取到解析度和幀率";
        return false;
    }
    // 設定解析度
    av_dict_set(&_pAVDictionary,
                "video_size",
                QString("%1x%2").arg(_listSize.at(_currentSizeFpsIndex).width())
                                .arg(_listSize.at(_currentSizeFpsIndex).height()).toUtf8().data(),
                0);
    // 設定幀率
    int frame = _listFps.at(_currentSizeFpsIndex);

    av_dict_set(&_pAVDictionary,
                "framerate",
                QString("%1").arg(frame).toUtf8().data(),
                0);

    LOG << "打開攝像頭:" << _cameraDescription
        << "解析度:" << _listSize.at(_currentSizeFpsIndex).width() << "x" << _listSize.at(_currentSizeFpsIndex).height()
        << "幀率:" << _listFps.at(_currentSizeFpsIndex);

    if(avformat_open_input(&_pAVFormatContext,
                           cameraDescription.toUtf8().data(),
                           _pAVInputFormat,
                           &_pAVDictionary) != 0)
    {
        LOG << "打開攝像頭失敗";
        return false;
    }
    LOG << "打開攝像頭成功";
    _first = true;
    _opened = true;
    QTimer::singleShot(0, this, SLOT(slot_captureOneFrame()));
    return true;
}
...

OpenCVManager.h:錄像與播放視頻類

#ifndef OPENCVMANAGER_H
#define OPENCVMANAGER_H

/************************************************************\
 * 控制元件名稱: OpenCVManager,OpenCV管理類
 * 控制元件描述:
 *          1.OpenCV操作支持
 *          2.支持錄像(.avi格式)
 * 作者:紅模仿    聯系方式:QQ21497936
 * 博客地址:https://blog.csdn.net/qq21497936
 *       日期             版本               描述
 *   2019年11月09日      v1.0.0   opencv拍照和錄像Demo
 *   2020年09月07日      v1.1.0   增加了單純錄像的介面
\************************************************************/

#include <QObject>
#include <QImage>
#include <QDateTime>
#include <QTimer>
// opencv
#include "opencv/highgui.h"
#include "opencv/cxcore.h"
#include "opencv2/core/core.hpp"
#include "opencv2/highgui/highgui.hpp"
#include "opencv2/opencv.hpp"

class OpenCVManager : public QObject
{
    Q_OBJECT
public:
    explicit OpenCVManager(QObject *parent = nullptr);
    ~OpenCVManager();

public:
    QString getWindowTitle() const;
    double getBrightness();
    double getContrast() const;
    double getSaturation() const;
    double getHue() const;
    double getGain() const;
    bool getShowProperty() const;
    double getExposure() const;
    int getRotate() const;
    bool getMirror() const;

public:
    void setBrightness(double value);
    void setContrast(double value);
    void setSaturation(double value);
    void setHue(double value);
    void setGain(double value);
    void setShowProperty(bool value);
    void setExposure(double value);
    void setRotate(int rotate);
    void setMirror(bool mirror);


signals:
    void signal_captureOneFrame(cv::Mat mat);   // 接收影像后拋出信號

public:
    bool startCapture(int usb, int width = 1280, int height = 720);
                                        // 打開攝像頭, 0...
    bool startCapture(QString url, int width = 1280, int height = 720);
                                        // 打開攝像頭, 網路攝像頭地址
    bool stopCapture();                 // 關閉攝像頭
    void startRecord(QString filePath); // 開始錄像(使用的是opencv打開的攝像頭)
    void stopRecord();                  // 停止錄像(使用的是opencv打開的攝像頭)

public slots:
    void slot_inputRecordImage(QImage image);
    void slot_stopRecordFormOut();

public:     // 單獨的一塊業務,使用的是開始錄像后,從類外面輸入QImage進行錄像
    void startRecordFromOut(QString filePath, int fps);
    void inputRecordImage(QImage image);
    void stopRecordFormOut();

public slots:
    bool start();                       // 開啟執行緒
    bool stop();                        // 關閉執行緒

protected slots:
    void slot_captrueFrame();           // 訊息回圈獲取影像
    void slot_stopCapture();            // 當正在采集中時(>>時),關閉攝像頭會導致程式崩潰,所以采集與停止放一個執行緒中(訊息回圈)

protected slots:
    void slot_startRecord(QString filePath);        // 錄像(使用的是opencv打開的攝像頭)
    void slot_stopRecord();                         // 停止錄屏(使用的是opencv打開的攝像頭)

public:
    static QImage cvMat2QImage(const cv::Mat &mat);
    static cv::Mat image2Mat(QImage image);    // Qimage 轉 cv::Mat
    static QImage mat2Image(cv::Mat mat);      // cv::Mat 轉 QImage

private:
    cv::VideoCapture *_pVideoCapture;   // 攝像頭實體
    cv::VideoWriter *_pVideoWrite;      // 錄像實體
    QString _recordFilePath;            // 錄制檔案路徑

    bool _running;                      // 執行緒是否運行
    bool _showProperty;                 // 是否顯示屬性引數
    double _brightness;                 // 亮度
    double _contrast;                   // 對比度
    double _saturation;                 // 飽和度
    double _hue;                        // 色調
    double _gain;                       // 增益
    double _exposure;                   // 曝光度

    int _width;                         // 寬度
    int _height;                        // 高度
    bool _recording;                    // 標志是否正在錄像
    bool _startRecording;

    int _rotate;                        // 旋轉度數
    bool _mirror;                       // 是否翻轉
    int _fps;                           // 幀率
    int _index;                         // 幀序號

private:
    cv::VideoWriter *_pVideoWriteForOut; // 錄像實體(從外部輸入影像,非從opencv打開攝像頭)
    QString _recordFilePathForOut;       // 錄像檔案路徑(從外部輸入影像,非從opencv打開攝像頭)

private:
    QString _windowTitle;

};

#endif // OPENCVMANAGER_H

OpenCVManager.h:錄像與播放視頻類

...
void OpenCVManager::inputRecordImage(QImage image)
{
    if(!_startRecording)
    {
        return;
    }
    cv::Mat mat = image2Mat(image);

    if(!_recording)
    {
        QString ext = _recordFilePath.mid(_recordFilePathForOut.lastIndexOf(".") + 1);
        int cvFourcc = 0;
        if(ext == "mpg")
        {
            cvFourcc = CV_FOURCC('D','I','V','X');
            qDebug() << __FILE__ << __LINE__<< ext << "DIVX" << cvFourcc;
        }else if(ext == "avi")
        {
            cvFourcc = CV_FOURCC('M','J','P','G');
            qDebug() << __FILE__ << __LINE__<< ext << "avi" << cvFourcc;
        }else if(ext == "mp4")
        {
            // mp4目前錄制不成功(可以生成檔案,但是打開失敗)
            cvFourcc = CV_FOURCC('M','P','4','2');
            qDebug() << __FILE__ << __LINE__<< ext << "MP42" << cvFourcc;
        }
        qDebug() << __FILE__ << __LINE__ << mat.type() << mat.channels();
        _pVideoWriteForOut->open(_recordFilePath.toStdString(), cvFourcc, _fps, cv::Size(mat.cols, mat.rows));

        std::vector<cv::Mat> listMat;
        cv::split(mat, listMat);
        std::vector<cv::Mat> listMat2;
        // 由于opencv對avi中mat的限制大小只能為0xFFFF,修改原始碼突破限制為0xFFFFFFFF后
        // 在錄像時,發現錄入的mat是正確的,錄制出來通道顏色變換了,需要手動對顏色通道進行修正
        // 注意:僅限avi使用mjpg編碼格式
        // 1 2 0 偏綠 
        // 0 1 2 偏藍
        // 0 2 1 偏綠
        // 1 2 3 嚴重不對
        // 2 0 1 偏藍
        // 2 1 0 偏藍
        listMat2.push_back(listMat.at(0));
        listMat2.push_back(listMat.at(1));
        listMat2.push_back(listMat.at(2));
        cv::merge(listMat2, mat);

        _pVideoWriteForOut->write(mat);
        _recording = true;
    }else{

        std::vector<cv::Mat> listMat;
        cv::split(mat, listMat);
        std::vector<cv::Mat> listMat2;
        listMat2.push_back(listMat.at(0));
        listMat2.push_back(listMat.at(1));
        listMat2.push_back(listMat.at(2));
        cv::merge(listMat2, mat);
        _pVideoWriteForOut->write(mat);
    }
}
...

若該文為原創文章,未經允許不得轉載
原博主博客地址:https://blog.csdn.net/qq21497936
原博主博客導航:https://blog.csdn.net/qq21497936/article/details/102478062
本文章博客地址:https://blog.csdn.net/qq21497936/article/details/108489004

轉載請註明出處,本文鏈接:https://www.uj5u.com/ruanti/2818.html

標籤:其他

上一篇:HarmonyOS應用開發—UI開發與預覽

下一篇:Python音視頻開發:消除抖音短視頻Logo的圖形化工具實作

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 面試突擊第一季,第二季,第三季

    第一季必考 https://www.bilibili.com/video/BV1FE411y79Y?from=search&seid=15921726601957489746 第二季分布式 https://www.bilibili.com/video/BV13f4y127ee/?spm_id_fro ......

    uj5u.com 2020-09-10 05:35:24 more
  • 第三單元作業總結

    1.前言 這應該是本學期最后一次寫作業總結了吧。總體來說,對作業的節奏也差不多掌握了,作業做起來的效率也更高了。雖然和之前的作業一樣,作業中都要用到新的知識,但是相比之前,更加懂得了如何利用工具以及資料。雖然之間卡過殼,但總體而言,這幾次作業還算完成的比較好。 2.作業程序總結 相比前兩個單元,此單 ......

    uj5u.com 2020-09-10 05:35:41 more
  • 北航OO(2020)第四單元博客作業暨課程總結博客

    北航OO(2020)第四單元博客作業暨課程總結博客 本單元作業的架構設計 在本單元中,由于UML圖具有比較清晰的樹形結構,因此我對其中需要進行查詢操作的元素進行了包裝,在樹的父節點中存盤所有孩子的參考。考慮到性能問題,我采用了快取機制,一次查詢后盡可能快取已經遍歷過的資訊,以減少遍歷次數。 本單元我 ......

    uj5u.com 2020-09-10 05:35:48 more
  • BUAA_OO_第四單元

    一、UML決議器設計 ? 先看下題目:第四單元實作一個基于JDK 8帶有效性檢查的UML(Unified Modeling Language)類圖,順序圖,狀態圖分析器 MyUmlInteraction,實際上我們要建立一個有向圖模型,UML中的物件(元素)可能與同級元素連接,也可與低級元素相連形成 ......

    uj5u.com 2020-09-10 05:35:54 more
  • 6.1邏輯運算子

    邏輯運算子 1. && 短路與 運算式1 && 運算式2 01.運算式1為true并且運算式2也為true 整體回傳為true 02.運算式1為false,將不會執行運算式2 整體回傳為false 03.只要有一個運算式為false 整體回傳為false 2. || 短路或 運算式1 || 運算式2 ......

    uj5u.com 2020-09-10 05:35:56 more
  • BUAAOO 第四單元 & 課程總結

    1. 第四單元:StarUml檔案決議 本單元采用了圖模型決議UML。 UML檔案可以抽象為圖、子圖、邊的邏輯結構。 在實作中,圖的節點包括類、介面、屬性,子圖包括狀態圖、順序圖等。 采用了三次遍歷UML元素的方法建圖,第一遍遍歷建點,第二、三次遍歷設定屬性、連邊,實作圖物件的初始化。這里借鑒了一些 ......

    uj5u.com 2020-09-10 05:36:06 more
  • 談談我對C# 多型的理解

    面向物件三要素:封裝、繼承、多型。 封裝和繼承,這兩個比較好理解,但要理解多型的話,可就稍微有點難度了。今天,我們就來講講多型的理解。 我們應該經常會看到面試題目:請談談對多型的理解。 其實呢,多型非常簡單,就一句話:呼叫同一種方法產生了不同的結果。 具體實作方式有三種。 一、多載 多載很簡單。 p ......

    uj5u.com 2020-09-10 05:36:09 more
  • Python 資料驅動工具:DDT

    背景 python 的unittest 沒有自帶資料驅動功能。 所以如果使用unittest,同時又想使用資料驅動,那么就可以使用DDT來完成。 DDT是 “Data-Driven Tests”的縮寫。 資料:http://ddt.readthedocs.io/en/latest/ 使用方法 dd. ......

    uj5u.com 2020-09-10 05:36:13 more
  • Python里面的xlrd模塊詳解

    那我就一下面積個問題對xlrd模塊進行學習一下: 1.什么是xlrd模塊? 2.為什么使用xlrd模塊? 3.怎樣使用xlrd模塊? 1.什么是xlrd模塊? ?python操作excel主要用到xlrd和xlwt這兩個庫,即xlrd是讀excel,xlwt是寫excel的庫。 今天就先來說一下xl ......

    uj5u.com 2020-09-10 05:36:28 more
  • 當我們創建HashMap時,底層到底做了什么?

    jdk1.7中的底層實作程序(底層基于陣列+鏈表) 在我們new HashMap()時,底層創建了默認長度為16的一維陣列Entry[ ] table。當我們呼叫map.put(key1,value1)方法向HashMap里添加資料的時候: 首先,呼叫key1所在類的hashCode()計算key1 ......

    uj5u.com 2020-09-10 05:36:38 more
最新发布
  • 【中介者設計模式詳解】C/Java/JS/Go/Python/TS不同語言實作

    * 中介者模式是一種行為型設計模式,它可以用來減少類之間的直接依賴關系,
    * 將物件之間的通信封裝到一個中介者物件中,從而使得各個物件之間的關系更加松散。
    * 在中介者模式中,物件之間不再直接相互互動,而是通過中介者來中轉訊息。 ......

    uj5u.com 2023-04-20 08:20:47 more
  • 露天煤礦現場調研和交流案例分享

    他們集團的資訊化公司及研究院在一個礦區正在做智能礦山的統一平臺的 試點,專案投資大概1億,包括了礦山的各方面的內容,顯示得我們這次交流有點多余。他們2年前開始做智能礦山的規劃,有很多煤礦行業專家的加持,他們的描述是非常完美,但是去年底應該上線的平臺,現在還沒有看到影子。他們確實有很多場景需求,但是被... ......

    uj5u.com 2023-04-20 08:20:25 more
  • 《社區人員管理》實戰案例設計&個人案例分享

    設計是一個讓人夢想成真程序,開始編碼、測驗、除錯之前進行需求分析和架構設計,才能保證關鍵方面都做正確 ......

    uj5u.com 2023-04-20 08:20:17 more
  • 軟體架構生態化-多角色交付的探索實踐

    作為一個技術架構師,不僅僅要緊跟行業技術趨勢,還要結合研發團隊現狀及痛點,探索新的交付方案。在日常中,你是否遇到如下問題 “ 業務需求排期長研發是瓶頸;非研發角色感受不到研發技改提效的變化;引入ISV 團隊又擔心質量和安全,培訓周期長“等等,基于此我們探索了一種新的技術體系及交付方案來解決如上問題。 ......

    uj5u.com 2023-04-20 08:20:10 more
  • 【中介者設計模式詳解】C/Java/JS/Go/Python/TS不同語言實作

    * 中介者模式是一種行為型設計模式,它可以用來減少類之間的直接依賴關系,
    * 將物件之間的通信封裝到一個中介者物件中,從而使得各個物件之間的關系更加松散。
    * 在中介者模式中,物件之間不再直接相互互動,而是通過中介者來中轉訊息。 ......

    uj5u.com 2023-04-20 08:19:44 more
  • 露天煤礦現場調研和交流案例分享

    他們集團的資訊化公司及研究院在一個礦區正在做智能礦山的統一平臺的 試點,專案投資大概1億,包括了礦山的各方面的內容,顯示得我們這次交流有點多余。他們2年前開始做智能礦山的規劃,有很多煤礦行業專家的加持,他們的描述是非常完美,但是去年底應該上線的平臺,現在還沒有看到影子。他們確實有很多場景需求,但是被... ......

    uj5u.com 2023-04-20 08:19:07 more
  • 《社區人員管理》實戰案例設計&個人案例分享

    設計是一個讓人夢想成真程序,開始編碼、測驗、除錯之前進行需求分析和架構設計,才能保證關鍵方面都做正確 ......

    uj5u.com 2023-04-20 08:18:57 more
  • 軟體架構生態化-多角色交付的探索實踐

    作為一個技術架構師,不僅僅要緊跟行業技術趨勢,還要結合研發團隊現狀及痛點,探索新的交付方案。在日常中,你是否遇到如下問題 “ 業務需求排期長研發是瓶頸;非研發角色感受不到研發技改提效的變化;引入ISV 團隊又擔心質量和安全,培訓周期長“等等,基于此我們探索了一種新的技術體系及交付方案來解決如上問題。 ......

    uj5u.com 2023-04-20 08:18:49 more
  • 05單件模式

    #經典的單件模式 public class Singleton { private static Singleton uniqueInstance; //一個靜態變數持有Singleton類的唯一實體。 // 其他有用的實體變數寫在這里 //構造器宣告為私有,只有Singleton可以實體化這個類! ......

    uj5u.com 2023-04-19 08:42:51 more
  • 【架構與設計】常見微服務分層架構的區別和落地實踐

    軟體工程的方方面面都遵循一個最基本的道理:沒有銀彈,架構分層模型更是如此,每一種都有各自優缺點,所以請根據不同的業務場景,并遵循簡單、可演進這兩個重要的架構原則選擇合適的架構分層模型即可。 ......

    uj5u.com 2023-04-19 08:42:41 more