在写windows C++代码的时候,从代码安全角度考虑,我们应该注意什么?

        在写windows C++代码的时候,从代码安全角度考虑,我们应该注意什么?分别是:输入验证、内存管理、错误处理、并发和线程安全、使用安全的API、避免使用不安全的函数、最小权限原则。

一、输入验证

1. 用户输入验证

#include <iostream>
#include <string>
#include <limits>

int main() {
    int age;
    while (true) {
        std::cout << "请输入您的年龄: ";
        if (std::cin >> age && age > 0 && age < 130) {
            break;
        } else {
            std::cout << "无效输入,请重新输入。\n";
            std::cin.clear(); // 清除失败的输入
            std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n'); // 忽略错误输入之后的字符
        }
    }

    std::cout << "输入的年龄是: " << age << std::endl;
    return 0;
}

2. 文件读取验证

        当从文件中读取数据时,您需要验证文件是否存在,是否可以被打开,以及读取的数据是否符合预期格式。例如,读取一个存储数字的文件:

#include <fstream>
#include <iostream>
#include <vector>

int main() {
    std::ifstream file("numbers.txt");
    std::vector<int> numbers;
    int number;

    if (!file) {
        std::cerr << "无法打开文件" << std::endl;
        return 1;
    }

    while (file >> number) {
        numbers.push_back(number);
    }

    if (!file.eof()) {
        std::cerr << "文件读取过程中发生错误" << std::endl;
        return 1;
    }

    std::cout << "读取到的数字有: ";
    for (int num : numbers) {
        std::cout << num << " ";
    }
    std::cout << std::endl;

    return 0;
}

3. 网络数据接收验证

        在处理网络通信时,接收到的数据应该经过严格验证。例如,如果您的程序是一个服务器,它可能需要验证从客户端接收到的消息格式是否正确:

// 假设这是一个简单的TCP服务器接收数据的代码片段
char buffer[1024];
int receivedBytes = recv(clientSocket, buffer, 1024, 0);
if (receivedBytes > 0) {
    // 验证接收到的数据
    if (isValidMessage(buffer, receivedBytes)) {
        // 处理消息
    } else {
        std::cerr << "接收到无效的消息格式" << std::endl;
    }
}

二、内存管理

1. 使用裸指针管理动态内存(传统方法)

int* ptr = new int(10); // 分配内存
// 使用 ptr...
delete ptr; // 释放内存
ptr = nullptr; // 防止悬挂指针

        在这个例子中,必须确保在 ptr 不再需要时释放分配的内存,并将指针设为 nullptr 以避免悬挂指针。这种方法容易出错,因为需要手动管理内存。

2. 使用智能指针管理动态内存(现代方法)

        C++11 引入的智能指针(如 std::unique_ptrstd::shared_ptr)自动管理内存,可以减少内存泄漏的风险。

#include <memory>

std::unique_ptr<int> ptr(new int(10));
// 使用 ptr...
// 不需要手动释放内存,当 ptr 离开作用域时,内存会自动被释放

3. 防止数组越界

4. 避免内存泄漏

void function() {
    int* ptr = new int(10); // 分配内存
    // 函数的其他操作...
    delete ptr; // 释放内存
}

        如果函数中有多个返回路径或可能抛出异常,使用智能指针来保证内存在任何情况下都能被正确释放。

三、错误处理机制

1. 使用异常处理机制

        C++ 提供了异常处理机制,允许在检测到错误时抛出异常,并在上层代码中捕获和处理这些异常。

#include <iostream>
#include <stdexcept>

double divide(double a, double b) {
    if (b == 0) {
        throw std::invalid_argument("除数不能为0");
    }
    return a / b;
}

int main() {
    try {
        double result = divide(10.0, 0.0);
        std::cout << "结果: " << result << std::endl;
    } catch (const std::invalid_argument& e) {
        std::cerr << "捕获到错误: " << e.what() << std::endl;
    }
    return 0;
}

