【C++】—— 异常处理

news2025/1/11 12:01:59

前言:

  • 本期,我将给大家讲解的是有关 异常处理 的相关知识!

目录

(一)C语言传统的处理错误的方式

(二)C++异常概念

(三)异常的使用

1、异常的抛出和捕获

1️⃣ 异常的抛出和匹配原则 

2️⃣ 在函数调用链中异常栈展开匹配原则

2、异常的重新抛出 

3、异常安全

4、异常规范

(四)C++标准库的异常体系

(五)异常的优缺点

总结


(一)C语言传统的处理错误的方式

首先,我们回顾一下C语言处理异常的相关方式:

  •  终止程序,如assert,缺陷:用户难以接受。如发生内存错误,除0错误时就会终止程序。

下面是使用宏进行异常处理的简单代码描述:

#include <stdio.h>
#include <assert.h>


int divide(int num1, int num2) 
{
    assert(num2 != 0);  // 断言num2不等于0

    return num1 / num2;
}

int main() 
{
    int result = divide(10, 0);
    // 如果编译时定义了NDEBUG宏,assert会被禁用,否则会触发异常并终止程序执行

    return 0;
}

【解释说明】

1、在上述代码中,函数用于实现两个整数的除法运算。通过在函数内使用宏,可以进行条件判断,确保除数不为零。如果为零,会触发异常,终止程序的执行。

2、在函数中,调用函数并传递除数为0的情况。如果编译时未定义宏(即未启用调试模式),会触发异常并终止程序执行。如果定义了宏,则会被禁用,不会触发异常,程序会继续执行后续的代码。

【输出展示】


  • 返回错误码,缺陷:需要程序员自己去查找对应的错误。如系统的很多库的接口函数都是通过把错误码放到errno中,表示错误

 下面是使用返回错误码的方式处理异常的示例代码:

#include <stdio.h>
int divide(int num1, int num2, int* res)
{
    if (num2 == 0) 
    {
        return -1; // 返回错误码 -1 表示除数为零的异常情况
    }

    *res = num1 / num2;
    return 0; // 返回 0 表示成功
}

int main() 
{
    int num1 = 10, num2 = 0, res;
    int num = divide(num1, num2, &res);

    if (num != 0)
    {
        printf("Error: Divide by zero\n");
        // 处理错误的逻辑
    }
    else 
    {
        printf("Result: %d\n", res);
        // 处理正常情况的逻辑
    }

    return 0;
}

 【解释说明】

  1. 在函数中,调用函数并传递除数为0的情况。函数返回的错误码被存储在变量中,通过判断的值,可以确定是正常情况还是异常情况。如果不等于0,则表示发生了异常,可以根据具体情况进行错误处理。
  2. 如果返回的错误码为0,表示除法运算成功,可以通过变量获得计算结果,执行相应的正常处理逻辑。

【输出展示】

实际中C语言基本都是使用返回错误码的方式处理错误,部分情况下使用终止程序处理非常严重的错误。


(二)C++异常概念
 

在C++中,异常(Exception)是一种用于处理程序运行时错误的机制。异常提供了一种跳出正常程序流程的方式,将错误信息传递到适当的处理程序进行处理。

以下是关于C++异常的一些概念:

  1. 异常抛出:当发生异常情况时,可以使用throw 语句将异常抛出。throw 语句通常包含一个异常对象,该对象可以是基本类型、类对象或指针;

  2. 异常捕获:异常被抛出后,程序可以使用try-catch语句块来捕获并处理异常。try 块包含可能发生异常的代码,而catch块用于捕获和处理异常;

  3. 异常处理程序catch 块是用于处理异常的代码块。在catch 块中,可以根据抛出的异常类型来执行相应的处理逻辑。可以有多个 catch 块,按照顺序逐个匹配异常类型并执行匹配的处理逻辑。

如果有一个块抛出一个异常,捕获异常的方法会使用 try 和 catch 关键字。try 块中放置可能抛
出异常的代码,try 块中的代码被称为保护代码。

  • 使用 try/catch 语句的语法如下所示:
     
try
{
    // 保护的标识代码
}catch( ExceptionName e1 )
{
    // catch 块
}catch( ExceptionName e2 )
{
    // catch 块
}catch( ExceptionName eN )
{
    // catch 块
}

