C++--内存管理和模板

news2025/1/14 0:50:06

       前言:在C++中,内存管理是一项关键的任务,因为程序需要为变量、对象和数据结构等动态分配内存。有效的内存管理是确保程序在运行期间高效使用系统资源的重要一环。此外,C++还引入了模板的概念,以提供一种通用的编程方式。模板允许编写可以处理多种类型的通用代码,并在编译时根据实际使用的类型来生成特定类型的代码。函数模板允许定义通用的函数,而类模板允许定义通用的类。模板在泛型编程中发挥了关键作用,它使得代码可以更灵活地处理不同类型的数据,提高了代码的重用性和可扩展性。那我们马上就开始了,

目录

1.C/C++内存分布

2.C++内存管理方式

malloc/calloc/realloc的异同

new/delete的类型匹配使用-针对自定义析构函数类

operator new与operator delete函数

new/delete实现原理

定位new表达式显示调用构造函数

3.常见的面试知识

3.1malloc/free和new/delete的区别

3.2 内存泄漏

什么是内存泄漏,内存泄漏的危害

堆内存泄漏(Heap leak)和系统资源泄漏

4.模板

4.1 函数模板

4.1.1 函数模板的实例化

4.1.2模板参数的匹配原则

4.2 类模板


1.C/C++内存分布

我们来看下面几个变量在内存中的存储位置

int globalVar = 1;
static int staticGlobalVar = 1;
void Test()
{
	static int staticVar = 1;
	int localVar = 1;
	int num1[10] = { 1, 2, 3, 4 };
	char char2[] = "abcd";
	const char* pChar3 = "abcd";
	int* ptr1 = (int*)malloc(sizeof(int) * 4);
	int* ptr2 = (int*)calloc(4, sizeof(int));
	int* ptr3 = (int*)realloc(ptr2, sizeof(int) * 4);
	free(ptr1);
	free(ptr3);
}

1. 栈又叫堆栈--非静态局部变量/函数参数/返回值等等,栈是向下增长的。

2. 内存映射段是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口创建共享共享内存,做进程间通信。

3. 堆用于程序运行时动态内存分配,堆是可以上增长的。

4. 数据段--存储全局数据和静态数据。

5. 代码段--可执行的代码/只读常量。

2.C++内存管理方式

      C语言中动态内存管理方式:malloc/calloc/realloc/free,C语言内存管理方式在C++中可以继续使用,但有些地方就无能为力,而且使用起来比较麻烦,因此C++又提出了自己的内存管理方式:通过new和delete操作符进行动态内存管理

malloc/calloc/realloc的异同

malloccallocrealloc是C语言中用于动态内存分配的函数。它们的异同如下:

  1. malloc

    • 函数原型:void* malloc(size_t size)
    • 功能:用于在堆中分配指定大小的内存块。
    • 参数:size表示要分配的字节数。
    • 返回值:返回一个指向分配内存的指针(void*)。如果分配失败,则返回NULL
    • 注意事项:malloc分配的内存块的内容是未初始化的,可能包含任意的值。
  2. calloc

    • 函数原型:void* calloc(size_t num, size_t size)
    • 功能:用于在堆中分配指定数量和大小的连续内存块,并将其初始化为零。
    • 参数:num表示要分配的元素数量,size表示每个元素的大小。
    • 返回值:返回一个指向分配内存的指针(void*)。如果分配失败,则返回NULL
    • 注意事项:calloc分配的内存块已被初始化为零,可以确保没有随机值。
  3. realloc

    • 函数原型:void* realloc(void* ptr, size_t size)
    • 功能:用于重新调整先前分配的内存块的大小。
    • 参数:ptr是指向之前分配的内存块的指针,size是希望调整为的新大小。
    • 返回值:返回一个指向重新调整大小后的内存块的指针(void*)。如果调整失败,则返回NULL
    • 注意事项:realloc可能会将内存块的内容复制到新的大小以适应新的内存需求。如果分配失败,原始内存块的内容将保持不变。

总结:

  • malloccalloc主要的区别在于内存初始化的行为,malloc分配的内存未初始化,而calloc分配的内存初始化为零。
  • realloc用于重新调整先前分配的内存块的大小,可以将内存大小减小或增大,并保留原始内存块的数据(如果可能的话)。
  • 所有这些函数都返回指向分配内存的指针,如果分配失败,则返回NULL