2. 使用返回值进行错误指示

        在某些情况下,使用返回值来指示错误是更加合适的。这种方法常用于 C 风格的代码或者需要保持与旧代码的兼容性。

#include <iostream>

bool safeDivide(double a, double b, double& result) {
    if (b == 0) {
        return false; // 错误指示
    }
    result = a / b;
    return true; // 操作成功
}

int main() {
    double result;
    if (!safeDivide(10.0, 0.0, result)) {
        std::cerr << "错误:除数不能为0" << std::endl;
    } else {
        std::cout << "结果: " << result << std::endl;
    }
    return 0;
}

3. 使用错误码

四、最小权限原则

        最小权限原则(Principle of Least Privilege, PoLP)是一种安全设计原则,意在确保程序、进程或用户仅具有完成其任务所必需的最小权限集。在 Windows C++ 编程中,这通常涉及到操作系统资源的访问和管理。以下是一些应用最小权限原则的例子:

1. 运行权限限制

        当开发一个应用程序时,确保它以最低必要的权限运行。例如,如果您的程序仅需读取文件,不应请求或具有写入文件的权限。

// 以只读方式打开文件
std::ifstream file("example.txt");
if (!file) {
    std::cerr << "无法打开文件,权限不足或文件不存在" << std::endl;
    // 错误处理
}

2. 数据库访问

        当您的程序需要与数据库交互时,为数据库用户分配只能执行其需求的操作的最小权限。例如,如果程序只需要从数据库读取数据,那么数据库用户不应该具有写入、修改或删除数据的权限。

// 假设这是一个数据库连接代码片段
// 这个数据库用户只有读取数据的权限
const char* connectionString = "User=ReadOnlyUser;Password=example;...";

3. 网络服务权限

        如果您的程序是一个网络服务,确保它在一个受限的环境中运行,例如在低权限的用户账户下运行,避免在需要管理员权限的账户下运行。

4. 文件和目录权限

        当您的程序需要创建文件或目录时,设置合理的访问控制列表(ACL),以确保只有必要的用户或程序有权访问这些资源。

五、并发和线程安全

        在并发编程中,线程安全是一个核心考虑因素。线程安全的代码可以在多线程环境中安全地被多个线程同时执行,而不会导致数据损坏或不一致的状态。以下是一些并发和线程安全的例子:

1. 使用互斥锁

        互斥锁(mutex)是保护共享资源免受多个线程同时访问的一种方法。

#include <iostream>
#include <thread>
#include <mutex>

std::mutex mtx; // 全局互斥锁
int sharedData = 0; // 共享数据

void increment() {
    mtx.lock(); // 获取锁
    ++sharedData; // 修改共享数据
    mtx.unlock(); // 释放锁
}

int main() {
    std::thread t1(increment);
    std::thread t2(increment);

    t1.join();
    t2.join();

    std::cout << "共享数据的值: " << sharedData << std::endl;
    return 0;
}

2. 使用条件变量

        条件变量是一种允许线程等待特定条件的同步机制。

#include <iostream>
#include <thread>
#include <mutex>
#include <condition_variable>

std::mutex mtx;
std::condition_variable cv;
bool ready = false;

void printNumber(int num) {
    std::unique_lock<std::mutex> lck(mtx);
    while (!ready) {
        cv.wait(lck);
    }
    std::cout << "Number: " << num << std::endl;
}

void setReady() {
    {
        std::lock_guard<std::mutex> lck(mtx);
        ready = true;
    }
    cv.notify_all();
}

int main() {
    std::thread t1(printNumber, 1);
    std::thread t2(printNumber, 2);

    setReady();

    t1.join();
    t2.join();

    return 0;
}

        在这个例子中,两个线程等待 ready 变量变为 true。一旦 setReady 函数被调用,条件变量通知所有等待的线程继续执行。