【小结】

  1. 通过合理使用异常处理机制,在程序中对错误进行捕获和处理,可以增加程序的健壮性和可维护性;
  2. 合适的异常处理可以使代码更清晰、可读性更好,并且可以更好地处理异常情况,提高程序的容错能力。

(三)异常的使用

1、异常的抛出和捕获

在C++中,异常的抛出和匹配原则遵循以下几个基本原则:

1️⃣ 异常的抛出和匹配原则
 

  1. 异常抛出:

    • 当程序发生异常情况时,可以使用throw语句将异常抛出。
    • throw语句通常包含一个异常对象,该对象可以是基本类型、类对象或指针。
  2. 异常匹配:

    • 异常的匹配是指根据抛出的异常类型来选择处理该异常的catch块。
    • C++的异常处理机制会按照顺序匹配try块中catch块的类型,找到能处理该异常类型的catch块。
  3. 异常类型匹配和继承关系:

    • C++允许异常类型形成继承关系,即派生类的异常对象可以被基类的catch块捕获。
    • 如果异常类型存在继承关系,派生类的catch块应该放在基类的catch块之前;否则,派生类的catch块将无法执行。
  4. 最匹配的异常处理:

    • C++异常处理机制会选择最匹配的catch块来处理抛出的异常。
    • 最匹配的catch块是指能够处理抛出的异常类型或其基类类型的catch块,即异常类型匹配的最接近情况。
  5. 异常未匹配的处理:

    • 如果在try块中抛出了异常,但没有找到匹配的catch块处理该异常,异常将传递到更高层的调用栈。
    • 如果异常一直没有被匹配的catch块处理,最终导致程序终止执行,并可能输出异常信息。

【注意事项】

  1. 异常的抛出和匹配原则是按照顺序匹配catch块来选择处理异常,因此在catch块的顺序布置上要谨慎;
  2. 通常,应从具体的异常类型开始,然后再向基类类型进行匹配,以确保异常可以被正确处理并执行相应的异常处理逻辑。

2️⃣ 在函数调用链中异常栈展开匹配原则

在函数调用链中,异常栈展开匹配原则主要指定了如何匹配异常类型并选择正确的异常处理代码。当异常发生时,C++运行时系统会从当前执行的函数开始,逐级检查调用栈中的函数调用,以查找与抛出的异常类型匹配的catch块。

以下是异常栈展开和匹配的原则:

  • 检查当前函数的try块:

    • 如果当前函数包含try块,运行时系统将查找匹配的catch块。
  • 检查当前函数的catch

    • 如果当前函数包含与抛出的异常类型匹配的catch块,那么该catch块将被执行。
    • 如果找到多个匹配的catch块,将选择最接近的(最近的)catch块来处理异常。
  • 如果当前函数没有匹配的catch

    • 异常栈展开到上一级调用函数。
    • 重复步骤 1 和步骤 2,直到找到匹配的catch块或达到调用栈的最顶层。
  • 如果在整个调用栈中没有找到匹配的catch

    • 程序的执行将终止,并调用标准库函数terminate()来终止程序。

 关于异常栈展开和匹配的重要注意事项:

  • 异常匹配时按照栈展开的顺序进行,而不是抛出异常的顺序。
  • 派生类的异常对象可以被基类的catch块捕获,因此在派生类的catch块之前应放置基类的catch块。
  • 如果在某个函数中抛出的异常没有匹配的catch块处理,异常会一直沿着调用栈向上传递,直到找到匹配的catch块或终止程序。
  • 异常的栈展开会跨越函数和线程边界,因此在多线程程序中也适用这些匹配原则。

 

例如以下示例:

 

接下来通过代码来具体的理解:

double Division(int a, int b) 
{
    if (b == 0)
        throw "Division by zero condition!";
    else
        return ((double)a / (double)b);
}

void Func() 
{
    int len, time;
   cin >> len >> time;
   cout << Division(len, time) << endl;
}

int main() 
{
    try {
        Func();
    }
    catch (const char* errmsg) {
        cout << errmsg << endl;
    }
    catch (...) {
        cout << "unknown exception" << endl;
    }

    return 0;
}

