C语言调试宏全面总结(六大板块)

news2024/9/29 23:37:33
  1. C语言调试宏进阶篇:实用指南与案例解析
  2. C语言调试宏高级技巧与最佳实践
  3. C语言调试宏的深度探索与性能考量
  4. C语言调试宏在嵌入式系统中的应用与挑战
  5. C语言调试宏在多线程环境中的应用与策略
  6. C语言调试宏在并发编程中的高级应用

68dbe2928ad44a0ba6573e485b52a0cd.jpeg

C语言调试宏进阶篇:实用指南与案例解析

前言

在C语言编程中,调试是提高代码质量和可靠性的关键步骤。而宏作为一种强大的文本替换工具,可以在调试过程中发挥巨大作用。本文将带你深入了解C语言中的调试宏,通过一系列实用的案例,让你掌握如何高效地使用这些工具。

一、调试宏基础

1.1 宏的定义与使用

宏是C语言中的一种预处理指令,用于在编译前对代码进行文本替换。宏的定义通常如下:

#define 宏名 替换文本

使用宏时,只需在代码中写入宏名,编译器会在预处理阶段将其替换为对应的文本。

1.2 宏的优点

  • 提高代码可读性:通过宏定义,可以将复杂的表达式或常量简化为一个易于理解的名称。
  • 方便代码维护:当需要修改某个常量或表达式时,只需修改宏定义即可,无需逐个修改代码中的每个实例。
  • 增强代码可移植性:通过宏定义,可以方便地适应不同的平台或编译器。

二、常用调试宏

2.1 ASSERT宏

ASSERT宏是一种常用的调试工具,用于检查程序中的假设是否成立。当ASSERT宏中的条件为假时,程序将终止执行。