3. 使用原子操作

        原子操作是不可分割的操作,可以在多线程环境中安全地执行,不需要额外的同步机制。

#include <iostream>
#include <thread>
#include <atomic>

std::atomic<int> count(0); // 原子变量

void increment() {
    for (int i = 0; i < 10000; ++i) {
        count++; // 原子操作
    }
}

int main() {
    std::thread t1(increment);
    std::thread t2(increment);

    t1.join();
    t2.join();

    std::cout << "计数: " << count << std::endl;
    return 0;
}

        在这个例子中,两个线程同时增加一个计数器。由于 count 是一个原子类型,每次增加操作都是线程安全的。

六、使用安全的函数(Windows api)

        这里只需注意需要使用安全的。

1. 字符串复制

char source[] = "Hello World";
char dest[11];
strcpy(dest, source); // 不安全,没有边界检查


char source[] = "Hello World";
char dest[11];
strncpy(dest, source, sizeof(dest) - 1); // 安全,指定最大复制长度
dest[sizeof(dest) - 1] = '\0'; // 确保字符串以空字符结尾

strcpy 的问题

  strcpy 函数用于将一个字符串复制到另一个字符串。它继续复制字符直到遇到源字符串的空终止字符('\0')。这个函数的问题在于它不考虑目标字符串的大小。如果源字符串长度超过了目标字符串的容量,strcpy 会继续写入,导致超出目标数组的边界,造成缓冲区溢出。这种溢出可能覆盖其他重要的内存数据,引起程序崩溃或安全漏洞。

        char source[] = "这个字符串很长,超过目标缓冲区的大小"; char dest[10]; // 只有10个字符的空间 strcpy(dest, source); // 危险:源字符串长度超过了dest的容量

        在这个例子中,dest 只能容纳9个字符和一个空终止符,但 source 的长度远远超过这个限制,导致 dest 的边界被溢出。

strncpy 的相对安全性

  strncpy 函数也是用于复制字符串,但它接受一个额外的参数来指定最大复制的字符数。这个函数在到达最大字符数或遇到源字符串的空终止符时停止复制。这有助于防止超出目标数组的边界,如果正确使用的话。

char source[] = "这个字符串很长,可能超过目标缓冲区的大小";
char dest[10];
strncpy(dest, source, sizeof(dest) - 1); // 复制最多9个字符
dest[sizeof(dest) - 1] = '\0'; // 确保有空终止符

strcpy 的问题

strcpy 函数用于将一个字符串复制到另一个字符串。它继续复制字符直到遇到源字符串的空终止字符('\0')。这个函数的问题在于它不考虑目标字符串的大小。如果源字符串长度超过了目标字符串的容量,strcpy 会继续写入,导致超出目标数组的边界,造成缓冲区溢出。这种溢出可能覆盖其他重要的内存数据,引起程序崩溃或安全漏洞。

strncpy 的相对安全性

strncpy 函数也是用于复制字符串,但它接受一个额外的参数来指定最大复制的字符数。这个函数在到达最大字符数或遇到源字符串的空终止符时停止复制。这有助于防止超出目标数组的边界,如果正确使用的话。

例子:使用 strncpy 防止缓冲区溢出

2. 字符串连接

char str[10] = "Hello";
strcat(str, " World"); // 不安全,可能导致缓冲区溢出


char str[15] = "Hello";
strncat(str, " World", sizeof(str) - strlen(str) - 1); // 安全,限制最大添加长度

3. 格式化字符串

char buffer[50];
sprintf(buffer, "Value: %d", 123); // 不安全,没有边界检查

char buffer[50];
snprintf(buffer, sizeof(buffer), "Value: %d", 123); // 安全,限制最大写入长度

        虽然 strncpystrcpy 安全,但它仍然需要小心使用。如果最大复制长度小于源字符串长度,strncpy 不会自动添加空终止符,可能导致目标字符串没有正确终止。因此,使用 strncpy 时,程序员需要确保目标字符串有足够的空间,并在需要时手动添加空终止符。

 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