输出展示:

 

 【解释说明】

  1. 通过异常处理机制来捕获并处理可能出现的除以零异常。当除以零发生时,会抛出一个字符串常量异常,并被catch (const char* errmsg)块捕获;
  2. 如果出现其他类型的异常,则被catch (...)块捕获并执行相应的处理逻辑。

 

2、异常的重新抛出
 

在C++中,异常的重新抛出允许在catch块内部对捕获的异常进行处理并将其重新抛出,以便让更高层的异常处理代码进一步处理该异常。可以使用throw语句将异常重新抛出。

以下是一个使用异常重新抛出的示例代码:

double Division(int a, int b)
{
	// 当b == 0时抛出异常
	if (b == 0)
	{
		throw "Division by zero condition!";
	}
	return (double)a / (double)b;
}
void Func()
{
	// 这里可以看到如果发生除0错误抛出异常,另外下面的array没有得到释放。
	// 所以这里捕获异常后并不处理异常,异常还是交给外面处理,这里捕获了再
	// 重新抛出去。
	int* array = new int[10];
	try {
		int len, time;
		cin >> len >> time;
		cout << Division(len, time) << endl;
	}
	catch (...)
	{
		cout << "delete []" << array << endl;
		delete[] array;
		throw;
	}
	// ...
	cout << "delete []" << array << endl;
	delete[] array;
}
int main()
{
	try
	{
		Func();
	}
	catch (const char* errmsg)
	{
		cout << errmsg << endl;
	}
	return 0;
}

【解释说明】

  1. Func函数中,如果除以零的异常发生,异常将被捕获,并输出删除array的信息。然后,array会被释放(使用delete[]),并使用throw语句重新抛出异常。这样,异常会传递到更高层级的代码中。
  2. main函数中,异常被最外层的catch块捕获,并输出异常信息。
  3. 通过在Func函数中重新抛出异常,并在捕获异常之前及其后释放array,可以确保在异常传递给更高层级之前,已经释放了相关的资源。

【小结】

通过异常的重新抛出,可以在异常被捕获的地方对异常进行适当处理,并在更高层级的代码中继续处理相同的异常或进行其他操作。这种机制提供了灵活性和错误的向上传递。 


3、异常安全

  • 构造函数完成对象的构造和初始化最好不要在构造函数中抛出异常,否则可能导致对象不完整或没有完全初始化
  • 析构函数主要完成资源的清理,最好不要在析构函数内抛出异常,否则可能导致资源泄漏(内存泄漏、句柄未关闭等)
  • C++中异常经常会导致资源泄漏的问题,比如在new和delete中抛出了异常,导致内存泄漏,在lock和unlock之间抛出了异常导致死锁,C++经常使用RAII来解决以上问题,关于RAII我们智能指针这节进行讲解

4、异常规范

 在C++中,异常规范是一种在函数声明中指定函数可能抛出的异常的方式。异常规范可以作为函数的一部分,用于标识函数可能引发的异常类型。具体来说,异常规范指定了函数可抛出的异常类型列表。

在C++98\03 中,异常规范使用了throw()声明。例如:

void foo() throw(int, std::exception);

【解释说明】

  1. 上述代码表示函数foo可能抛出int类型和exception类型的异常;
  2. 如果函数抛出了未在异常规范中列出的其他异常类型,程序会调用unexpected函数,默认情况下会导致terminate被调用终止程序。

更多示例如下图所示: 

// 这里表示这个函数会抛出A/B/C/D中的某种类型的异常
void fun() throw(A,B,C,D);

// 这里表示这个函数只会抛出bad_alloc的异常
void* operator new (std::size_t size) throw (std::bad_alloc);

// 这里表示这个函数不会抛出异常
void* operator delete (std::size_t size, void* ptr) throw();

在C++11开始引入了更为灵活和安全的异常处理机制,即异常规范的替代方案:异常说明。异常说明使用noexcept关键字指定函数是否允许抛出异常。

使用noexcept关键字的函数可以被称为“noexcept函数”或“不抛异常函数”。它们在以下方面有一些重要的用途和优点:

  1. 优化性能:编译器可以基于对noexcept的显式承诺做出一些优化;
  2. 异常传播:有助于避免异常传播到不应该处理异常的上下文中;

下面是一些使用noexcept的示例:

void myFunction() noexcept {
  // 函数体,不会抛出异常
}