class A
{
public:
	A(int a = 0)
		: _a(a)
	{
		cout << "A():" << this << endl;
	}
	~A()
	{
		cout << "~A():" << this << endl;
	}
private:
	int _a;
};
int main()
{
	
	A* p1 = (A*)malloc(sizeof(A));
	A* p2 = new A(1);
	free(p1);
	delete p2;

	// 内置类型是几乎是一样的
	int* p3 = (int*)malloc(sizeof(int)); // C
	int* p4 = new int;
	free(p3);
	delete p4;

	A* p5 = (A*)malloc(sizeof(A) * 10);
	A* p6 = new A[10];
	free(p5);
	delete[] p6;//注意匹配使用
	return 0;
}

       new/delete 和 malloc/free最大区别是 new/delete对于【自定义类型】除了开空间还会调用构造函数和析构函数,在C++中经常使类和对象,所以,采用new/delete来代替malloc/free与C++更具有兼容性。

new/delete的类型匹配使用-针对自定义析构函数类

      我们知道,new只有一种形式使用,但是对于释放连续的空间并且其类的析构函数是自定义的,在我们使用delete时,就必须要使用delete[]的形式来释放空间,具体我们看下面的样例:

       相同的两个类,只是其中一个采用的是我们自定义的析构函数,一个采用系统默认生成的析构函数(系统默认生成的析构函数一般什么也不做,只是一个空函数),但是采用自定义析构函数的对象在申请连续的空间时,在释放时编译器会自动识别为其在申请的空间前添加一个字节,这个字节存储的是需要申请的个数,代表我们要对这个连续的空间调用多少次析构函数,这也属于编译器底层优化的一部分,如果不使用delete [ ],就会导致指针指向出现问题,析构函数的调用的次数等问题,所以一定要匹配使用。

operator new与operator delete函数

       new和delete是用户进行动态内存申请和释放的操作符,operator new 和operator delete是系统提供的全局函数,new在底层调用operator new全局函数来申请空间,delete在底层通过 operator delete全局函数来释放空间。

     operator new:该函数实际通过malloc来申请空间,当malloc申请空间成功时直接返回;申请空间失败,尝试执行空间不足应对措施,如果改应对措施用户设置了,则继续申请,否则抛异常。

//函数库内的函数(了解)
void* __CRTDECL operator new(size_t size) _THROW1(_STD bad_alloc)
{
	// try to allocate size bytes
	void* p;
	while ((p = malloc(size)) == 0)
		if (_callnewh(size) == 0)
		{
			// report no memory
			// 如果申请内存失败了,这里会抛出bad_alloc 类型异常
			static const std::bad_alloc nomem;
			_RAISE(nomem);
		}
	return (p);
}

operator delete: 该函数最终是通过free来释放空间的

void operator delete(void* pUserData)
{
	_CrtMemBlockHeader* pHead;
	RTCCALLBACK(_RTC_Free_hook, (pUserData, 0));
	if (pUserData == NULL)
		return;
	_mlock(_HEAP_LOCK);  /* block other threads */
	__TRY
		/* get a pointer to memory block header */
		pHead = pHdr(pUserData);
	/* verify block type */
	_ASSERTE(_BLOCK_TYPE_IS_VALID(pHead->nBlockUse));
	_free_dbg(pUserData, pHead->nBlockUse);
	__FINALLY
		_munlock(_HEAP_LOCK);  /* release other threads */
	__END_TRY_FINALLY
		return;
}
/*
free的实现
*/
#define   free(p)               _free_dbg(p, _NORMAL_BLOCK)

       通过上述两个全局函数的实现知道,operator new 实际也是通过malloc来申请空间,如果 malloc申请空间成功就直接返回,否则执行用户提供的空间不足应对措施,如果用户提供该措施 就继续申请,否则就抛异常。operator delete 最终是通过free来释放空间的

new/delete实现原理

       如果申请的是内置类型的空间,new和malloc,delete和free基本类似,不同的地方是:new/delete申请和释放的是单个元素的空间,new[]和delete[]申请的是连续空间,而且new在申请空间失败时会抛异常,malloc会返回NULL。

