在Visutal Studio 2022中完成D3D12初始化

在Visutal Studio 2022中完成DirectX设备初始化

  • 1 DirectX12
    • 1.1 DirectX 简介
    • 1.2 DirectX SDK安装
  • 2 D3D12初始化
    • 2.1 创建Windwos桌面项目
    • 2.2 修改符合模式
    • 2.3 下载d3dx12.h文件
    • 2.4 创建一个异常类D3DException,定义抛出异常实例的宏ThrowIfFailed
  • 3 D3D12的初始化步骤
    • 3.1 初始化前的准备
      • 3.1.1 增加头文件的引用:
      • 3.1.2 引用Direct3D库文件、IDXGI库文件
      • 3.1.5 增加InitDirect3D函数
      • 3.1.6 启用D3D12的调试层
    • 3.2 创建Direct3D设备
      • 3.2.1 增加IDXGIFactory和ID3D12Device两个全局的COM对象智能指针
      • 3.2.2 创建一个D3D12设备
      • 3.2.3 如果创建主显示适配器失败,使用WARP软件适配器
    • 3.3 创建围栏
      • 3.3.1 增加全局围栏和描述符变量
      • 3.3.2 创建D3D12围栏
      • 3.3.3 我们来了解下资源与描述符
    • 3.4 检测对4X MSAA 质量级别的支持
      • 3.4.1 增加全局资源数据格式和4X MASS质量级别变量
    • 3.5 检测支持的D3D最高版本
    • 3.6 创建命令队列和列表
      • 3.6.1 增加全局变量
      • 3.6.2 创建命令队列和命令列表
    • 3.7 描述并创建交换链
      • 3.7.1 创建全局变量
      • 3.7.2 描述并创建交换链
      • 3.7.3 处理CPU和GPU的同步
    • 3.8 创建描述符堆
      • 3.8.1 增加全局变量
      • 3.8.2 创建描述符堆
    • 3.9 创建渲染目标视图
      • 3.9.1 增加全局变量
      • 3.9.2 调整后台缓冲区的大小,并为它创建渲染目标视图 / 描述符
    • 3.9 创建深度/模板缓冲区及其视图
    • 3.10 设置视口和裁剪矩形
  • 4 完整代码
  • 错误:“&”要求左值
  • 错误:D3D12GetDebugInterface: This method requires the D3D12 SDK Layers for Windows 10, but they are not present on the system.。

1 DirectX12

1.1 DirectX 简介

DirectX 是 Windows 中的一组组件,允许游戏、软件直接与视频和音频硬件结合使用。 使用 DirectX的游戏可以更有效地使用内置于硬件的多媒体加速器功能,从而改善整体的多媒体体验。

1.2 DirectX SDK安装

  • 方法1,通过Windows 10 SDK安装,官网下载最新版本:Windows SDK
    在这里插入图片描述
  • 方法2,通过VS安装,做为一个开发人员,我更喜欢这种方式
    在这里插入图片描述
    安装完成后,我们可以了解下DirectX SDK头文件目录和库文件目录。
    头文件目录:
C:\Program Files (x86)\Windows Kits\10\Include\10.0.19041.0\shared
C:\Program Files (x86)\Windows Kits\10\Include\10.0.19041.0\um

库文件目录 :

C:\Program Files (x86)\Windows Kits\10\Lib\10.0.19041.0\um\x64

(注意:10.0.19041.0是我电脑上面的版本,大家的可能不一样)

2 D3D12初始化

有了开发环境,我们来完成D3D12初始化。

2.1 创建Windwos桌面项目

我们创建一个Windwos桌面项目,从头开始学习D3D11的初始化过程。
打开Visutal Studio 2022,选择菜单:文件->新建->项目,创建新项目。选择C++类型,输入 Windows桌面向导 进行筛选,在结果中选择“Windows桌面向导”,点击下一步。
在这里插入图片描述
配置项目名称和地址,点击创建
在这里插入图片描述
在弹出来的对话框中选择“桌面应用程序(.exe)”,勾选 “空项目”,点击确定建建一个空的Windwos桌面项目。
我这里没有勾选“空项目”,主要是偷懒不想写注册窗口和消息处理。
在这里插入图片描述

2.2 修改符合模式