void anotherFunction() {
  // 函数体,可能会抛出异常
}

void myFunction2() noexcept(true) {
  // 与上面的 myFunction 等效,不会抛出异常
}

void myFunction3() noexcept(false) {
  // 与 anotherFunction 等效,可能会抛出异常
}

//不会抛出异常
thread (thread&& x) noexcept;

【注意事项】

  1. 在C++11中,noexcept关键字可以作为函数类型的一部分,标志着函数是否抛出异常;
  2. 在C++17之后,noexcept支持函数表达式,以动态地决定是否抛出异常。这使得异常规范在一些特定的情况下更加灵活和动态。

(四)C++标准库的异常体系

C++ 提供了一系列标准的异常,定义在 中,我们可以在程序中使用这些标准的异常。它们是以父
子类层次结构组织起来的,如下所示:
 

 

 

说明:实际中我们可以可以去继承exception类实现自己的异常类。但是实际中很多公司像上面一样自己定义一套异常继承体系。因为C++标准库设计的不够好用。
 

int main()
{
	try {
		vector<int> v(10, 5);
		// 这里如果系统内存不够也会抛异常
		v.reserve(1000000000);
		// 这里越界会抛异常
		v.at(10) = 100;
	}
	catch (const exception& e) // 这里捕获父类对象就可以
	{
		cout << e.what() << endl;
	}
	catch (...)
	{
		cout << "Unkown Exception" << endl;
	}
	return 0;
}

(五)异常的优缺点

C++异常的优点:

  • 1. 异常对象定义好了,相比错误码的方式可以清晰准确的展示出错误的各种信息,甚至可以包含堆栈调用的信息,这样可以帮助更好的定位程序的bug。
  • 2. 返回错误码的传统方式有个很大的问题就是,在函数调用链中,深层的函数返回了错误,那么我们得层层返回错误,最外层才能拿到错误,具体看下面的详细解释。
// 1.下面这段伪代码我们可以看到ConnnectSql中出错了,先返回给ServerStart,
ServerStart再返回给main函数,main函数再针对问题处理具体的错误。
// 2.如果是异常体系,不管是ConnnectSql还是ServerStart及调用函数出错,都不用检查,因
为抛出的异常异常会直接跳到main函数中catch捕获的地方,main函数直接处理错误。
int ConnnectSql()
{
	// 用户名密码错误
	if (...)
		return 1;
	// 权限不足
	if (...)
		return 2;
}
int ServerStart() {
	if (int ret = ConnnectSql() < 0)
		return ret;
	int fd = socket()
		if(fd < 0)
		return errno;
}
int main()
{
	if (ServerStart() < 0)
		...
		return 0;
}
  • 3. 很多的第三方库都包含异常,比如boost、gtest、gmock等等常用的库,那么我们使用它们也需要使用异常。
  • 4. 部分函数使用异常更好处理,比如构造函数没有返回值,不方便使用错误码方式处理。比如T& operator这样的函数,如果pos越界了只能使用异常或者终止程序处理,没办法通过返回值表示错误。

C++异常的缺点:

  • 1. 异常会导致程序的执行流乱跳,并且非常的混乱,并且是运行时出错抛异常就会乱跳。这会导致我们跟踪调试时以及分析程序时,比较困难。
  • 2. 异常会有一些性能的开销。当然在现代硬件速度很快的情况下,这个影响基本忽略不计。
  • 3. C++没有垃圾回收机制,资源需要自己管理。有了异常非常容易导致内存泄漏、死锁等异常安全问题。这个需要使用RAII来处理资源的管理问题。学习成本较高。
  • 4. C++标准库的异常体系定义得不好,导致大家各自定义各自的异常体系,非常的混乱。
  • 5. 异常尽量规范使用,否则后果不堪设想,随意抛异常,外层捕获的用户苦不堪言。所以异常规范有两点:一、抛出异常类型都继承自一个基类。二、函数是否抛异常、抛什么异常,都使用 func() throw();的方式规范化。

总结

以上便是关于 c++11 有关异常的全部知识。接下来,简单的回顾下本文!!!

  1. 异常总体而言,利大于弊,所以工程中我们还是鼓励使用异常的;
  2. 另外OO的语言基本都是用异常处理错误,这也可以看出这是大势所趋。
     