#define ASSERT(condition) \
    if (!(condition)) { \
        fprintf(stderr, "Assertion failed: %s, file %s, line %d\n", #condition, __FILE__, __LINE__); \
        exit(EXIT_FAILURE); \
    }

2.2 DEBUG宏

DEBUG宏用于在调试模式下输出调试信息,而在发布模式下不产生任何代码。

#ifdef DEBUG
#define DEBUG_PRINT(fmt, ...) fprintf(stderr, fmt, __VA_ARGS__)
#else
#define DEBUG_PRINT(fmt, ...)
#endif

2.3 UNUSED宏

UNUSED宏用于标记函数中的未使用参数,避免编译器产生警告。

#define UNUSED(x) (void)(x)

三、调试宏案例解析

3.1 ASSERT宏的使用

int divide(int a, int b) {
    ASSERT(b != 0);
    return a / b;
}

在这个案例中,我们使用ASSERT宏来检查除数是否为0。如果b为0,程序将终止执行,并输出错误信息。

3.2 DEBUG宏的使用

int main() {
    int a = 5, b = 10;
    DEBUG_PRINT("a = %d, b = %d\n", a, b);
    return 0;
}

在这个案例中,我们使用DEBUG宏在调试模式下输出变量a和b的值。在发布模式下,这些调试信息将不会产生任何代码。

3.3 UNUSED宏的使用

void func(int a, int b, int c) {
    UNUSED(c);
    // 使用a和b进行操作
}

在这个案例中,我们使用UNUSED宏来标记未使用的参数c,避免编译器产生警告。

四、总结

调试宏是C语言中一种强大的调试工具,可以提高代码的可读性、可维护性和可移植性。通过本文的介绍,相信你已经掌握了如何使用这些调试宏,并在实际编程中灵活运用。

最后,记住,调试是编程过程中不可或缺的一环,而调试宏正是你在这场战斗中的得力助手。善用它们,让你的代码更加健壮和可靠!

C语言调试宏高级技巧与最佳实践

前言

在上一篇博客中,我们探讨了C语言调试宏的基础知识和一些实用案例。本文将深入探讨更高级的调试宏技巧,并分享一些最佳实践,帮助你在C语言编程中更高效地使用调试宏。

一、条件编译的高级应用

1.1#ifdef、#ifndef和#endif的妙用

在上一篇中,我们提到了#ifdef#endif的用法。除此之外,#ifndef(如果未定义)也是非常有用的。这些指令可以帮助我们根据不同的编译条件包含或排除代码段。

#ifndef DEBUG
#define NDEBUG
#endif
#include <assert.h>

在这个例子中,如果DEBUG未定义,我们定义NDEBUG,这样assert.h中的断言就不会执行,从而在发布版本中去除调试代码。

1.2 #if、#elif和#else的高级用法

#if#elif#else可以用来根据特定的条件编译代码。这对于创建可适应不同平台和配置的代码非常有用。

#if defined(DEBUG) && defined(_WIN32)
// Windows平台下的调试代码
#elif defined(DEBUG) && defined(__linux__)
// Linux平台下的调试代码
#else
// 其他情况下的代码
#endif

二、调试宏的高级技巧

2.1 带参数的宏与字符串化操作

在上一篇中,我们介绍了ASSERT宏。通过字符串化操作(#),我们可以将宏参数转换为字符串,这在调试输出中非常有用。

#define ASSERT(condition) \
    if (!(condition)) { \
        fprintf(stderr, "Assertion failed: %s, file %s, line %d\n", #condition, __FILE__, __LINE__); \
        exit(EXIT_FAILURE); \
    }

2.2 带参数的宏与连接操作

通过连接操作(##),我们可以在宏定义中连接多个参数,创建更灵活的宏。

#define JOIN(a, b) a##b
#define TOKENPASTE(x, y) JOIN(x, y)
#define MY_MACRO_NAME TOKENPASTE(MY_PREFIX, MacroName)

在这个例子中,MY_MACRO_NAME将被展开为MY_PREFIXMacroName

三、调试宏的最佳实践

3.1 保持宏的可读性和简洁性

在定义宏时,应尽量保持其简洁和易于理解。复杂的宏可能会隐藏错误,并使代码难以维护。

3.2 使用宏时避免副作用

宏展开可能会产生副作用,特别是在带参数的宏中。确保宏的每个参数只被计算一次,以避免潜在的问题。

3.3 使用宏进行代码保护

在编写库或模块时,使用宏来保护代码免受多重包含的影响是一种良好的实践。

#ifndef MY_HEADER_H
#define MY_HEADER_H
// 头文件内容
#endif

四、总结

调试宏是C语言编程中的强大工具,但它们也需要谨慎使用。通过本文,我们探讨了条件编译的高级应用,带参数宏的技巧,以及调试宏的最佳实践。希望这些内容能帮助你更有效地使用调试宏,提高代码质量。

记住,调试宏的正确使用可以大大提高代码的可维护性和可调试性。继续探索和实验,让这些工具成为你编程技能的一部分!

C语言调试宏的深度探索与性能考量

前言

在前两篇博客中,我们介绍了C语言调试宏的基础知识、进阶技巧和最佳实践。本文将深入探讨调试宏的高级应用,特别是如何在不牺牲性能的前提下,利用调试宏进行高效的程序调试和性能分析。

一、调试宏与性能分析

1.1 性能分析宏的定义

性能分析是软件开发中的一个重要环节。通过定义特定的性能分析宏,我们可以在不修改代码逻辑的情况下,测量代码段的执行时间。

#define PROFILE(func) \
    do { \
        clock_t start = clock(); \
        func; \
        clock_t end = clock(); \
        double time_spent = (double)(end - start) / CLOCKS_PER_SEC; \
        printf("Function %s took %f seconds to execute\n", #func, time_spent); \
    } while (0)

1.2 使用性能分析宏

使用上述宏,可以轻松地分析函数或代码块的执行时间。

PROFILE(slow_function());

在这个例子中,slow_function()的执行时间将被测量并打印出来。

二、调试宏与代码覆盖率

2.1 代码覆盖率宏的概念

代码覆盖率是衡量测试完整性的一个重要指标。通过定义特定的宏,我们可以在测试过程中跟踪哪些代码被执行。

#define COVERED() \
    do { \
        static bool covered = false; \
        if (!covered) { \
            covered = true; \
            covered_lines++; \
        } \
    } while (0)

2.2 使用代码覆盖率宏

在代码的关键部分使用COVERED()宏,可以统计测试过程中覆盖的代码行数。

if (condition) {
    COVERED();
    // 关键代码
}

三、调试宏与内存泄漏检测

3.1 内存泄漏检测宏的实现

内存泄漏是C语言程序中常见的问题。通过定义特定的宏,可以在分配和释放内存时进行跟踪。

#define MALLOC(size) my_malloc(size, __FILE__, __LINE__)
#define FREE(ptr) my_free(ptr, __FILE__, __LINE__)

void *my_malloc(size_t size, const char *file, int line);
void my_free(void *ptr, const char *file, int line);

3.2 使用内存泄漏检测宏

使用自定义的MALLOCFREE宏,可以更容易地检测内存泄漏。

int *ptr = MALLOC(sizeof(int) * 10);
FREE(ptr);

四、调试宏与性能优化

4.1 性能优化宏的应用

在某些情况下,我们可能需要在调试和发布版本中执行不同的代码路径。通过条件编译,可以实现这一目标。

#ifdef NDEBUG
#define OPTIMIZED_CODE
#endif

// ...

#ifdef OPTIMIZED_CODE
// 性能优化代码
#else
// 调试代码
#endif

五、总结

调试宏在C语言编程中扮演着重要角色,不仅可以帮助我们找到和修复错误,还可以用于性能分析和代码覆盖率统计。通过本文的深度探索,我们学习了如何在不牺牲性能的前提下,利用调试宏进行高效的程序调试和性能分析。

记住,调试宏的正确使用可以大大提高代码的可维护性和可调试性。继续探索和实验,让这些工具成为你编程技能的一部分!

C语言调试宏在嵌入式系统中的应用与挑战

前言

在前几篇博客中,我们探讨了C语言调试宏在通用编程中的应用。然而,在嵌入式系统的开发中,调试宏的使用面临着独特的挑战和限制。本文将深入探讨嵌入式系统中调试宏的应用,以及如何克服这些挑战。

一、嵌入式系统中的调试限制

1.1 资源限制

嵌入式系统通常资源有限,如内存、处理器速度和存储空间。因此,在使用调试宏时,需要特别注意其对资源的影响。

1.2 实时性要求

许多嵌入式系统有严格的实时性要求。调试宏可能会引入额外的执行时间,影响系统的响应时间。

1.3 缺乏调试工具

与桌面或服务器环境不同,嵌入式系统可能缺乏成熟的调试工具,如调试器和性能分析器。

二、嵌入式系统中调试宏的应用

2.1 优化宏的使用

为了减少资源消耗,可以在编译时通过条件编译来控制宏的启用和禁用。

#ifdef NDEBUG
#define LOG(msg, ...)
#else
#define LOG(msg, ...) printf(msg, __VA_ARGS__)
#endif

2.2 使用轻量级宏

在嵌入式系统中,应避免使用复杂的宏。简单的宏可以减少代码大小和执行时间。

#define MIN(a, b) ((a) < (b) ? (a) : (b))

2.3 使用宏进行错误处理

在嵌入式系统中,错误处理非常重要。使用宏可以简化错误处理的代码。

#define CHECK(condition, error_action) \
    if (!(condition)) { \
        error_action; \
    }

三、嵌入式系统中的调试技巧

3.1 使用串口输出

许多嵌入式系统都有串口,可以通过串口输出调试信息。

#define SERIAL_LOG(msg, ...) serial_printf(msg, __VA_ARGS__)

3.2 使用LED指示

在资源非常有限的系统中,可以使用LED指示程序的状态或错误。

#define LED_ERROR led_on(LED_RED)

四、总结

在嵌入式系统的开发中,调试宏的应用需要特别注意资源消耗和实时性要求。通过优化宏的使用、使用轻量级宏和特定的调试技巧,可以在不牺牲系统性能的前提下,有效地使用调试宏。

记住,嵌入式系统的调试是一个挑战,但也是一个机会,通过创造性地使用调试宏,你可以开发出更可靠、更高效的嵌入式软件。继续探索和实验,让这些工具成为你编程技能的一部分!

C语言调试宏在多线程环境中的应用与策略

前言

在之前的博客中,我们探讨了C语言调试宏在通用编程和嵌入式系统中的应用。然而,当涉及到多线程程序时,调试宏的使用变得更加复杂。本文将讨论在多线程环境中使用调试宏的策略,以及如何确保调试信息的准确性和线程安全性。

一、多线程环境中的调试挑战

1.1 线程安全问题

在多线程程序中,多个线程可能同时访问和修改共享资源,包括调试信息。这可能导致数据竞争和不可预测的行为。

1.2 调试信息的同步

确保调试信息同步是非常重要的,以便能够准确地跟踪和分析线程的执行。

1.3 调试性能的影响

在多线程环境中,调试宏可能会对程序的性能产生更大的影响,因为它们可能会引入线程间的竞争和同步开销。

二、多线程环境中调试宏的应用策略

2.1 使用线程局部存储(TLS)

线程局部存储是一种机制,允许每个线程都有自己的变量实例。这可以用于存储线程特定的调试信息。

__thread char tls_buffer[256];
#define LOG_THREAD(msg, ...) sprintf(tls_buffer, msg, __VA_ARGS__); printf("%s", tls_buffer)

2.2 使用互斥锁保护共享资源

对于需要跨线程共享的调试资源,可以使用互斥锁来保护它们,防止数据竞争。

pthread_mutex_t log_mutex = PTHREAD_MUTEX_INITIALIZER;

#define LOG_SHARED(msg, ...) \
    pthread_mutex_lock(&log_mutex); \
    printf(msg, __VA_ARGS__); \
    pthread_mutex_unlock(&log_mutex)

2.3 使用条件编译优化调试代码

在多线程环境中,可以通过条件编译来优化调试代码,减少对性能的影响。

#ifdef NDEBUG
#define THREAD_SAFE_LOG(msg, ...)
#else
#define THREAD_SAFE_LOG(msg, ...) LOG_SHARED(msg, __VA_ARGS__)
#endif

三、多线程调试技巧

3.1 使用线程特定的日志文件

每个线程可以将调试信息写入自己的日志文件,这样可以更容易地分析和跟踪线程的执行。

__thread FILE *thread_log_file = NULL;
#define THREAD_LOG(msg, ...) \
    if (!thread_log_file) { \
        char filename[32]; \
        sprintf(filename, "thread_%d.log", pthread_self()); \
        thread_log_file = fopen(filename, "w"); \
    } \
    fprintf(thread_log_file, msg, __VA_ARGS__); \
    fflush(thread_log_file)

3.2 使用静态或局部变量减少锁的使用

尽可能使用静态或局部变量来存储调试信息,以减少对互斥锁的需求。

static int static_counter = 0;
#define INC_COUNTER() static_counter++

四、总结

在多线程环境中,调试宏的使用需要特别注意线程安全性和性能影响。通过使用线程局部存储、互斥锁和条件编译,可以有效地管理调试信息,并确保其准确性和线程安全性。

记住,多线程程序的调试是一个复杂的过程,但通过适当的策略和技巧,你可以更有效地使用调试宏,提高代码的质量和可靠性。继续探索和实验,让这些工具成为你编程技能的一部分!

C语言调试宏在并发编程中的高级应用

前言

在之前的博客中,我们探讨了C语言调试宏在多线程环境中的应用。然而,随着并发编程的复杂性增加,对调试宏的需求也变得更加高级。本文将深入探讨并发编程中调试宏的高级应用,包括如何处理复杂的同步问题、如何跟踪并发执行流程,以及如何优化调试性能。

一、并发编程中的调试挑战

1.1 同步问题

并发编程中的同步问题可能导致死锁、竞态条件和饥饿等问题,这些都使得调试变得复杂。

1.2 跟踪并发执行

在并发程序中,多个线程或进程的执行流程可能交织在一起,这使得跟踪程序的执行变得困难。

1.3 性能影响

调试宏可能会对并发程序的性能产生显著影响,特别是在高负载和高并发的场景下。

二、并发编程中调试宏的高级应用

2.1 使用原子操作和锁

在并发编程中,使用原子操作和锁来保护共享资源是至关重要的。调试宏可以帮助我们跟踪这些操作的正确性。

pthread_mutex_t lock;
#define LOCK() \
    if (pthread_mutex_lock(&lock) != 0) { \
        fprintf(stderr, "Failed to acquire lock at %s:%d\n", __FILE__, __LINE__); \
    }
#define UNLOCK() \
    if (pthread_mutex_unlock(&lock) != 0) { \
        fprintf(stderr, "Failed to release lock at %s:%d\n", __FILE__, __LINE__); \
    }

2.2 使用日志和跟踪工具

在并发编程中,使用日志和跟踪工具可以帮助我们理解程序的并发行为。

#define LOG_CONCURRENT(msg, ...) \
    printf("[%ld] " msg "\n", pthread_self(), __VA_ARGS__)

2.3 使用断言和静态分析工具

断言和静态分析工具可以帮助我们在编译时和运行时检测并发错误。

#define ASSERT_CONCURRENT(condition) \
    if (!(condition)) { \
        fprintf(stderr, "Concurrent assertion failed: %s at %s:%d\n", #condition, __FILE__, __LINE__); \
        exit(EXIT_FAILURE); \
    }

三、并发编程调试技巧

3.1 使用线程池和任务队列

通过使用线程池和任务队列,可以简化并发编程的调试,因为这样可以减少线程创建和销毁的开销。

3.2 使用调试器和性能分析器

现代调试器和性能分析器提供了许多用于并发编程的调试功能,如线程视图、锁视图和性能分析。

3.3 使用模拟和测试框架

使用模拟和测试框架可以帮助我们在开发和测试阶段发现并发错误。

四、总结

在并发编程中,调试宏的应用需要特别注意同步问题、执行流程的跟踪和性能影响。通过使用原子操作和锁、日志和跟踪工具、断言和静态分析工具,以及调试器和性能分析器,可以有效地管理并发程序的调试过程。

记住,并发编程的调试是一个复杂的过程,但通过适当的策略和技巧,你可以更有效地使用调试宏,提高代码的质量和可靠性。继续探索和实验,让这些工具成为你编程技能的一部分!

 

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

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

相关文章

嵌入式人工智能(44-基于树莓派4B的扩展板-LED按键数码管TM1638)

树莓派性能非常强悍&#xff0c;但是对于某些复杂的项目来说&#xff0c;会出现心有余而口不足的情况&#xff0c;为了解决这类问题&#xff0c;可以在树莓派上使用扩展板&#xff0c;我们介绍几款常见的扩展板&#xff0c;不仅可以扩展到树莓派&#xff0c;其他单片机或嵌入式…

Vue3 列表自动滚动播放(表头固定、列表内容自动滚动播放)+ vue3-seamless-scroll - 附完整示例

vue3-seamless-scroll&#xff1a;Vue3.0 无缝滚动组件&#xff0c;支持Vite2.0&#xff0c;支持服务端打包 目前组件支持上下左右无缝滚动&#xff0c;单步滚动&#xff0c;并且支持复杂图标的无缝滚动&#xff0c;目前组件支持平台与Vue3.0支持平台一致。 目录 效果 一、介绍…

安装vscode -- linux

前言 相信很多人在刚开始使用linux时&#xff0c;不知道怎么安装vscode来辅助我们编程&#xff0c;那么我将在此记录我所用的安装vscode的方法。 安装方法 方法一&#xff1a;snap 第一步&#xff1a;检查软件更新状况 sudo apt update在终端输入上述命令&#xff0c;会提…

大模型学习笔记 - LLM 之RLHF人类对齐的简单总结

LLM - RLHF人类对齐的简单总结 LLM-人类对齐 1. RLHF(Reinforcement Learning from Human Feedback, RLHF),基于人类反馈的强化学习2 奖励模型训练3 强化学习训练 3.1 PPO介绍3.2 进阶的RLHF的介绍 3.2.1. 过程监督奖励模型3.2.2. 基于AI反馈的强化学习3.2.3. 非强化学习的对齐…

卷积神经网络 - 基本卷积函数的变体篇

序言 在深度学习和卷积神经网络&#xff08; CNN \text{CNN} CNN&#xff09;的广阔领域中&#xff0c;基本卷积函数是构建网络结构的基础&#xff0c;它们通过滑动窗口的方式对输入数据进行特征提取。然而&#xff0c;随着应用场景和数据复杂性的增加&#xff0c;单一的卷积方…

苹果Vision Pro生态发展:现状、挑战与未来展望

苹果公司以其创新技术和强大的生态系统闻名于世。在最近的财报会议上,CEO蒂姆库克分享了Vision Pro平台的最新进展,引发了业界的广泛关注。本文将深入探讨Vision Pro生态的现状、面临的挑战以及与其他XR平台的对比分析。 一、Vision Pro生态现状 据库克介绍,Vision Pro平台…

爬1688商品---(测试版)

半成品. from DrissionPage import ChromiumPage import time from selenium import webdriver urlhttps://p4psearch.1688.com/hamlet.html?scene6&cositebaidujj_pz&locationre&trackid885662561117990122602pageChromiumPage()page.get(url)def key_wof():inde…

C++ QT开发 学习笔记(3)

C QT开发 学习笔记(3) - WPS项目 标准对话框 对话框类说明静态函数函数说明QFileDialog文件对话框getOpenFileName()选择打开一个文件getOpenFileNames()选择打开多个文件getSaveFileName()选择保存一个文件getExistingDirectory()选择一个己有的目录getOpenFileUrl()选择打幵…

荒原之梦考研:考研二战会很难吗?

考研二战是不是很难&#xff0c;其实很大程度上取决于我们自己&#xff0c;我们能否认清自己的优势&#xff0c;能否指定和执行合理的计划&#xff0c;有没有强大的心理支撑等&#xff0c;都是决定考研二战能否成功&#xff0c;或者能否比较轻松的成功的关键。 在本文中&#…

HCIP重修总笔记(中)

第八节 BGP基础 一、BGP产生背景 BGPBorder Gateway Protocol&#xff0c;边界网关协议)是一种用于自治系统间的动态路出协议&#xff0c;是一种外部网关协议。 自治系统AS:一组同一个管理机构进行管理&#xff0c;对外呈现统一选路策略的路由器的集合。 自治系统编号: …

浅谈基础的图算法——强联通分量算法(c++)

文章目录 强联通分量SCC概念例子有向图的DFS树代码例题讲解[POI2008] BLO-Blockade题面翻译题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 思路AC代码 【模板】割点&#xff08;割顶&#xff09;题目背景题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 提示…

数据结构实验报告-顺序表

桂 林 理 工 大 学 实 验 报 告 一、实验名称 实验1 顺序表 二、实验内容&#xff1a; 1.将书中介绍的顺序表的基本算法(如初始化、求长度、插人、删除、输出等)汇总在一起&#xff0c;用一个完整的程序实现顺序表的基本运算&#xff0c;并且编写顺序表的判空、判满等基…

最常见的AI大模型总结

前言&#xff1a;大模型可以根据其主要的应领域和功能&#xff0c;可以分类为“文生文”&#xff08;Text-to-Text&#xff09;、“文生图”&#xff08;Text-to-Image&#xff09;和“文生视频”&#xff08;Text-to-Video&#xff09;&#xff0c;都是基于自然语言处理&#…

JVM从入门到放弃

前言&#xff1a;关于JVM&#xff0c;其实有很多大厂开发了不同版本的JVM&#xff0c;比较知名的有&#xff1a;Sun HotSpot VM、BEA JRockit VM、IBM J9 VM、 Azul VM、 Apache Harmony、 Google Dalvik VM、 Microsoft JVM等等。现在使用的比较多的JDK8版本就是Sun HotSpot V…

「C++系列」指针

文章目录 一、指针的定义二、指针的基本概念1. 基本概念2. 案例代码示例 1&#xff1a;基本指针使用示例 2&#xff1a;指针与数组 3. 注意事项 三、指针的用途1. 指针的用途2. 案例代码案例1. 动态内存分配案例2. 函数参数&#xff08;通过指针修改值&#xff09;案例3. 数组和…

poky yocto(04):编译在vmware上运行的镜像

编译镜像 bitbake build-appliance-image 得到文件&#xff1a;build-appliance-image-qemux86-64.wic.vmdk 问题的关键来了&#xff0c;如何启动这个东西呢&#xff1f;由名字可知&#xff0c;这是一个vmware的硬盘文件&#xff0c;需要创建一个新的虚拟机加载它。 创建虚拟…

黑神话悟空游戏电脑配置要求 黑神话悟空Steam销量全球两连冠 黑神话悟空苹果笔记本电脑能玩吗 黑神话悟空是什么类型的游戏

相信不少游戏爱好者&#xff0c;近期被《黑神话&#xff1a;悟空》这款游戏刷屏了&#xff0c;备受期待的国产单机大作《黑神话:悟空》将于8月20日全球同步上线&#xff0c;登陆 PC (Steam / Epic / WeGame) 和 PS5 平台。凭借空前的关注度&#xff0c;该游戏有望成为国产游戏行…

sql注入漏洞复现

and 11 正常 and 12 报错 从这就已经说明是sql数字型注入了 上sqlmap验证一下 存在布尔盲注&#xff0c;时间盲注...... 我是在漏洞盒子上提交的&#xff0c;能不能通过看运气吧 下面这个漏洞已经是很久之前的了&#xff0c;现在已经是修复了&#xff0c;当时还是太年轻了...…

文献阅读:扩散波模型的物理信息神经网络

目录 摘要 Abstract 扩散波模型的物理信息神经网络 文献摘要 讨论|结论 理论知识 扩散波动方程&#xff08;曼宁方程&#xff09; 梯度停止&#xff08;Stop-gradient&#xff09;技术 时分PINN(TPINN) 新结构(fPINN) 实验设置 稳定流问题 等速和扩散问题 非线性速…

基于STM32的简易音频播放系统

目录 引言环境准备工作 硬件准备软件安装与配置系统设计 系统架构硬件连接代码实现 初始化代码音频播放代码应用场景 简易音频播放语音提示系统常见问题及解决方案 常见问题解决方案结论 1. 引言 音频播放系统在日常生活中有着广泛的应用&#xff0c;从简单的语音提示到复杂…