创建项目后在项目属性找c/c++,找到语言,找到符合模式选否。不改碰到错误时再改也行。
在这里插入图片描述

2.3 下载d3dx12.h文件

打开官方github上面的d3dx12.h文件,把里面的代码拷贝下来。在工程解决方案里创建一个叫DXUtils的文件夹,在这个文件夹下新建一个d3dx12.h文件,把内容放在里面。
这个文件里面是官方写好的一些辅助结构体,不属于DirectX 12 SDK的核心部分,但是可以通过微软官方网站下载获得,方便我们后面开发。
以CD3DX12作为前缀的结构体全都定义在d3dx12.h头文件当中。

2.4 创建一个异常类D3DException,定义抛出异常实例的宏ThrowIfFailed

#pragma once#include <windows.h>	// Windows 头文件
#include<string>        // 提供wsring类,在Windows平台上应该使用wstring和wchar_t// 定义异常类
class D3DException
{
public:D3DException() = default;// 显示:异常函数的返回值、函数名、代码所处文件名,所处代码行数D3DException(HRESULT hr, const std::wstring& functionName, const std::wstring& filename, int lineNumber){ErrorCode = hr;FunctionName = functionName;Filename = filename;LineNumber = lineNumber;}std::wstring ToString()const;HRESULT ErrorCode = S_OK;std::wstring FunctionName;std::wstring Filename;int LineNumber = -1;
};// AnsiToWString函数(将字符串映射到 UTF-16 (宽字符) 字符串)
inline std::wstring AnsiToWString(const std::string& str)
{WCHAR buffer[512];MultiByteToWideChar(CP_ACP, 0, str.c_str(), -1, buffer, 512);return std::wstring(buffer);
}// 如果发生异常,抛出一个异常实例
#ifndef ThrowIfFailed
#define ThrowIfFailed(x)                                              \
{                                                                     \HRESULT hr__ = (x);                                               \std::wstring wfn = AnsiToWString(__FILE__);                       \if(FAILED(hr__)) { throw D3DException(hr__, L#x, wfn, __LINE__); } \
}
#endif

3 D3D12的初始化步骤

准备工作做发了,下面我们就来完成D3D12的初始化

要初始化D3D12,首先需要创建D3D12设备(ID3D12Device)。我们可以通过该ID3D12Device与硬件进行交互,命令硬件完成一些工作(比如:在显存中分配资源、清空后台缓冲区、将资源绑定到各种管线阶段、绘制几何体)。具体而言:

接口说明
ID3D12Device代表着一个显示适配器,显示适配器一般指3D图形硬件即显卡。显示适配器也可以用软件通过模拟硬件显卡的计算处理过程来代替,软件也可以作为适配器(如WARP适配器)。
  • Direct3D设备既可以检测系统环境对功能的支持情况,又能创建所有其他的Direct3D接口对象(如资源、视图和命令列表)。
  • 创建Direct3D设备使用函数D3D12CreateDevice。

现在我们开始在MyD3DApp.cpp中来增加设备的初始化。

3.1 初始化前的准备

3.1.1 增加头文件的引用:

#include<wrl.h>         // 提供了ComPtr类,它是COM对象的智能指针,使我们无需手动Release
#include<d3d12.h>       // Direct3D12头文件,ID3D12开头类型始于此
#include<dxgi1_4.h>     // DirectX图形基础设施头文件,IDXGI开头类型始于此
#include<string>        // 提供wsring类,在Windows平台上应该使用wstring和wchar_t
#include<assert.h>
#include "DXUtils/d3dx12.h"

3.1.2 引用Direct3D库文件、IDXGI库文件

// 引用Direct3D库文件
#pragma comment (lib, "d3d12.lib")
// 引用IDXGI库文件
#pragma comment (lib, "dxgi.lib")

增加使用Windows运行时库(Windows Runtime Library,WRL)为COM对象提供的COM对象的智能指针Microsoft::WRL::ComPtr类的命名空间。

using namespace Microsoft::WRL; // 方便使用Microsoft::WRL::ComPtr类   

3.1.5 增加InitDirect3D函数

// 初始化Direct3D
bool InitDirect3D()
{
}

除了全局变量,后面的代码都在InitDirect3D中增加

3.1.6 启用D3D12的调试层

#if defined(DEBUG) || defined(_DEBUG) // 如果在Debug模式,启用D3D12的调试层,// 启用调试后,D3D会在错误发生时向VC++的输出窗口发送调试信息{ComPtr<ID3D12Debug> debugController;ThrowIfFailed(D3D12GetDebugInterface(IID_PPV_ARGS(&debugController)));debugController->EnableDebugLayer();}
#endif

3.2 创建Direct3D设备

3.2.1 增加IDXGIFactory和ID3D12Device两个全局的COM对象智能指针

// IDXGIFactory: DXGI中最关键的接口之一,可以枚举显示适配器
ComPtr<IDXGIFactory4> mdxgiFactory;// ID3D12Device: 代表一个显示适配器(显卡)
ComPtr<ID3D12Device> md3dDevice;
  • D3D12Device 就是Direct3D中用于提供显卡控制接口的对象,它代表着当前系统中的显示适配器。一般来说,它是一个3D图形硬件(如显卡), 但是,操作系统在没有显卡的时候也能正常的显示图像,这时候使用的就是软件显示适配器,如(WARP适配器)
  • 获取显示适配器
    显示适配器是真正实现了图形处理能力的对象,上面的D3D12Device是对显示适配器的进一步封装。
    这是我系统中的显示适配器
    在这里插入图片描述
    那么在程序中我们怎么才能知道使用的是哪个适配器呢,毕竟游戏的使用性能较高的适配较好。
  • 我们了解下DXGI的概念。
    DXGI是一种与Direct3D配合使用的API,设计DXGI的基本理念是使得多种图形API中的底层任务能够使用通用的API,比如3D和2D的图形API在底层都可以使用相同的,比如Direct3D和Direct2D内部实现交换链时可以使用同一套接口。
    我们在获取系统的可用显示适配器时,会使用到 IDXGIFactory,主要用于创建SwapChain以及枚举显示适配器
    后面的代码会使用IDXGIFactory4来枚举系统中的显示适配器

3.2.2 创建一个D3D12设备

    // 创建可用于生成其他 DXGI 对象的 DXGI 1.0 FactoryThrowIfFailed(CreateDXGIFactory1(IID_PPV_ARGS(&mdxgiFactory)));// 创建一个D3D硬件适配器// D3D12CreateDevice参数为:(适配器指针,应用程序所需最低功能级别,所建ID3D12Device接口的COM ID)// 返回所创建的D3D12设备HRESULT hardwareResult = D3D12CreateDevice(nullptr,                   // 适配器指针,创建设备时用的显示适配器,适配器指针传入nullptr表示使用主显示适配器D3D_FEATURE_LEVEL_12_0,    // 应用程序需要硬件支持的最低功能级别IID_PPV_ARGS(&md3dDevice));// IID_PPV_ARGS是Direct3D为我们提供的一个工具宏,它为我们生成了接口中的后两个参数

IID_PPV_ARGS宏会展开为两项,第一项根据__uuidof获取了COM对象的ID(全局唯一标识符,GUID),IID_PPV_ARGS辅助函数本质是把指针强制转换为void类型

3.2.3 如果创建主显示适配器失败,使用WARP软件适配器

    // 创建主显示适配器失败,使用WARP软件适配器if (FAILED(hardwareResult)){ComPtr<IDXGIAdapter> pWarpAdapter;ThrowIfFailed(mdxgiFactory->EnumWarpAdapter(IID_PPV_ARGS(&pWarpAdapter)));// 不同windows版本的WARP最高支持的功能级别也不同// 再次创建D3D设备时,仅适配器指针参数不同,传入WARP适配器指针ThrowIfFailed(D3D12CreateDevice(pWarpAdapter.Get(),D3D_FEATURE_LEVEL_12_0,IID_PPV_ARGS(&md3dDevice)));}

3.3 创建围栏

CPU和GPU的同步需要使用到围栏,第一步中我们创建好设备,第二步就可以创建围栏了。
另外,如果使用描述符进行工作,需要获取描述符的大小。描述符在不同GPU平台上的大小不同,因此我们需要将获取的描述符大小信息缓存起来,方便需要时直接引用。

3.3.1 增加全局围栏和描述符变量

// ID3D12Fence: 表示围栏
ComPtr<ID3D12Fence> mFence;// RTV描述符大小,RTV描述符: 渲染目标视图资源
UINT mRtvDescriptorSize = 0;
// DSV描述符大小,DSV描述符: 深度/模板视图资源
UINT mDsvDescriptorSize = 0;
// CbvSrvUav描述符大小,CBV描述符: 常量缓冲区视图资源...
UINT mCbvSrvUavDescriptorSize = 0;

3.3.2 创建D3D12围栏

bool CreateD3DFence()
{// 创建围栏ThrowIfFailed(md3dDevice->CreateFence(0,D3D12_FENCE_FLAG_NONE,IID_PPV_ARGS(&mFence)));// 获取描述符大小mRtvDescriptorSize = md3dDevice->GetDescriptorHandleIncrementSize(D3D12_DESCRIPTOR_HEAP_TYPE_RTV);mDsvDescriptorSize = md3dDevice->GetDescriptorHandleIncrementSize(D3D12_DESCRIPTOR_HEAP_TYPE_DSV);mCbvSrvUavDescriptorSize = md3dDevice->GetDescriptorHandleIncrementSize(D3D12_DESCRIPTOR_HEAP_TYPE_CBV_SRV_UAV);
}

可以看到新出现的四个变量都是通过 md3dDevice 对象获取到的,这说明:
 1. Direct3D12 设备对象能创建所有其他的Direct3D接口对象。
 2. COM对象需要使用特定方法或通过其他COM对象的方法获取。

3.3.3 我们来了解下资源与描述符

  • 在渲染过程中,GPU需要对资源进行读和写。但是GPU和资源并不是直接绑定的,而是通过一个中间角色 “描述符”来进行绑定的。
  • 描述符是一种对送往GPU的资源进行描述的轻量级结构,它是一个中间层。当GPU需要对资源进行读或写时,GPU就会问描述符:资源在哪里,我应该按照哪种数据格式进行读写。
  • 描述符的作用有两点:
     1. 指定资源数据。
     2. 为GPU解释资源信息。
  • 创建资源时可用无类型格式,如DXGI_FORMAT_R8G8B8A8_TYPELESS类型,它是4个分量组成,每个分量占8个位。如果某个资源在创建时采用了无类型格式,那么在为它创建描述符时必须指明其具体类型。
  • 注意:视图和描述符的含义是等价的。
  • 每个描述符都有一种具体类型,用来明确资源的具体作用,常用的描述符如下:
描述符说明
CBV常量缓冲区视图
SRV着色器资源视图
UAV无序访问视图
sampler采样器资源描述符
RTV渲染目标视图资源
DSV深度/模板视图资源
  • 描述符堆中存有一系列描述符,本质上是存放用户程序中某特定类型描述符的一块内存。我们需要为每一种类型的描述符创建出单独的描述符,当然同一种描述符也可以创建多个描述符堆。
  • 我们可以用不同的描述符来描述同一个资源,达到以不同的数据格式或内容部分去读写资源的目的。
  • 由于创建描述符的过程中需要执行一些类型的检测和验证工作,最好不要在运行时才创建描述符,创建描述符的最佳时机为初始化期间。
  • 当然有时确实需要使用无类型资源所带来的灵活性,此时在一定限度内,可以考虑在运行时创建描述符。

3.4 检测对4X MSAA 质量级别的支持

我们要使用4X MSAA,之所以选择4X,是因为借助此采样数量就可以获取开销不高但性能非凡的效果。而使用4X MSAA之前,我们要先检查设备是否支持4X MSAA 质量级别的图像。

3.4.1 增加全局资源数据格式和4X MASS质量级别变量

void Check4XMSAA()
{// DXGI_FORMAT: 资源数据的格式,一种枚举类型 DXGI_FORMAT mBackBufferFormat = DXGI_FORMAT_R8G8B8A8_UNORM;// 检测对4X MASS质量级别的支持D3D12_FEATURE_DATA_MULTISAMPLE_QUALITY_LEVELS msQualityLevels;msQualityLevels.Format = mBackBufferFormat;                         // 纹理格式msQualityLevels.SampleCount = 4;                                    // 采样次数msQualityLevels.Flags = D3D12_MULTISAMPLE_QUALITY_LEVELS_FLAG_NONE; // 默认选项msQualityLevels.NumQualityLevels = 0;                               // 质量级别// 使用ID3D12Device::CheckFeatureSupport函数,查询我们设置的这种图像质量的级别ThrowIfFailed(md3dDevice->CheckFeatureSupport(D3D12_FEATURE_MULTISAMPLE_QUALITY_LEVELS

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

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

相关文章

pytorch实现水果2分类(蓝莓,苹果)

1.数据集的路径&#xff0c;结构 dataset.py 目的&#xff1a; 输入&#xff1a;没有输入&#xff0c;路径是写死了的。 输出&#xff1a;返回的是一个对象&#xff0c;里面有self.data。self.data是一个列表&#xff0c;里面是&#xff08;图片路径.jpg&#xff0c;标签&…

JMH325【剑侠情缘3】第2版80级橙武网游单机更稳定亲测视频安装教学更新整合收集各类修改教学补丁兴趣可以慢慢探索

资源介绍&#xff1a; 是否需要虚拟机&#xff1a;是 文件大小&#xff1a;压缩包约14G 支持系统&#xff1a;win10、win11 硬件需求&#xff1a;运行内存8G 4核及以上CPU独立显卡 下载方式&#xff1a;百度网盘 任务修复&#xff1a; 1&#xff0c;掌门任务&#xff08…

【Android组件】封装加载弹框

&#x1f4d6;封装加载弹框 ✅1. 构造LoadingDialog✅2. 调用LoadingDialog 效果&#xff1a; ✅1. 构造LoadingDialog 构造LoadingDialog类涉及到设计模式中的建造者模式&#xff0c;进行链式调用&#xff0c;注重的是构建的过程&#xff0c;设置需要的属性。 步骤一&#x…

[数据结构] 归并排序快速排序 及非递归实现

&#xff08;&#xff09;标题&#xff1a;[数据结构] 归并排序&&快速排序 及非递归实现 水墨不写bug &#xff08;图片来源于网络&#xff09; 目录 (一)快速排序 类比递归谋划非递归 快速排序的非递归实现&#xff1a; &#xff08;二&#xff09;归并排序 归…

Elasticsearch文档_id以数组方式返回

背景需求是只需要文档的_id字段&#xff0c;并且_id组装成一个数组。 在搜索请求中使用 script_fields 来整理 _id 为数组输出&#xff1a; POST goods_info/_search?size0 {"query": {"term": {"brand": {"value": "MGC"…

明白这两大关键点,轻松脱单不再是难题!

很多未婚男女都渴望找到心仪的伴侣&#xff0c;建立稳定的情感关系&#xff0c;但往往在脱单的过程中跌跌撞撞。平时与同学、同事之间相处得很融洽&#xff0c;一旦遇到心仪的异性&#xff0c;情商直接掉线&#xff0c;难道情商也会选择性地发挥作用吗&#xff1f;其实&#xf…

什么牌子的开放式耳机好用?南卡、Cleer、小米、开石超值机型力荐!

​开放式耳机在如今社会中已经迅速成为大家购买耳机的新趋势&#xff0c;深受喜欢听歌和热爱运动的人群欢迎。当大家谈到佩戴的稳固性时&#xff0c;开放式耳机都会收到一致好评。对于热爱运动的人士而言&#xff0c;高品质的开放式耳机无疑是理想之选。特别是在近年来的一些骑…

AnimateLCM:高效生成连贯真实的视频

视频扩散模型因其能够生成连贯且高保真的视频而日益受到关注。然而&#xff0c;迭代去噪过程使得这类模型计算密集且耗时&#xff0c;限制了其应用范围。香港中文大学 MMLab、Avolution AI、上海人工智能实验室和商汤科技公司的研究团队提出了AnimateLCM&#xff0c;这是一种允…

电子电气架构 --- 关于DoIP的一些闲思 上

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 屏蔽力是信息过载时代一个人的特殊竞争力,任何消耗你的人和事,多看一眼都是你的不对。非必要不费力证明自己,无利益不试图说服别人,是精神上的节…

JavaDS —— 单链表 与 LinkedList

顺序表和链表区别 ArrayList &#xff1a; 底层使用连续的空间&#xff0c;可以随机访问某下标的元素&#xff0c;时间复杂度为O&#xff08;1&#xff09; 但是在插入和删除操作的时候&#xff0c;需要将该位置的后序元素整体往前或者向后移动&#xff0c;时间复杂度为O&…

什么是智能制造?

科技的每一次飞跃都深刻改变着我们的生产生活方式。其中&#xff0c;智能制造作为工业4.0的核心概念&#xff0c;正引领着全球制造业向更加高效、灵活、智能的方向迈进。那么&#xff0c;究竟什么是智能制造&#xff1f;它如何重塑我们的工业版图&#xff0c;又将对未来社会产生…

TTT架构超越Transformer,ML模型替代RNN隐藏状态!

目录 01 算法原理 02 骨干架构 03 实验结果 一种崭新的大语言模型&#xff08;LLM&#xff09;架构有望取代当前主导 AI 领域的 Transformer&#xff0c;并在性能上超越 Mamba。 论文地址&#xff1a;https://arxiv.org/abs/2407.04620 本周一&#xff0c;关于 Test-Time Tr…

修复 Ubuntu 24.04 Dock 丢失应用程序图标

找出应用程序窗口的类名 首先&#xff0c;您需要启动应用程序窗口。然后&#xff0c;按 Alt F2 启动“运行 Command”对话框。当对话框打开时&#xff0c;输入 lg 并按 Enter 键。 在该窗口中&#xff0c;单击Windows按钮&#xff0c;然后找出目标应用程序窗口的类名称。 在/…

Flutter——最详细(Table)网格、表格组件使用教程

背景 用于展示表格组件&#xff0c;可指定线宽、列宽、文字方向等属性 属性作用columnWidths列的宽度defaultVerticalAlignment网格内部组件摆放方向border网格样式修改children表格里面的组件textDirection文本排序方向 import package:flutter/material.dart;class CustomTa…

公众号运营秘籍:8 大策略让你的粉丝翻倍!

在当今信息爆炸的时代&#xff0c;微信公众号的运营者们面临着前所未有的挑战&#xff1a;如何在这个充满竞争的红海中脱颖而出&#xff0c;吸引并留住粉丝&#xff1f;事实上&#xff0c;微信公众号的红利期并未完全过去&#xff0c;关键在于我们如何策略性地运营&#xff0c;…

使用PEFT库进行ChatGLM3-6B模型的QLORA高效微调

PEFT库进行ChatGLM3-6B模型QLORA高效微调 QLORA微调ChatGLM3-6B模型安装相关库使用ChatGLM3-6B模型GPU显存占用准备数据集加载数据集数据处理数据集处理加载量化模型-4bit预处理量化模型配置LoRA适配器训练超参数配置开始训练保存LoRA模型模型推理合并模型使用微调后的模型 QLO…

【Pytorch实用教程】transformer中创建嵌入层的模块nn.Embedding的用法

文章目录 1. nn.Embedding的简单介绍1.1 基本用法1.2 示例代码1.3 注意事项2. 通俗的理解num_embeddings和embedding_dim2.1 num_embeddings2.2 embedding_dim2.3 使用场景举例结合示例1. nn.Embedding的简单介绍 nn.Embedding 是 PyTorch 中的一个模块,用于创建一个嵌入层。…

准大一新生开学千万要带证件照用途大揭秘

1、提前关注好都有哪些考场&#xff0c;以及这些考场大致在网页的哪个位置。比如我选对外经贸大学&#xff0c;我就直接找到第二个点进去。 2、电脑上同时开了谷歌浏览器和IE浏览器&#xff0c;以及手机也登陆了。亲测下来&#xff0c;同一时间刷新&#xff0c;谷歌浏览器能显示…

​cesium、three.js,三维GIS为啥那么热?到底怎么学呢?

​cesium、three.js&#xff0c;三维GIS为啥那么热&#xff1f;他们的应用场景都是什么呢&#xff1f;接下来我们可以一起来看看~ 三维GIS的应用 GIS和3D的应用是趋势&#xff0c;目前已经有很多应用案例&#xff0c;例如BIM&#xff0c;智慧城市&#xff0c;数字孪生等。如下…

汇聚荣拼多多电商实力强吗?

汇聚荣拼多多电商实力强吗?汇聚荣拼多多&#xff0c;作为中国电商领域的后起之秀&#xff0c;已经在市场上占据了一席之地。那么&#xff0c;它的实力究竟如何呢?在回答这个问题之前&#xff0c;我们需要先了解一下拼多多的基本情况。拼多多是一家以社交电商为主要模式的购物…