带你精通chrony服务器

华子目录 为什么会出现Chrony&#xff1f;Linux的两个时钟NTP介绍Chrony介绍安装与配置安装Chrony配置文件分析实验1实验2chronyc命令查看时间服务器chronyc sources输出分析其他命令 常见时区 为什么会出现Chrony&#xff1f; 由于IT系统中&#xff0c;准确的计时非常重要&am…

迭代新品 | 第四代可燃气体监测仪,守护燃气管网安全快人一步

城市地下市政基础设施是城市有序运行的生命线&#xff0c;事关城市安全、健康运行和高质量发展。近年来&#xff0c;我国燃气事故多发、频发。2020、2021、2022 年分别发生燃气事故668、1140 起、802 起&#xff0c;造成92、106、66 人死亡&#xff0c;560、763、487 人受伤。尤…

「C++」map和set的使用介绍

&#x1f4bb;文章目录 &#x1f4c4;前言前置知识关联式容器键值对map和set的底层结构 setset的构造函数set 的修改操作set的使用 mapmap的函数map的使用 multiset 和 multimap&#x1f4d3;总结 &#x1f4c4;前言 stl容器分为两类&#xff0c;分别是序列容器和关联式容器&am…

Java 高等院校分析与推荐系统

1&#xff09;项目简介 随着我国高等教育的大众化&#xff0c;高校毕业生就业碰到了前所未有的压力&#xff0c;高校学生就业问题开始进入相关研究者们的视野。在高校学生供给忽然急剧增加的同时&#xff0c;我国高校大学生的就业机制也在发生着深刻的变化&#xff0c;作为就业…

操作系统:进程(一)

进程的基本概念 一般的解释是&#xff1a;进程是程序的一个执行实例&#xff0c;是正在执行的程序。我们写的程序编译后是一段二进制的文件。启动的时候加载到系统里面执行&#xff0c;就是以进程的形式执行。也就是说&#xff0c;我们编译后的可执行程序是一个静态的概念&…

C++ STL之string初始

我最近开了几个专栏&#xff0c;诚信互三&#xff01; > |||《算法专栏》&#xff1a;&#xff1a;刷题教程来自网站《代码随想录》。||| > |||《C专栏》&#xff1a;&#xff1a;记录我学习C的经历&#xff0c;看完你一定会有收获。||| > |||《Linux专栏》&#xff1…

JSP基本表单和Request对象使用例子

表单的jsp&#xff1b; <%page contentType"text/html;charsetgbk" pageEncoding"UTF-8"%> <!DOCTYPE html> <html><head><meta http-equiv"Content-Type" content"text/html; charsetUTF-8"><titl…

golang学习笔记——接口interfaces

文章目录 Go 语言接口例子空接口空接口的定义空接口的应用空接口作为函数的参数空接口作为map的值 类型断言接口值 类型断言例子001类型断言例子002 Go 语言接口 接口&#xff08;interface&#xff09;定义了一个对象的行为规范&#xff0c;只定义规范不实现&#xff0c;由具…

数据库大事记

数据库分类分类方法为&#xff1a;按数据模型分类、按业务类型分类、按部署方式分类、按存储介质分类。 按数据模型分类 按业务类型分类 按部署方式分类 按存储介质分类 喜欢点赞收藏&#xff0c;下期再见。

【Redux】Redux 基本使用

1. Redux 快速上手 Redux 是 React 最常用的集中状态管理工具&#xff0c;类似于Vue中的Pinia&#xff08;Vuex&#xff09;&#xff0c;可以独立于框架运行。 <button id"decrement">-</button> <span id"count">0</span> <…

多线程Thread(初阶一:认识线程)

目录 一、引用线程的原因 二、线程的概念 三、进程和线程的区别 四、多线程编程 一、引用线程的原因 多任务操作系统&#xff0c;希望系统能同时运行多个任务。所以会涉及到进程&#xff0c;需要对进程进行管理、调度等。 而单任务操作系统&#xff0c;就完全不涉及到进程…

