异常处理/CC++ 中 assert 断言 应用实践和注意事项

news2025/1/16 14:52:28

文章目录

  • 概述
  • assert 本质浅析
  • Release版本下的assert是否生效
    • 默认设置下 QtCreator环境 assert 过程
    • 默认配置下 VS环境 assert 过程
    • 配置VS发布模式下的断言生效
    • VS环境Release版本的UI程序
    • Release下请当我不生效
  • 请勿滥用assert
    • 导致逻辑错误
    • 再强调'不要在assert内执行逻辑功能'
    • 怎敢默认release下绝不会发生此错误?
    • 要不要在Release版本下使用断言
    • 使用assert的其他建议
  • 静态断言
  • 自定义断言
  • 小结

概述

本文主要讲解了 assert 断言机制,在编程中的作用和注意事项,如 assert 的工作原理、Release程序版本下的断言生效问题、为什么要杜绝在assert内执行逻辑、如何自定义断言等。断言机制是在开发和调试阶段快速发现程序中的错误和逻辑问题的重要手段,它可以帮助开发人员在程序中插入检查点,以验证程序的正确性和健壮性,一旦发现断言失败,开发人员可以通过查看错误消息和堆栈跟踪来定位和解决问题。

转载请标明原文链接,
https://blog.csdn.net/quguanxin/category_6223029.html

assert 本质浅析

标准C/C++库中的 assert 并不是一个函数,事实上,它是个宏,其用法像是一种"契约式编程",其表达的意思是,程序在我的假设条件下,能够正常良好的运作,否则就告警并调用 abort 终止程序。其使用场景大概是这样的,大多数情况下,我们要进行验证的假设,只是属于偶然性事件,又或者我们仅仅想测试一下,一些最坏情况是否发生。

int main() {
    printf("当前源文件名:%s, 当前代码行号:%s\n", FILE_NAME, __LINE__);
    assert(0 == 1);
    system("pause"); return 0;
}

在这里插入图片描述
通过上述测试,可以猜测,assert 内部极有可能是封装了 FILELINE 预处理宏的。看一下源码,

#ifdef NDEBUG
    #define assert(expression) ((void)0)