对于自定义类型:

new的原理

1. 调用operator new函数申请空间

2. 在申请的空间上执行构造函数,完成对象的构造

delete的原理

1. 在空间上执行析构函数,完成对象中资源的清理工作

2. 调用operator delete函数释放对象的空间

new T[N]的原理

1. 调用operator new[]函数,在operator new[]中实际调用operator new函数完成N个对象空间的申请

2. 在申请的空间上执行N次构造函数

delete[]的原理

1. 在释放的对象空间上执行N次析构函数,完成N个对象中资源的清理

2. 调用operator delete[]释放空间,实际在operator delete[]中调用operator delete来释 放空间

定位new表达式显示调用构造函数

定位new表达式是在已分配的原始内存空间中调用构造函数初始化一个对象

使用格式为:new (place_address) type或者new (place_address) type(参数表)

      定位new表达式在实际中一般是配合内存池使用。因为内存池分配出的内存没有初始化,所以如果是自定义类型的对象,需要使用new的定义表达式进行显示调构造函数进行初始化。

何为内存池?

      内存池是一种用于管理和分配内存的技术。它是在程序启动时预先分配一块连续的内存空间,并将其划分为多个固定大小的块或对象。这些块或对象可以被程序动态地分配和释放,以满足程序运行时的内存需求。

      内存池的主要目的是减少内存分配和释放的开销,提高程序的性能。相比于频繁地使用newdelete来进行内存分配和释放,内存池可以通过预先分配一块连续的内存空间来减少内存碎片和系统调用的次数,从而提高内存分配和释放的效率。

class A
{
public:
	A(int a = 0)
		: _a(a)
	{
		cout << "A():" << this << endl;
	}
	~A()
	{
		cout << "~A():" << this << endl;
	}
private:
	int _a;
};
// 定位new/replacement new
int main()
{
	// p1现在指向的只不过是与A对象相同大小的一段空间,还不能算是一个对象,因为构造函数没有执行
		A* p1 = (A*)malloc(sizeof(A));
	new(p1)A(1);  // 注意:如果A类的构造函数有参数时,此处需要传参
	p1->~A();
	free(p1);
	A* p2 = (A*)operator new(sizeof(A));
	new(p2)A(10);
	p2->~A();
	operator delete(p2);
	return 0;
}

3.常见的面试知识

3.1malloc/free和new/delete的区别

malloc/free和new/delete的共同点是:

都是从堆上申请空间,并且需要用户手动释放。

不同的地方是:

1. malloc和free是函数,new和delete是操作符

2. malloc申请的空间不会初始化,new可以初始化

3. malloc申请空间时,需要手动计算空间大小并传递,new只需在其后跟上空间的类型即可, 如果是多个对象,[ ]中指定对象个数即可

4. malloc的返回值为void*, 在使用时必须强转,new不需要,因为new后跟的是空间的类型

5. malloc申请空间失败时,返回的是NULL,因此使用时必须判空,new不需要,但是new需 要捕获异常

6. 申请自定义类型对象时,malloc/free只会开辟空间,不会调用构造函数与析构函数,而new 在申请空间后会调用构造函数完成对象的初始化,delete在释放空间前会调用析构函数完成 空间中资源的清理

3.2 内存泄漏

什么是内存泄漏,内存泄漏的危害

       内存泄漏:内存泄漏指因为疏忽或错误造成程序未能释放已经不再使用的内存的情况。内存泄漏并不是指内存在物理上的消失,而是应用程序分配某段内存后,因为设计错误,失去了对 该段内存的控制,因而造成了内存的浪费。

       内存泄漏的危害:长期运行的程序出现内存泄漏,影响很大,如操作系统、后台服务等等,出现内存泄漏会导致响应越来越慢,最终卡死。

堆内存泄漏(Heap leak)和系统资源泄漏

       堆内存指的是程序执行中依据须要分配通过malloc / calloc / realloc / new等从堆中分配的一块内存,用完后必须通过调用相应的 free或者delete 删掉。假设程序的设计错误导致这部分内存没有被释放,那么以后这部分空间将无法再被使用,就会产生Heap Leak。

       系统资源泄漏 指程序使用系统分配的资源,比方套接字、文件描述符、管道等没有使用对应的函数释放掉,导致系统资源的浪费,严重可导致系统效能减少,系统执行不稳定。