YOLOv8-Seg改进策略:全新的聚焦式线性注意力模块Focused Linear Attention | ICCV2023

🚀🚀🚀本文改进:深入分析了现有线性注意力方法的缺陷,并提出了一个全新的聚焦的线性注意力模块(Focused Linear Attention),同时具有高效性和很强的模型表达能力。 🚀🚀🚀YOLOv8-seg创新专栏:http://t.csdnimg.cn/KLSdv 学姐带你学习YOLOv8,从入门到创新,…

深度学习系列53:mmdetection上手

1. 安装 使用openmim安装&#xff1a; pip install -U openmim mim install "mmengine>0.7.0" mim install "mmcv>2.0.0rc4"2. 测试案例 下载代码和模型&#xff1a; git clone https://github.com/open-mmlab/mmdetection.git mkdir ./checkpoi…

2023年【熔化焊接与热切割】考试试卷及熔化焊接与热切割试题及解析

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 熔化焊接与热切割考试试卷考前必练&#xff01;安全生产模拟考试一点通每个月更新熔化焊接与热切割试题及解析题目及答案&#xff01;多做几遍&#xff0c;其实通过熔化焊接与热切割模拟考试很简单。 1、【单选题】 对…

React+后端实现导出Excle表格的功能

最近在做一个基于Reactantd前端框架的Excel导出功能&#xff0c;我主要在后端做了处理&#xff0c;这个功能完成后&#xff0c;便总结成一篇技术分享文章&#xff0c;感兴趣的小伙伴可以参考该分享来做导出excle表格功能&#xff0c;以下步骤同样适用于vue框架&#xff0c;或者…

“轻松实现文件夹批量重命名:使用顺序编号批量改名“

你是否曾经遇到过需要批量重命名文件夹&#xff0c;却因为繁琐的手动操作而感到困扰&#xff1f;现在&#xff0c;我们为你带来了一款全新的工具——轻松实现文件夹批量重命名&#xff0c;使用顺序编号批量改名。这款工具将帮助你轻松解决文件夹重命名的问题&#xff0c;提高工…

SpringSecurity5|12.实现RememberMe 及 实现原理分析

security/day08 这个功能大家还熟悉么&#xff1f;我们在登录网站的时候&#xff0c;除了让你输入用户名和密码&#xff0c;还会有个勾选框&#xff1a; 记住我&#xff01;&#xff01;&#xff01;不是让大家记住我哈。 值得一提的是&#xff0c;Spring Security 也提供了这个…

2023年汉字小达人市级比赛在线模拟题更新:40分钟150题完整对标

今天是2023年11月19日&#xff0c;距离11月30日的汉字小达人市级比赛还有11天。许多孩子正在利用难得的周末抓紧练习和备赛。 结合一些孩子的反馈和需求&#xff0c;我把150题的在线模拟题做了更新&#xff0c;增加了前面的个人信息填写的部分&#xff0c;并且把整个试卷的完成…

python自动化标注工具+自定义目标P图替换+深度学习大模型(代码+教程+告别手动标注)

省流建议 本文针对以下需求&#xff1a; 想自动化标注一些目标不再想使用yolo想在目标检测/语意分割有所建树计算机视觉项目想玩一玩大模型了解自动化工具了解最前沿模型自定义目标P图替换… 确定好需求&#xff0c;那么我们发车&#xff01; 实现功能与结果 该模型将首先…

python——第九天

今日目标&#xff1a; 偏函数 递归 字符串对象 切片 常见排序和查找 偏函数&#xff1a; python中存在一种函数的特殊使用&#xff0c;称为偏函数 如果在调用某个函数时&#xff0c;恰好某一个或者&#xff0c;某一些参数都是一个固定值&#xff08;正好不是默认值&#xff09;…
最新文章