#else
    _ACRTIMP void __cdecl _wassert(
        _In_z_ wchar_t const* _Message,
        _In_z_ wchar_t const* _File,
        _In_   unsigned       _Line
        );

    #define assert(expression) (void)(                                                       \
            (!!(expression)) ||                                                              \
            (_wassert(_CRT_WIDE(#expression), _CRT_WIDE(__FILE__), (unsigned)(__LINE__)), 0) \
        )
#endif

与我们《》自定义函数的使用方法一致,先以 char* 和 int 为参数类型,定义函数,然后,使用宏封装此函数,这样,LINEFILE 就可以用来代表宏函数(assert) 的调用位置。

Release版本下的assert是否生效

验证下,assert断言在发布模式下是否起作用-/-
能有多灵验呢,能否在崩溃的时候,显示出来—
实际运行中,我在发布版里头没怎么体现出来呢?
最好的办法还是为自己的软件编写错误处理框架–

默认设置下 QtCreator环境 assert 过程

void Widget::on_pushButton_clicked()
{
    ui->label->setText("before assert");
    //assert(false);  //Line:22
    Q_ASSERT(false);  //Line:23
    ui->label->setText("after assert");
}

使用C库的assert测试
编译Debug版本和Release版本,脱离开发环境进行执行测试,结果是都可以准确的定位到断言所在的代码行。因此我们初步得出的结论是,在QtCreator开发环境下,默认的情况下,Release版本的程序中,C库中的assert是可以生效的。

使用Qt库的Q_ASSERT测试
使用Qt自带的Q_ASSERT断言宏时,经过测试,在Release版本的程序中并不生效。

#if !defined(Q_ASSERT)
#  if defined(QT_NO_DEBUG) && !defined(QT_FORCE_ASSERTS)
#    define Q_ASSERT(cond) static_cast<void>(false && (cond))
#  else
#    define Q_ASSERT(cond) ((cond) ? static_cast<void>(0) : qt_assert(#cond, __FILE__, __LINE__))
#  endif
#endif

/*
  The Q_ASSERT macro calls this function when the test fails.
*/
void qt_assert(const char *assertion, const char *file, int line) Q_DECL_NOTHROW
{
    QMessageLogger(file, line, nullptr).fatal("ASSERT: \"%s\" in file %s, line %d", assertion, file, line);
}

通过上述Qt框架下的宏定义,我们可以看出,如果没有定义QT_FORCE_ASSERTS强制使用断言,而且定义了QT_NO_DEBUG(估计它在Release编译模式下会有定义),Q_ASSERT相当于没有定义。

默认配置下 VS环境 assert 过程

纯 C++ 代码

#include <iostream>
#include <assert.h>

int main() {
    int i = 0;
    std::cout << "Hello World!\n";
    assert(i != 0);
    std::cout << "Hello weifang!\n";
    system("pause");
}

在Debug模式下运行,断言如下。在Release模式下,断言不生效。
在这里插入图片描述

配置VS发布模式下的断言生效

还是上一小节中的项目,我们查看其项目属性,C/C++,预处理器,预处理器定义-
Debug配置:
在这里插入图片描述
Release配置:
在这里插入图片描述
我们针对Release模式下的预处理器定义,删除其中的NDEBUG宏,重新运行Release版本,
在这里插入图片描述
如上,只要删除NDEBUG宏定义,则assert在Release模式下也是生效的

注意一个现象,使用NDEBUG宏和删除该宏的情况下,编译生成的可执行文件的大小和占用空间大小保持一致不变。即使是在Release模式下预编译器配置上增加_DEBUG宏,其效果也是与不设置NDEBUG宏一致,与Debug模式下配置_DEBUG宏是不一样的。

另外,在 C++ 中,assert 宏通常在调试模式下才会生效,而在发布模式下会被编译器忽略掉。如果你希望在发布模式下也启用 assert 断言,可以在 “预处理器定义” 字段中,删除NDEBUG宏,或进一步添加 _DEBUG宏定义。如此便确保 _DEBUG 宏在发布模式和调试模式下都被定义。要注意的是, assert 断言主要用于在开发和调试阶段发现问题。在发布版本中,最好使用其他方式进行错误检查和处理,如异常处理、返回错误码或输出错误日志等。

VS环境Release版本的UI程序

在脱离开发环境的情况下,运行上一节的控制台测试程序。Debug版本程序(配有_DEBUG)的运行会弹出提示。Release版本程序(删除NDEBUG宏)的运行,在控制台有断言提示,没有弹窗提示,且控制台在数秒后自动退出。本小节我们进一步看看,在没有NDEBUG的情况下,R版的UI程序会是什么执行现象。

在VS下新建Qt项目,D版和R版的默认预处理器定义一致为$(Qt_DEFINES_);%(PreprocessorDefinitions),这里我暂时没有在Qt_DEFINES_变量中看到_DEBUG或NDEBUG宏的身影。它一定藏在其他地方了,我们暂不深究。

TestAssertUI::TestAssertUI(QWidget *parent) : QWidget(parent) , ui(new Ui::TestAssertUIClass())
{
    ui->setupUi(this);
    //
    connect(ui->pushButton, &QPushButton::clicked, []{
        int i = 0;
        assert(i != 0);
        qDebug() << "The assertion is not in effect";
    });
}

编译并执行my_D版本的程序,
在这里插入图片描述
编译并执行my_R版本的程序,断言并未生效。那,我们在my_R预处理器定义中增加 _DEBUG宏定义,重新编译执行。结果,断言依然是被优化不执行的。

还好我眼尖,在预处理器定义项目的下面发现了"取消预处理器定义"这个配置项,我将NDEBUG配置进去。
在这里插入图片描述
重新编译的过程提示,cl : 命令行 warning D9025: 正在重写“/DNDEBUG”(用“/UNDEBUG”),这一看就是起作用的样子啊。编译后执行,果然,与上述DEBUG版程序的弹窗告警一致

重点总结,
在使用VS做集成开发环境时,如果想使得R版本程序中C库的assert生效,需要在项目属性配置,C/C++,预处理器,取消预处理器,这个配置项中配置取消 “NDEBUG宏”。(仅)在预处理定义中增加_DEBUG宏定义是无效的,这可能是因为NDEBUG宏等项目属性中隐含的定义,会在编译过程靠后的阶段进行加载和覆盖。
在VS开发环境下,针对Release配置。其中非Qt项目,其预处理器定义中会直接标明NDEBUG,若想打开assert调试功能,只需要删除即可。针对Qt项目,由于NDEBUG没有直接定义在预处理器配置项中,你需要在取消预处理定义这个配置项中添加NDEBUG宏。

Release下请当我不生效

为了保险起见,一种可行的做法是,认为任何IDE下的任何ASSERT语句,是不生效的,不执行的。

请勿滥用assert

虽然 assert 方便好用,但勿滥用!

导致逻辑错误

知道assert在release模式下"可能"不被执行,可还是手欠。为了图省事,伪代码如下:

//!!危险!!
assert(SomethingBegin());
//函数定义
bool SomethingBegin()
{
	if (s_bBuseFlage)
		return false;
	//重置上下文	
	s_count = 0; s_bBuseFlage = false;
	return true;
}

如上,我assert了一个函数的执行返回值。在VS环境下,当编译release版本时,上述assert语句将被优化掉,也就是说其中的含有逻辑功能的函数将不会被执行,这必然导致运行异常。Debug下没有问题,release时抓瞎,通常会火烧眉毛。

再强调’不要在assert内执行逻辑功能’

//这条代码在R模式下可能不执行,从而导致逻辑异常
 assert(m_pObserver->Subscribe_Open(_ID_STREAM_INFO));

上述是曾经在实际项目中犯下的错误,而且那还是在总结过assert使用注意事项,已经很清楚assert内不可以执行逻辑代码。总归有那么几个脑子变浆糊的时刻,手欠。因此,遇到D模式和R模式执行不一致的情况,assert使用,算是一个检查点。

 //不要偷懒,只在assert中判断结果值
bOk &= m_pObserver->Subscribe_Open(_ID_STREAM_INFO);
assert(bOk);

怎敢默认release下绝不会发生此错误?

一种情形是这样的,Debug下并没有触发assert,但release下却发生了要捕获的异常,由于编码和编译原因,若release版本中assert被优化掉,那么,你相当于是放弃了对此异常情形的处理!那么就危险了,这种危险远不止是你的程序异常退出了一次,而是你没有什么可用信息去定位异常位置。

要不要在Release版本下使用断言

首先表明立场,不建议在Release版本中使用assert断言生效。你最好使用其他形式的错误处理和日志记录来补充 assert 断言,以提供更好的用户体验和容错能力。

好处:
错误检测:assert 断言是一种在运行时检测程序中的错误和异常的方法。在 Release 版本中启用 assert 断言可以帮助及早发现潜在的问题和错误,提高代码质量和可靠性。
调试信息:assert 断言通常会输出有关断言失败的相关信息,例如断言所在的文件和行号等。在 Release 版本中启用 assert 断言可以提供有用的调试信息,以便更好地理解和排查问题。
安全验证:通过启用 assert 断言,您可以在 Release 版本中对关键的安全验证进行检查。这可以帮助捕获潜在的安全漏洞和错误用法,提高程序的安全性。

坏处:
性能影响:assert 断言通常在断言失败时会导致程序终止。这会对程序的性能产生一定的负面影响。因此,在 Release 版本中启用 assert 断言可能会导致性能下降,尤其是在大规模或性能敏感的应用程序中。
用户体验:断言失败可能导致程序异常终止,这可能会对用户体验产生负面影响。在某些情况下,这可能会导致数据丢失或不可预测的行为。因此,在启用 assert 断言时需要谨慎权衡用户体验和错误检测的需求。
可移植性问题:某些平台或环境可能不支持 assert 断言,或者对其行为进行了修改。因此,在跨平台或跨环境的应用程序中,依赖于 assert 断言的特定行为可能会导致可移植性问题。

使用assert的其他建议

assert 在那些一次性执行的代码语句上可以使用,这些语句往往是非常的硬,如果一旦有错误错误,程序将没有继续运行的必要。
在那些动态频繁执行的函数中,如果使用assert来进行某些校验,往往会给自己带来不少麻烦。

虽然启用断言可能会带来性能影响和一些不太友好的用户体验,但在开发和调试阶段,启用断言可以帮助发现和修复潜在的错误和异常,从而提高代码质量和可靠性。在发布和部署阶段,需要谨慎权衡性能与错误检测需求之间的平衡。

像是动态内存申请之类的操作,强烈建议不要用assert检验返回结果。但有时候你实在懒得一坨,加个assert并打开release的DEBUG开关,也是种手段,这至少比你置之不理要好很多。因为空指针通常会导致程序毫无征兆的死掉,让你束手无策,那不仅不优雅,还会让运维和开发工程师很头疼。故,当你的软件没有完备的异常处理或日志记录机制,那就用assert做最后的救命稻草吧!

静态断言

C/C++ 中的静态断言机制(Static Assertion)是一种在编译时进行静态检查的机制,用于在编译器发现错误之前捕获问题。static_assert 接受一个编译时求值为布尔值的表达式作为参数,并在表达式为假时触发编译错误。如果表达式为真,则静态断言不会产生任何代码或运行时开销。如下,是ROS2.0 异常处理模块中的源码片段,

/// Struct wrapping a fixed-size c string used for returning the formatted error string.
typedef struct rcutils_error_string_s {
  /// The fixed-size C string used for returning the formatted error string.
  char str[RCUTILS_ERROR_MESSAGE_MAX_LENGTH];
} rcutils_error_string_t;  //该结构的长度与rcutils_error_state_t长度相同

/// Struct which encapsulates the error state set by RCUTILS_SET_ERROR_MSG().
typedef struct rcutils_error_state_s {
  /// User message storage, limited to RCUTILS_ERROR_STATE_MESSAGE_MAX_LENGTH characters.
  char message[RCUTILS_ERROR_STATE_MESSAGE_MAX_LENGTH];
  /// __FILE__宏代表的代码文件名 File name, limited to what's left from RCUTILS_ERROR_STATE_MAX_SIZE characters after subtracting storage for others.
  char file[RCUTILS_ERROR_STATE_FILE_MAX_LENGTH];
  /// __LINE__宏代表的代码行号 Line number of error.
  uint64_t line_number;
} rcutils_error_state_t;  //该结构的长度与rcutils_error_string_t长度相同

// make sure our math is right... //编译时进行静态断言from C++ 11
#if __STDC_VERSION__ >= 201112L
static_assert(
  sizeof(rcutils_error_string_t) == (  /* 1024 == 768 + 229 + 20 + 6 + 1(null terminating character) */
    RCUTILS_ERROR_STATE_MESSAGE_MAX_LENGTH + RCUTILS_ERROR_STATE_FILE_MAX_LENGTH + RCUTILS_ERROR_STATE_LINE_NUMBER_STR_MAX_LENGTH + RCUTILS_ERROR_FORMATTING_CHARACTERS + 1),
    "Maximum length calculations incorrect");
#endif

自定义断言

如在heap4中有类似如下定义,

//定义错误信息输出函数
#define vAssertCalled(charFile, intLine) AflDebugError("Error:%s,%d\r\n",charFile, intLine)
//利用 __FILE__,__LINE__ 预定义宏
#define configASSERT(x) if((x)==0) vAssertCalled(__FILE__,__LINE__)

早期在打印调试单元中定义的断言,不去中断程序执行,

//assert 自定义断言
#define AFLPRINTF_ASSERT(bAssert)  \
    if (!(bAssert))                \
        { \
            printf("AFLPRINTF_ASSERT %s,%d:", FUNCNAME, __LINE__); \
            printf("\r\n");   \
            fflush(stdout);   \
        }  \

其他的定义可参考 《异常处理/LINEFILE 宏在调试和异常处理中的高级使用》 等文章。

小结

在百科有提到 assert.h 常用于防御式编程。,防御式编程的主要思想是:子程序应该不因传入错误数据而被破坏,哪怕是由其他子程序产生的错误数据。这种思想是将可能出现的错误造成的影响控制在有限的范围内。这里也临时总结几个使用断言的几个原则,
(1)使用断言捕捉不应该发生的非法情况。不要混淆非法情况与错误情况之间的区别,后者是必然存在的并且是一定要作出处理的。
(2)使用断言对函数的参数进行确认。
(3)在编写函数时,要进行反复的考查,并且自问:"我打算做哪些假定?"一旦确定了的假定,就要使用断言对假定进行检查。
(4)一般教科书都鼓励程序员们进行防错性的程序设计,但要记住这种编程风格会隐瞒错误。当进行防错性编程时,如果"不可能发生"的事情的确发生了,则要使用断言进行报警。
ASSERT 是一个调试程序时经常使用的宏,在程序运行时它计算括号内的表达式,如果表达式为FALSE (0), 程序将报告错误,并终止执行。如果表达式不为0,则继续执行后面的语句。这个宏通常原来判断程序中是否出现了明显非法的数据,如果出现了终止程序以免导致严重后果,同时也便于查找错误。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1668390.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

华为交换机配置导出备份python脚本

一、脚本编写思路 &#xff08;一&#xff09;针对设备型号 主要针对华为&#xff08;Huawei&#xff09;和华三&#xff08;H3C&#xff09;交换机设备的配置备份 &#xff08;二&#xff09;导出前预处理 1.在配置导出前&#xff0c;自动打开crt软件或者MobaXterm软件&am…

IEEE 802.11标准

在IEEE 802.11标准中使用了扩频通信技术&#xff0c;主要作用是使得抗干扰性更强。 IEEE 802.11在MAC层采用了CSMA/CA协议。 IEEE 802.1x是一种基于端口认证协议。

【数据分析面试】41.如何分析处理Netflix流失用户?(业务分析)

题目 假设有一百万 Netflix 用户在过去六个月内没有登录到 Netflix。 你会如何确定原因&#xff1f;以及你会如何处理这些用户&#xff1f; Netflix 是一家总部位于美国的全球性流媒体娱乐服务公司&#xff0c;提供在线视频点播服务。通过其网站和移动应用&#xff0c;用户可…

内容检索(2024.05.12)

随着创作数量的增加&#xff0c;博客文章所涉及的内容越来越庞杂&#xff0c;为了更为方便地阅读&#xff0c;后续更新发布的文章将陆续在此汇总并附上原文链接&#xff0c;感兴趣的小伙伴们可持续关注文章发布动态&#xff01; 本期更新内容&#xff1a; 1. 信号仿真类话题-…

C++笔记(体系结构与内核分析)

1.OOP面向对象编程 vs. GP泛型编程 OOP将data和method放在一起&#xff0c;目的是通过封装、继承、多态提高软件的可维护性和可扩展性GP将data和method分开&#xff0c;可以将任何容器与任何算法结合使用&#xff0c;只要容器满足塞饭所需的迭代器类型 2.算法与仿函数的区别 …

Android性能:高版本Android关闭硬件加速GPU渲染滑动卡顿掉帧

Android性能&#xff1a;高版本Android关闭硬件加速GPU渲染滑动卡顿掉帧 如果在Androidmanifest.xml配置&#xff1a; <application android:hardwareAccelerated"false" > 或者某个特点View使用代码&#xff1a; myView.setLayerType(View.LAYER_TYPE_SOFT…

【Python-爬虫】

Python-爬虫 ■ 爬虫分类■ 1. 通用网络爬虫&#xff1a;&#xff08;搜索引擎使用&#xff0c;遵守robots协议&#xff09;■ robots协议&#xff08;君子协议&#xff09; ■ 2. 聚集网络爬虫&#xff1a;自己写的爬虫程序 ■ urllib.request&#xff08;要导入的模块&#x…

字典是如何实现的?Rehash 了解吗?

字典是 Redis 服务器中出现最为频繁的复合型数据结构。除了 hash 结构的数据会用到字典外&#xff0c;整个 Redis 数据库的所有 key 和 value 也组成了一个 全局字典&#xff0c;还有带过期时间的 key 也是一个字典。(存储在 RedisDb 数据结构中) 字典结构是什么样的呢&#xf…

PDF Squeezer for Mac,让PDF压缩更高效

还在为PDF文件过大而烦恼吗&#xff1f;试试PDF Squeezer for Mac吧&#xff01;它拥有强大的压缩功能&#xff0c;可以快速将PDF文件压缩至更小的体积&#xff0c;让你的文件传输更快捷。同时&#xff0c;它还支持多种压缩方式&#xff0c;满足你的不同需求。赶快下载体验吧&a…

【机器学习300问】88、什么是Batch Norm算法?

一、什么是Batch Norm&#xff1f; &#xff08;1&#xff09;Batch Norm的本质 神经网络中的Batch Normalization&#xff08;批量归一化&#xff0c;简称BatchNorm或BN&#xff09;是一种改进神经网络训练过程的规范化方法&#xff0c;BatchNorm的主要目的是加速神经网络的训…

每日两题 / 23. 合并 K 个升序链表 94. 二叉树的中序遍历(LeetCode热题100)

23. 合并 K 个升序链表 - 力扣&#xff08;LeetCode&#xff09; 若lists有k个元素&#xff0c;调用k - 1次&#xff08;两个有序链表的合并&#xff09;即可 /*** Definition for singly-linked list.* struct ListNode {* int val;* ListNode *next;* ListNod…

【Delphi】OpenCV 实战(一):OpenCV简介及开发环境配置

目录 一、OpenCV 功能模块 二、Delphi 中使用OpenCV 三、OpenCV 4.7 Delphi开发环境配置 1. 环境配置 2. OpenCV 中Demo程序的编译配置 3. 运行 Demo (OpenCV for Delphi) OpenCV 是世界上最大的计算机视觉库。 它是开源的,包含 2500 多种算法,由非营利…

八、VUE内置指令

一、初识VUE 二、再识VUE-MVVM 三、VUE数据代理 四、VUE事件处理 五、VUE计算属性 六、Vue监视属性 七、VUE过滤器 七、VUE内置指令 九、VUE组件 v-text 向其所在的节点中渲染文本内容。 (纯文本渲染)与插值语法的区别&#xff1a;v-text会替换掉节点中的内容&#xff0c;{{x…

技术爱好者必看:如何用AI问答API彻底改变用户体验!

AI 问答 API 对接说明 我们知道&#xff0c;市面上一些问答 API 的对接还是相对没那么容易的&#xff0c;比如说 OpenAI 的 Chat Completions API&#xff0c;它有一个 messages 字段&#xff0c;如果要完成连续对话&#xff0c;需要我们把所有的上下文历史全部传递&#xff0…

猫头虎分享已解决Bug || **Vue.js脚手架安装失败** Error: unable to fetch template`

猫头虎分享已解决Bug &#x1f42f; || Vue.js脚手架安装失败 &#x1f6ab;Error: unable to fetch template 博主猫头虎的技术世界 &#x1f31f; 欢迎来到猫头虎的博客 — 探索技术的无限可能&#xff01; 专栏链接&#xff1a; &#x1f517; 精选专栏&#xff1a; 《面试题…

Maven多环境与SpringBoot多环境配置

1. Maven多环境配置与应用 1.1 多环境开发 我们平常都是在自己的开发环境进行开发&#xff0c; 当开发完成后&#xff0c;需要把开发的功能部署到测试环境供测试人员进行测试使用&#xff0c; 等测试人员测试通过后&#xff0c;我们会将项目部署到生成环境上线使用。 这个时…

Django——会话.CookieSession

Django——会话.Cookie&Session 一、Cookie 会话指的是浏览器与web服务器之间的通信。HTTP协议是无状态协议。web服务器无法知道用户上一次会话数据&#xff0c;用来维护用户在访问网站过程中的状态 &#xff0c; 会话控制使用 Cookie 和 Session 一起实现。 通常把 Ses…

maven远程仓库访问顺序

首先需要了解一下各个配置文件&#xff0c;主要分为三类&#xff1a; 全局配置文件(${maven.home}/conf/settings.xml)&#xff0c;maven安装路径下的/conf/settings.xml用户配置文件(%USER_HOME%/.m2/settings.xml)&#xff0c;windows用户文件夹下项目配置文件&#xff1a;p…

【JVM基础篇】类加载器分类介绍

文章目录 类加载器什么是类加载器类加载器的作用是什么应用场景类加载器的分类启动类加载器用户扩展基础jar包 扩展类加载器和应用程序类加载器扩展类加载器通过扩展类加载器去加载用户jar包&#xff1a; 应用程序加载器 Arthas中类加载器相关功能 文章说明 类加载器 什么是类…

【Gitlab远程访问本地仓库】Gitlab如何安装配置并结合内网穿透实现远程访问本地仓库进行管理

文章目录 前言1. 下载Gitlab2. 安装Gitlab3. 启动Gitlab4. 安装cpolar5. 创建隧道配置访问地址6. 固定GitLab访问地址6.1 保留二级子域名6.2 配置二级子域名 7. 测试访问二级子域名 前言 GitLab 是一个用于仓库管理系统的开源项目&#xff0c;使用Git作为代码管理工具&#xf…