4.模板

我们都知道函数重载可以实现同一个函数名实现不同的功能的特点,但是其也有一些不好的地方:

1. 重载的函数仅仅是类型不同,代码复用率比较低,只要有新类型出现时,就需要用户自己增加对应的函数

2. 代码的可维护性比较低,一个出错可能所有的重载均出错

那么,下面我们就来看一种更加通用的模子,可以让编译器根据不同的类型利用该模子来生成代码

4.1 函数模板

typename是用来定义模板参数关键字,也可以使用class

template<typename T>
void Swap(T& left, T& right)
{
	T temp = left;
	left = right;
	right = temp;
}

      经典的swap函数使我们比较熟悉的,相信大多数学校开C++的应该都拿这个题当过板子题,emm~~~,那咱就整点没见过的~~

4.1.1 函数模板的实例化

      用不同类型的参数使用函数模板时,称为函数模板的实例化。模板参数实例化分为:隐式实例化和显式实例化。

隐式实例化:让编译器根据实参推演模板参数的实际类型

template<class T>
T Add(const T& left, const T& right)
{
	return left + right;
}
int main()
{
	int a1 = 10, a2 = 20;
	double d1 = 10.0, d2 = 20.0;
	Add(a1, a2);
	Add(d1, d2);

	/*
	该语句不能通过编译,因为在编译期间,当编译器看到该实例化时,需要推演其实参类型
	通过实参a1将T推演为int,通过实参d1将T推演为double类型,但模板参数列表中只有一个T,
	编译器无法确定此处到底该将T确定为int 或者 double类型而报错
	注意:在模板中,编译器一般不会进行类型转换操作,因为一旦转化出问题,编译器就需要背黑锅
	Add(a1, d1);
	*/

	// 此时有两种处理方式:1. 用户自己来强制转化 2. 使用显式实例化
	Add(a1, (int)d1);
	return 0;
}

显示实例化:在函数名后的< >中指定模板参数的实际类型,对于一些编译器没办法推出类型的场景需要使用

int main(void)
{
	int a = 10;
	double b = 20.0;

	// 显式实例化
	Add<int>(a, b);
	return 0;
}

4.1.2模板参数的匹配原则

        函数模板是一个蓝图,它本身并不是函数,是编译器用使用方式产生特定具体类型函数的模具。所以其实模板就是将本来应该我们做的重复的事情交给了编译器。在编译器编译阶段,对于模板函数的使用,编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用。比如:当用double类型使用函数模板时,编译器通过对实参类型的推演,将T确定为double类型,然 后产生一份专门处理double类型的代码,对于字符类型也是如此。

1. 一个非模板函数可以和一个同名的函数模板同时存在,而且该函数模板还可以被实例化为这个非模板函数

// 专门处理int的加法函数
int Add(int left, int right)
{
	return left + right;
}
// 通用加法函数
template<class T>
T Add(T left, T right)
{
	return left + right;
}
void Test()
{
	Add(1, 2); // 与非模板函数匹配,编译器不需要特化
	Add<int>(1, 2); // 调用编译器特化的Add版本
}

2. 对于非模板函数和同名函数模板,如果其他条件都相同,在调动时会优先调用非模板函数而不会从该模 板产生出一个实例。如果模板可以产生一个具有更好匹配的函数, 那么将选择模板

// 专门处理int的加法函数
int Add(int left, int right)
{
	return left + right;
}
// 通用加法函数
template<class T1, class T2>
T1 Add(T1 left, T2 right)
{
	return left + right;
}
void Test()
{
	Add(1, 2); // 与非函数模板类型完全匹配,不需要函数模板实例化
	Add(1, 2.0); // 模板函数可以生成更加匹配的版本,编译器根据实参生成更加匹配的Add函数
}

3.模板函数不允许自动类型转换,但普通函数可以进行自动类型转换

4.2 类模板

类模板大致和函数模板相似,但是也有些许区别:

类模板中函数放在类外进行定义时,需要加模板参数列表,比如:

template <class T>
Vector<T>::~Vector()
{
    ...
}