到此,关于本篇便到此为止了。感谢大家的观看与支持!!!

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

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

相关文章

Windows Qt 5.12.10下载与安装

Qt 入门实战教程&#xff08;目录&#xff09; C自学精简实践教程 目录(必读) 1 Qt5.12.10下载 qt-opensource-windows-x86-5.12.10.exe 官方离线安装包 Download Source Package Offline Installers | Qt 下载巨慢&#xff08;也可能很快&#xff09; 只能下载到最新的&…

C语言——类型转换

数据有不同的类型&#xff0c;不同类型数据之间进行混合运算时涉及到类型的转换问题。 转换的方法有两种&#xff1a; 自动转换(隐式转换)&#xff1a;遵循一定的规则&#xff0c;由编译系统自动完成强制类型转换&#xff1a;把表达式的运算结果强制转换成所需的数据类型 语法格…

自动化测试(三):接口自动化pytest测试框架

文章目录 1. 接口自动化的实现2. 知识要点及实践2.1 requests.post传递的参数本质2.2 pytest单元测试框架2.2.1 pytest框架简介2.2.2 pytest装饰器2.2.3 断言、allure测试报告2.2.4 接口关联、封装改进YAML动态传参&#xff08;热加载&#xff09; 2.3 pytest接口封装&#xff…

20 MySQL(下)

文章目录 视图视图是什么定义视图查看视图删除视图视图的作用 事务事务的使用 索引查询索引创建索引删除索引聚集索引和非聚集索引影响 账户管理&#xff08;了解非DBA&#xff09;授予权限 与 账户的相关操作 MySQL的主从配置 视图 视图是什么 通俗的讲&#xff0c;视图就是…

编绎和优化,脚本代码小米加步枪赶超英法美

编程达人&#xff1a;冰冻牡蛎 测试&#xff0c;总结》》 今有空&#xff0c;继续看了一下竹笋大师几天前提出的“使用for循环查找10亿内可被7整除的数的个数”的题目&#xff08;相关文件&#xff1a;群文件 10亿以内多少个数字可以整除7.7z &#xff09; 1. 论输出的exe大小…

Pytorch-以数字识别更好地入门深度学习

目录 一、数据介绍 二、下载数据 三、可视化数据 四、模型构建 五、模型训练 六、模型预测 一、数据介绍 MNIST数据集是深度学习入门的经典案例&#xff0c;因为它具有以下优点&#xff1a; 1. 数据量小&#xff0c;计算速度快。MNIST数据集包含60000个训练样本和1000…

Java实现根据商品ID获取1688商品详情跨境属性数据,1688商品重量数据接口,1688API接口封装方法

要通过1688的API获取商品详情跨境属性数据&#xff0c;您可以使用1688开放平台提供的接口来实现。以下是一种使用Java编程语言实现的示例&#xff0c;展示如何通过1688开放平台API获取商品详情属性数据接口&#xff1a; 首先&#xff0c;确保您已注册成为1688开放平台的开发者…

网工内推 | IT网工,华为、华三认证优先,15k*13薪

01 广东善能科技发展股份有限公司 招聘岗位&#xff1a;IT网络工程师 职责描述&#xff1a; 1、负责公司项目售后技术支持工作&#xff1b; 2、负责项目交付实施&#xff0c;配置调试、运维等&#xff1b; 3、参加合作厂商产品技术知识培训&#xff1b; 4、参加合作厂商工程师…

运维Shell脚本小试牛刀(二)

运维Shell脚本小试牛刀(一) 运维Shell脚本小试牛刀(二) 一: if---else.....fi 条件判断演示 [rootwww shelldic]# cat checkpass.sh #!/bin/bash - # # # # FILE: checkpass.sh # USAGE: ./checkpass.sh # DESCRI…

贝叶斯人工智能大脑与 ChatGPT

文章目录 一、前言二、主要内容 &#x1f349; CSDN 叶庭云&#xff1a;https://yetingyun.blog.csdn.net/ 一、前言 论文地址&#xff1a;https://arxiv.org/abs/2308.14732 这篇论文旨在研究 Chat Generative Pre-trained Transformer&#xff08;ChatGPT&#xff09;在贝叶斯…