类模板实例化与函数模板实例化不同,类模板实例化需要在类模板名字后跟< >,然后将实例化的类型放在<> 中即可,类模板名字不是真正的类,而实例化的结果才是真正的类

// Vector类名,Vector<int>才是类型
Vector<int> s1;
Vector<double> s2;

       

       请你相信,只要你还愿意为了自己努力,世界就不会吝啬给你惊喜。也请你相信,只要你不断努力着,那么,最差的结果也不过就是大器晚成。

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

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

相关文章

C++模板与泛型编程:条款41~48

"绝境之中才窥见 winner winner 无限的精彩" 条款41: 了解隐式接口和编译器多态 我们给出一组类定义和函数实现(无意义): class Widget { public:Widget();virtual ~Widget();virtual size_t size() const;virtual void normalize();void swap(Widget&…

19项第一之上,是63%的极致带宽降低

近日&#xff0c;2022 MSU世界视频编码器大赛成绩正式揭晓。报告显示&#xff0c;阿里媒体处理服务MPS&#xff08;Alibaba Media Processing Service&#xff09;s264及s265编码器共计斩获19项评测第一&#xff0c;相较大赛指定基准编码器&#xff08;AWS Elemental MediaConv…

【Java8特性】——函数式接口方法引用

一、函数式&#xff08;Functional&#xff09;接口 1. 概述 如果一个接口中&#xff0c;只声明了一个抽象方法&#xff0c; 则这个接口就称为函数式接口。 注解&#xff1a;FunctionalInterface 显式指明改接口是一个函数式接口。可以检验是否是一个函数式接口&#xff0c;同…

利用 Databend 助力 CDH 分析 | 大参林

作者&#xff1a; 黄志武 大参林医药集团股份有限公司&#xff0c;信息中心数据库组组长&#xff0c;13年数据库行业从业经历&#xff0c;Oracle OCM&#xff0c;关注Oracle、MySQL、Redis、MongoDB、Oceanbase、Tidb、Polardb-X、TDSQL、CDH、Clickhouse、Doris、Databend等多…

【ES6】—【必备知识】—对象的扩展

一、属性简洁表示法 ES5 写法 let name xiao let age 30 let obj {name: name,age: age } console.log(obj) // {name: xiao, age: 30}ES6 简洁写法 对象的属性名 和 属性值的变量名相同&#xff0c;可以简写成 一个属性名 let name xiao let age 30 let obj {name,age …

如果你还不知道电商(淘宝京东1688)API,就看这里!

随着电商的蓬勃发展&#xff0c;现在已经进入了全民电商的时代&#xff1b;从国内电商到跨境电商&#xff0c;可以说是百家争艳&#xff0c;同时&#xff0c;电商运营也变得更精细化&#xff0c;各种运营工具也相继涌现&#xff0c;为店铺业绩做保障&#xff0c;电商API就是这样…

【回眸】牛客网刷刷刷!(七)——通信协议之 网络通讯

目录 前言 1、TCP/IP分层模型 2、ARP缓存 3、TCP 协议之所以提供可靠传输&#xff0c;不怕丢包、乱序的主要的原因是 4、以太网数据链路层MII/GMII/RMII/RGMII四种常用接口 5、在以太网通信协议LWIP中&#xff0c;数据包管理机构采用数据结构pbuf 分类包括 6、关于以太网…

关于css 的选择器和 css变量

css 选择器 常用的选择器 1. 后代选择器&#xff1a;也就是我们常见的空格选择器&#xff0c;选择的对象为该元素下的所有子元素 。例如&#xff0c;选择所有 元素下的 元素 div p{font-size:14px}2. 子元素选择器 ‘>’ 选择某元素下的直接子元素。例如&#xff0c;选择所…

龙蜥白皮书精选:云原生混部资源隔离技术

文/云原生 SIG 01 技术方案简介 混部就是将不同类型的业务在同一台机器上混合部署起来&#xff0c;让它们共享机器上的 CPU、内存、IO 等资源&#xff0c;目的就是最大限度地提高资源利用率&#xff0c;从而降低采购和运营等成本。 混部通常是将不同优先级的任务混合在一起&a…

前端开发进阶:前端开发中如何高效渲染大数据量?

在日常工作中&#xff0c;有时会遇到一次性往页面中插入大量数据的场景&#xff0c;在数栈的离线开发&#xff08;以下简称离线&#xff09;产品中&#xff0c;就有类似的场景。本文将通过分享一个实际场景中的前端开发思路&#xff0c;介绍当遇到大量数据时&#xff0c;如何实…

【1782. 统计点对的数目】

来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 描述&#xff1a; 给你一个无向图&#xff0c;无向图由整数 n &#xff0c;表示图中节点的数目&#xff0c;和 edges 组成&#xff0c;其中 edges[i] [ui, vi] 表示 ui 和 vi 之间有一条无向边。同时给你一个代表查询…

display设为inline-block时引发的高度问题,大坑

今天在写小程序&#xff0c;点击让这个遮罩层显示&#xff0c;结果一直下移&#xff0c;莫名其妙。 解决方案&#xff1a; 在元素的CSS中添加&#xff1a;vertical-align: bottom;

minAreaRect 函数新版与旧版对比

minAreaRect 函数 cv2.minAreaRect (InputArray_points) 入参 points 是点的集合&#xff0c;如轮廓 返回值 RotatedRect,带角度的旋转矩形框,其值形如(center(x,y), (width, height), angle of rotation ) center(x,y), (width, height)分别是旋转矩形框中心的坐标和矩…

chrome浏览器账号密码输入框自动填充时背景色不变

处理前 使用延时的方式解决 .login-box input,password:-webkit-autofill .login-box input,password:-webkit-autofill:hover, .login-box input,password:-webkit-autofill:focus, .login-box input,password:-webkit-autofill:active {-webkit-transition-delay: 999999…

【HMS Core】在线语种检测返回结果错误

【关键字】 在线语种检测、机器学习 【问题描述】 集成在线语种检测服务&#xff0c;检测蒙古文之后&#xff0c;返回结果为中文 【问题分析】 1、在线语种服务目前不支持蒙古文&#xff0c;具体可见官网语种支持列表&#xff1a; 【ML Kit】语种检测支持的语言列表 2、目前…

【JAVA毕设|课设】基于SpringBoot+vue的在线考试系统-以计算机网络为例,可自行录入题库-附下载地址

基于SpringBootvue的在线考试系统-以计算机网络为例&#xff0c;可自行录入题库 一、项目简介二、开发环境三、项目技术四、功能结构五、运行截图六、功能实现七、数据库设计八、源码获取 一、项目简介 随着信息技术的迅猛发展&#xff0c;教育行业正面临着巨大的变革和挑战。…

RTP/RTCP的 NACK, PLI,SLI,FIR

1&#xff0c;概述 在网络环境不是太好的情况下&#xff0c;比如网络拥塞比较严重&#xff0c;丢包率可能比较高&#xff0c;简单实用NACK重传的机制&#xff0c;这样就会有大量的RTCP NACK报文&#xff0c;发送端收到相应的报文&#xff0c;又会发送大量指定的RTP报文&#x…

5G NR:PRACH时域资源

PRACH occasion时域位置由高层参数RACH-ConfigGeneric->prach-ConfigurationIndex指示&#xff0c;根据小区不同的频域和模式&#xff0c;38.211的第6.3.3节中给出了prach-ConfigurationIndex所对应的表格。 小区频段为FR1&#xff0c;FDD模式(paired频谱)/SUL&#xff0c;…

【前端从0开始】JavaSript——循环控制语句

循环控制语句 while语句 While 循环会在指定条件为真时循环执行代码块。 While循环&#xff0c;先进行条件判断&#xff0c;再执行循环体的代码 while (条件表达式){循环体 }注意&#xff1a;当前循环中&#xff0c;如果不满足条件&#xff0c;一次都不会执行 var i 1; whi…

利用tidevice+mysql+grafana实现ios性能测试

利用tidevicemysqlgrafana实现ios性能测试 1.什么是tidevice&#xff1f; tidevice是一个可以和ios设备进行通信的工具&#xff0c;提供以下功能&#xff1a; 截图获取手机信息ipa包的安装和卸载根据bundleID 启动和停止应用列出安装应用信息模拟Xcode运行XCTest&#xff0c…