【AI】数学基础——高数(积分部分)

高数&#xff08;函数&微分部分&#xff09; 文章目录 1.4 微积分1.4.1 基本思想1.4.2 定积分定义定义计算定积分定积分性质定理N-L公式泰勒公式麦克劳林公式 1.5 求极值1.5.1 无条件极值1.5.2 条件极值1.5.3 多条件极值1.5.4 凹函数与凸函数 1.4 微积分 用于求解速度、面积…

开始MySQL之路——MySQL 事务(详解分析)

MySQL 事务概述 MySQL 事务主要用于处理操作量大&#xff0c;复杂度高的数据。比如说&#xff0c;在人员管理系统中&#xff0c;你删除一个人员&#xff0c;你即需要删除人员的基本资料&#xff0c;也要删除和该人员相关的信息&#xff0c;如信箱&#xff0c;文章等等&#xf…

JS小球绕着椭圆形的轨迹旋转并且近大远小

在ivx中案例如下&#xff1a; VxEditor 效果如下&#xff0c;近大远小 主要代码如下&#xff1a; const centerX 360 / 2; // 椭圆中心的X坐标 const centerY 120 / 2; // 椭圆中心的Y坐标 const a 100; // 长半轴 const b 60; // 短半轴const elementsWithClassName d…

uniapp项目实战系列(1):导入数据库,启动后端服务,开启代码托管

目录 前言前期准备1.数据库的导入2.运行后端服务2.1数据库的后端配置2.2后端服务下载依赖&#xff0c;第三方库2.3启动后端服务 3.开启gitcode代码托管 ✨ 原创不易&#xff0c;还希望各位大佬支持一下&#xff01; &#x1f44d; 点赞&#xff0c;你的认可是我创作的动力&…

虚拟化技术:云计算发展的核心驱动力

文章目录 虚拟化技术的概念和作用虚拟化技术的优势虚拟化技术对未来发展的影响结论 &#x1f389;欢迎来到AIGC人工智能专栏~虚拟化技术&#xff1a;云计算发展的核心驱动力 ☆* o(≧▽≦)o *☆嗨~我是IT陈寒&#x1f379;✨博客主页&#xff1a;IT陈寒的博客&#x1f388;该系…

RabbitMQ---Spring AMQP

Spring AMQP 1. 简介 Spring有很多不同的项目&#xff0c;其中就有对AMQP的支持&#xff1a; Spring AMQP的页面&#xff1a;http://spring.io/projects/spring-amqp 注意这里一段描述&#xff1a; Spring-amqp是对AMQP协议的抽象实现&#xff0c;而spring-rabbit 是对协…

如何在有或没有WiF适配器的情况下把台式机接入WiFi

Wi-Fi在台式电脑中越来越普遍,但并不是所有的台式电脑都有。添加Wi-Fi,你就可以无线连接到互联网,并为其他设备托管Wi-Fi热点。 这是一个简单、廉价的过程。买一个合适的小适配器,你甚至可以随身携带,通过将一个小设备插入USB端口,可以快速将Wi-Fi添加到你遇到的任何桌面…

Particle Life粒子生命演化的MATLAB模拟

Particle Life粒子生命演化的MATLAB模拟 0 前言1 基本原理1.1 力影响-吸引排斥行为1.2 距离rmax影响 2 多种粒子相互作用2.1 双种粒子作用2.1 多种粒子作用 3 代码 惯例声明&#xff1a;本人没有相关的工程应用经验&#xff0c;只是纯粹对相关算法感兴趣才写此博客。所以如果有…

Android屏幕显示 android:screenOrientation configChanges 处理配置变更

显示相关 屏幕朝向 https://developer.android.com/reference/android/content/res/Configuration.html#orientation 具体区别如下&#xff1a; activity.getResources().getConfiguration().orientation获取的是当前设备的实际屏幕方向值&#xff0c;可以动态地根据设备的旋…

Windows10 系统安装教程

多虚不如少实。 一、 下载安装包 下载前景&#xff1a;网上下载的 windows10 系统一般都有捆绑软件&#xff0c;用户体验不爽&#xff0c;所以建议到 正规渠道下载 windows10 系统的不同版本。另外网上也有一些 windows10 系统的镜像文件 可以直接一键安装&#xff0c;…