探索C++与Live555实现RTSP服务器的艺术(一)

简介: 探索C++与Live555实现RTSP服务器的艺术

一、引言(Introduction)

1.1 RTSP服务器的重要性(Importance of RTSP Server)

RTSP(Real Time Streaming Protocol,实时流传输协议)服务器在现代网络传输中扮演着至关重要的角色。RTSP服务器主要用于控制音频或视频的多媒体会话,它为互联网上的音频视频服务提供了一种控制框架,使得用户可以方便地播放、暂停、快进、倒退等操作,极大地提高了用户体验。

RTSP服务器的重要性主要体现在以下几个方面:

  1. 实时性(Real-time):RTSP服务器支持实时传输,这对于直播、视频会议等应用场景至关重要。用户可以实时接收并播放音视频数据,无需等待整个文件下载完成。
  2. 控制性(Controllability):RTSP服务器提供了丰富的控制命令,如PLAY、PAUSE、TEARDOWN等,用户可以通过这些命令对音视频播放进行精确控制。
  3. 扩展性(Scalability):RTSP服务器可以支持大量的并发连接,适应大规模用户的需求。同时,RTSP服务器还支持多种传输协议,如TCP、UDP等,具有很好的扩展性。
  4. 互动性(Interactivity):RTSP服务器支持双向数据传输,可以实现用户与服务器之间的交互,如聊天、投票等。
  5. 安全性(Security):RTSP服务器支持多种安全机制,如基于证书的身份验证、数据加密等,可以保护用户的隐私和数据安全。

因此,深入理解和掌握RTSP服务器的构建和优化,对于提升我们的网络应用服务质量,满足用户对于音视频服务的高需求具有重要的意义。在接下来的章节中,我们将详细介绍如何使用C++和Live555库来构建高效、稳定的RTSP服务器。

1.2 C++与Live555库的优势(Advantages of C++ and Live555)

在构建RTSP服务器的过程中,选择合适的编程语言和库是非常关键的。C++和Live555库在这方面具有显著的优势。

C++的优势:

  1. 性能优越(Superior Performance):C++是一种静态类型的编译语言,它提供了丰富的语言特性和强大的性能,可以满足高性能服务器的需求。
  2. 内存管理(Memory Management):C++提供了直接的内存管理能力,开发者可以精确地控制内存的分配和释放,这对于处理大量的并发连接和数据流非常重要。
  3. 面向对象(Object-Oriented):C++是一种支持面向对象编程的语言,它的封装、继承和多态等特性使得代码更加模块化,易于维护和扩展。

Live555库的优势:

  1. 专业性(Professionalism):Live555是一个专门用于实现RTSP服务器的开源库,它提供了一套完整的API,可以方便地实现RTSP的各种功能。
  2. 稳定性(Stability):Live555库经过了长时间的测试和使用,稳定性非常高,可以满足生产环境的需求。
  3. 跨平台(Cross-platform):Live555库支持多种操作系统,包括Windows、Linux、Mac OS等,开发者可以在不同的平台上进行开发和部署。
  4. 社区支持(Community Support):Live555有一个活跃的开源社区,开发者可以从社区获取帮助,分享经验,同时也可以为社区贡献代码。

综上,C++和Live555库是构建RTSP服务器的理想选择。在接下来的章节中,我们将详细介绍如何使用C++和Live555库来构建RTSP服务器。

1.3 可能用到的类和接口介绍

在Live555库中,以下是一些可能会用到的类和方法:

  1. UsageEnvironment类:这是一个抽象基类,用于定义事件循环。你可能会使用到它的taskScheduler()方法来获取任务调度器。
  2. BasicUsageEnvironment类:这是UsageEnvironment的一个具体实现,你可能会使用它的createNew()方法来创建一个新的环境。
  3. RTSPServer类:这个类用于创建和管理RTSP服务器。你可能会使用到它的createNew()方法来创建一个新的服务器,以及addServerMediaSession()方法来添加一个新的媒体会话。
  4. ServerMediaSession类:这个类表示一个RTSP服务器的媒体会话。你可能会使用到它的addSubsession()方法来添加一个新的子会话。
  5. H264VideoStreamFramer类:这个类用于创建H.264视频流。你可能会使用到它的createNew()方法来创建一个新的视频流。

以上只是一些基本的类和方法,实际使用时可能还需要其他的类和方法。具体的使用方法和参数应根据Live555的官方文档和源代码来确定。

在使用Live555库时,可能会用到以下一些函数和接口:

  1. RTSPClient::createNew: 创建一个新的RTSP客户端实例。这个函数需要一个环境变量,RTSP URL,以及一个可选的应用程序名称和端口号。
  2. RTSPServer::createNew: 创建一个新的RTSP服务器实例。这个函数需要一个环境变量和一个可选的服务器端口号。
  3. RTSPClient::sendPlayCommand: 发送一个PLAY命令到RTSP服务器。这个函数需要一个媒体会话和一个可选的回调函数。
  4. RTSPClient::sendPauseCommand: 发送一个PAUSE命令到RTSP服务器。这个函数需要一个媒体会话和一个可选的回调函数。
  5. RTSPClient::sendTeardownCommand: 发送一个TEARDOWN命令到RTSP服务器。这个函数需要一个媒体会话和一个可选的回调函数。
  6. RTSPServer::setUpTunnelingOverHTTP: 设置RTSP服务器以通过HTTP进行隧道传输。这个函数需要一个HTTP端口号。
  7. RTSPServer::addUserRecord: 添加一个用户记录到RTSP服务器的用户数据库。这个函数需要一个用户名和密码。
  8. RTSPServer::removeUserRecord: 从RTSP服务器的用户数据库中删除一个用户记录。这个函数需要一个用户名。

以上信息是根据GitHub上的代码片段提取的,可能并不完全准确,具体使用时还需要参考Live555的官方文档和源代码。

二、C++与Live555库概述(Overview of C++ and Live555)

2.1 C++语言特性(Features of C++)

C++是一种通用的编程语言,它在C语言的基础上增加了面向对象编程(Object-Oriented Programming,简称OOP)的特性。C++的主要特性可以归纳为以下几点:

  1. 面向对象编程(Object-Oriented Programming):C++支持封装(Encapsulation)、继承(Inheritance)和多态(Polymorphism)等面向对象的基本特性。这些特性使得C++能够更好地处理复杂的问题,提高代码的可重用性和可维护性。
  2. 泛型编程(Generic Programming):C++的模板(Template)特性支持泛型编程,这使得我们可以编写出通用的、可重用的代码。
  3. 低级操作(Low-Level Manipulation):C++允许程序员进行直接的内存操作,这使得C++能够编写出高效的代码,满足对性能要求较高的场景。
  4. 丰富的库支持(Rich Library Support):C++有着丰富的标准库(如STL)和第三方库(如Live555),这些库提供了大量的数据结构和算法,极大地提高了开发效率。
  5. 兼容C语言(Compatibility with C):C++几乎完全兼容C语言,这使得C++可以直接使用大量的C语言代码和库。

以上特性使得C++成为了一种强大而灵活的编程语言,它被广泛应用于各种领域,包括操作系统、游戏开发、嵌入式系统、实时系统、网络编程等。在RTSP服务器的开发中,C++的这些特性也能够得到充分的利用。

2.2 Live555库简介(Introduction to Live555)

Live555是一个开源的流媒体库,它提供了一系列的API接口,用于实现RTSP(Real Time Streaming Protocol)服务器和客户端的功能。Live555库的主要特性可以归纳为以下几点:

  1. 支持多种流媒体协议:Live555支持多种流媒体协议,包括RTSP、RTP(Real-time Transport Protocol)、RTCP(RTP Control Protocol)等。这使得Live555能够处理各种复杂的流媒体传输任务。
  2. 支持多种媒体格式:Live555支持多种媒体格式,包括H.264、H.265、AAC、MP3等。这使得Live555能够处理各种类型的媒体数据。
  3. 支持多平台:Live555可以在多种操作系统上运行,包括Windows、Linux、Mac OS等。这使得Live555具有很好的跨平台性。
  4. 高效的性能:Live555在设计和实现上注重效率,它能够处理大量的并发连接,满足高性能的需求。
  5. 丰富的示例代码:Live555提供了丰富的示例代码,这些代码覆盖了各种常见的使用场景,对于开发者来说是很好的学习资料。

在RTSP服务器的开发中,Live555库的这些特性使得它成为了一个理想的选择。通过使用Live555,我们可以更加方便和高效地实现RTSP服务器的功能。

2.3 Live555在RTSP服务器中的应用(Application of Live555 in RTSP Server)

在RTSP服务器的开发中,Live555库发挥着重要的作用。以下是一些主要的应用场景:

  1. 媒体流的封装和传输:Live555提供了一系列的类和函数,用于封装和传输媒体流。例如,我们可以使用H264VideoStreamFramer类来封装H.264视频流,然后使用RTPSink类来将封装后的流发送到客户端。
  2. RTSP会话的管理:Live555提供了RTSPServer类,用于管理RTSP会话。RTSPServer类可以处理客户端的连接请求,管理媒体流的播放和暂停,以及处理其他的RTSP命令。
  3. 媒体源的管理:Live555提供了MediaSession类,用于管理媒体源。MediaSession类可以管理一个或多个媒体流,它提供了一种统一的接口,使得我们可以方便地控制媒体流的播放和暂停。
  4. 错误处理和日志记录:Live555提供了一系列的错误处理和日志记录功能,这些功能可以帮助我们更好地调试和维护RTSP服务器。

通过以上的应用,我们可以看到,Live555库为RTSP服务器的开发提供了强大的支持。在后续的章节中,我们将详细介绍如何使用Live555库来构建RTSP服务器。

三、RTSP服务器的构建流程(Building Process of RTSP Server)

3.1 RTSP服务器的基本架构(Basic Architecture of RTSP Server)

RTSP(Real Time Streaming Protocol,实时流传输协议)服务器是一种专门用于控制音频或视频流的传输的服务器。它的主要功能是控制数据流的播放、暂停、回放等操作,而数据流的传输则通常由RTP(Real-time Transport Protocol,实时传输协议)或者UDP(User Datagram Protocol,用户数据报协议)来完成。

RTSP服务器的基本架构主要包括以下几个部分:

  1. 客户端(Client):客户端是发起RTSP请求的一方,通常是用户的播放设备,如电脑、手机等。
  2. 服务器(Server):服务器是接收和处理RTSP请求的一方,负责对请求进行响应,并控制数据流的传输。
  3. 媒体流(Media Stream):媒体流是服务器传输的音频或视频数据,通常由RTP或UDP协议进行传输。
  4. 控制协议(Control Protocol):控制协议是用于控制媒体流传输的协议,通常是RTSP协议。

在这个架构中,客户端通过发送RTSP请求(如PLAY、PAUSE等)给服务器,服务器接收到请求后,通过控制协议来控制媒体流的传输。例如,当客户端发送一个PLAY请求时,服务器会开始传输媒体流;当客户端发送一个PAUSE请求时,服务器会暂停传输媒体流。

这种架构的优点是,它将数据流的传输和控制分离开来,使得服务器可以更灵活地控制数据流的传输,而客户端也可以更方便地控制数据流的播放。同时,由于RTSP协议是基于TCP(Transmission Control Protocol,传输控制协议)的,因此它可以提供可靠的数据传输,确保数据流的完整性和准确性。

在下一节中,我们将详细介绍如何使用C++和Live555库来构建一个RTSP服务器,包括如何设置服务器、如何处理客户端的请求,以及如何控制媒体流的传输等。

3.2 使用C++和Live555构建RTSP服务器(Building RTSP Server with C++ and Live555)

Live555是一个开源的流媒体库,提供了一套完整的API,可以方便地构建RTSP服务器。在这一节中,我们将详细介绍如何使用C++和Live555库来构建一个RTSP服务器。

步骤一:环境配置(Environment Configuration)

首先,我们需要在我们的开发环境中安装Live555库。Live555库支持多种操作系统,包括Windows、Linux和Mac OS等。我们可以从Live555的官方网站下载源代码,然后按照官方的安装指南进行安装。

步骤二:创建RTSP服务器(Creating RTSP Server)

创建RTSP服务器的第一步是创建一个RTSP服务器实例。在Live555库中,我们可以使用RTSPServer::createNew函数来创建一个RTSP服务器实例。这个函数需要一个UsageEnvironment参数,这是Live555库中的一个核心类,用于处理事件循环和错误报告。

TaskScheduler* scheduler = BasicTaskScheduler::createNew();
UsageEnvironment* env = BasicUsageEnvironment::createNew(*scheduler);
RTSPServer* rtspServer = RTSPServer::createNew(*env, 8554);

在上面的代码中,我们首先创建了一个TaskScheduler实例和一个UsageEnvironment实例,然后使用这两个实例作为参数创建了一个RTSP服务器实例。这个RTSP服务器监听8554端口。

步骤三:创建媒体会话(Creating Media Session)

创建RTSP服务器后,我们需要创建一个或多个媒体会话(Media Session)。每个媒体会话对应一个媒体流,可以是音频流、视频流或者其他类型的媒体流。

在Live555库中,我们可以使用ServerMediaSession::createNew函数来创建一个媒体会话。这个函数需要一个UsageEnvironment参数,一个会话名称,以及一些其他的参数,如媒体源、媒体类型等。

ServerMediaSession* sms = ServerMediaSession::createNew(*env, "testStream", "testStream",
    "Session streamed by \"testStream\"", True /*SSM*/);
rtspServer->addServerMediaSession(sms);

在上面的代码中,我们创建了一个名为"testStream"的媒体会话,并将这个会话添加到了RTSP服务器中。

步骤四:启动事件循环(Starting Event Loop)

最后,我们需要启动事件循环,让RTSP服务器开始处理客户端的请求。

env->taskScheduler().doEventLoop(); // does not return

在上面的代码中,我们调用了doEventLoop函数,这个函数会阻塞当前线程,直到事件循环被停止。

Here is an example of a C++ program that uses the Live555 library to create an RTSP server:

#include "liveMedia.hh"
#include "BasicUsageEnvironment.hh"
int main(int argc, char** argv) {
  // Begin by setting up our usage environment:
  TaskScheduler* scheduler = BasicTaskScheduler::createNew();
  UsageEnvironment* env = BasicUsageEnvironment::createNew(*scheduler);
  UserAuthenticationDatabase* authDB = NULL;
#ifdef ACCESS_CONTROL
  // To implement client access control to the RTSP server, do the following:
  authDB = new UserAuthenticationDatabase;
  authDB->addUserRecord("username1", "password1"); // replace these with real strings
  // Repeat the above with each <username>, <password> that you wish to allow
  // access to the server.
#endif
  // Create the RTSP server. Try first with the default port number (554),
  // and then with the alternative port number (8554):
  RTSPServer* rtspServer;
  portNumBits rtspServerPortNum = 554;
  rtspServer = RTSPServer::createNew(*env, rtspServerPortNum, authDB);
  if (rtspServer == NULL) {
    rtspServerPortNum = 8554;
    rtspServer = RTSPServer::createNew(*env, rtspServerPortNum, authDB);
  }
  if (rtspServer == NULL) {
    *env << "Failed to create RTSP server: " << env->getResultMsg() << "\n";
    exit(1);
  }
  char* descriptionString
    = (char*)"Session streamed by \"testOnDemandRTSPServer\"";
  // A H.264 video elementary stream:
  {
    char const* streamName = "h264ESVideoTest";
    char const* inputFileName = "test.264";
    ServerMediaSession* sms
      = ServerMediaSession::createNew(*env, streamName, streamName,
              descriptionString);
    sms->addSubsession(H264VideoFileServerMediaSubsession
           ::createNew(*env, inputFileName, False));
    rtspServer->addServerMediaSession(sms);
    announceStream(rtspServer, sms, streamName, inputFileName);
  }
  env->taskScheduler().doEventLoop(); // does not return
  return 0; // only to prevent compiler warning
}

This program creates an RTSP server that streams a H.264 video file named “test.264”. The server listens on port 554 or 8554 if 554 is not available. The video stream is named “h264ESVideoTest”. The server runs an event loop that does not return, waiting for and responding to RTSP requests.

Please note that you need to replace "test.264" with the actual path to your H.264 video file. Also, this is a simplified example and does not include error handling and other necessary production-level code.

You can find the full code and more examples in the Live555 GitHub repository.

3.3 代码示例与解析(Code Examples and Analysis)

在这一部分,我们将通过一个具体的代码示例来展示如何使用C++和Live555库构建RTSP服务器。我们将详细解析每一行代码的作用,以帮助读者更好地理解和掌握这一过程。

首先,我们需要包含必要的头文件。Live555库提供了一系列的头文件,这些头文件定义了我们在构建RTSP服务器时所需要的各种类和函数。

#include "liveMedia.hh"
#include "BasicUsageEnvironment.hh"
#include "GroupsockHelper.hh"

接下来,我们需要创建一个UsageEnvironment实例。UsageEnvironment是Live555库中的一个核心类,它提供了一系列的基础设施,如日志记录、错误处理等。

TaskScheduler* scheduler = BasicTaskScheduler::createNew();
UsageEnvironment* env = BasicUsageEnvironment::createNew(*scheduler);

然后,我们需要创建一个RTSPServer实例。RTSPServer是Live555库中的一个核心类,它封装了RTSP服务器的主要功能。

RTSPServer* rtspServer = RTSPServer::createNew(*env, 8554);
if (rtspServer == NULL) {
  *env << "Failed to create RTSP server: " << env->getResultMsg() << "\n";
  exit(1);
}

在创建RTSPServer实例后,我们需要创建一个ServerMediaSession实例,并将其添加到RTSPServer中。ServerMediaSession代表了一个媒体会话,它可以包含一个或多个媒体流。

ServerMediaSession* sms = ServerMediaSession::createNew(*env, "testStream", "testStream",
                             "Session streamed by \"testRTSPServer\"",
                             True /*SSM*/);
rtspServer->addServerMediaSession(sms);

最后,我们需要启动事件循环,开始处理客户端的请求。

env->taskScheduler().doEventLoop(); // does not return

以上就是一个简单的RTSP服务器的构建过程。通过这个过程,我们可以看到,使用C++和Live555库构建RTSP服务器是一件相对简单的事情。只需要理解并掌握Live555库中的几个核心类和函数,就可以轻松地构建出一个功能强大的RTSP服务器。

这是一个使用Live555库实现的RTSP服务器的C++示例代码。

#include "liveMedia.hh" // 引入liveMedia库
#include "BasicUsageEnvironment.hh" // 引入BasicUsageEnvironment库
#include "GroupsockHelper.hh" // 引入GroupsockHelper库
UsageEnvironment* env; // 定义环境变量
char const* inputFileName = "test.mp4"; // 定义输入文件名
FramedSource* videoSource; // 定义视频源
RTPSink* videoSink; // 定义RTP接收器
void play(); // 定义播放函数
int main(int argc, char** argv) { // 主函数
  TaskScheduler* scheduler = BasicTaskScheduler::createNew(); // 创建新的任务调度器
  env = BasicUsageEnvironment::createNew(*scheduler); // 创建新的环境
  // 创建输入文件源
  videoSource = ByteStreamFileSource::createNew(*env, inputFileName);
  if (videoSource == NULL) {
    *env << "Unable to open file \"" << inputFileName
         << "\" as a byte-stream file source\n";
    exit(1);
  }
  Groupsock rtpGroupsock(*env, destinationAddress, rtpPortNum, ttl); // 创建RTP组套接字
  videoSink = H264VideoRTPSink::createNew(*env, &rtpGroupsock, 96); // 创建新的H264视频RTP接收器
  // 开始播放
  *env << "Beginning to read from file...\n";
  play();
  env->taskScheduler().doEventLoop(); // 进入事件循环
  return 0; // 返回0表示程序正常结束
}
void afterPlaying(void* clientData) { // 播放结束后的回调函数
  *env << "...done reading from file\n";
  videoSink->stopPlaying(); // 停止播放
  Medium::close(videoSink); // 关闭接收器
  Medium::close(videoSource); // 关闭源
  exit(0); // 退出程序
}
void play() { // 播放函数
  videoSink->startPlaying(*videoSource, afterPlaying, videoSink); // 开始播放
}

这个示例中,首先创建了一个任务调度器和环境,然后创建了一个输入文件源。接着,创建了一个RTP组套接字和一个H264视频RTP接收器。最后,开始播放视频,并在播放结束后关闭接收器和源。

这只是一个简单的示例,实际的RTSP服务器可能需要处理更多的复杂情况,例如处理多个客户端的连接,处理不同的RTSP命令等等。


探索C++与Live555实现RTSP服务器的艺术(二)https://developer.aliyun.com/article/1465125

目录
相关文章
|
2月前
|
编解码 监控 网络协议
采用Qt+Live555搭建RTSP服务器
Live555是一个跨平台的流媒体开发库,支持多种流媒体协议,包括RTSP、SIP、RTP等,可以帮助我们快速实现视频流的传输和处理。
205 1
采用Qt+Live555搭建RTSP服务器
|
4月前
|
编解码 C++ 流计算
探索C++与Live555实现RTSP服务器的艺术(三)
探索C++与Live555实现RTSP服务器的艺术
197 1
|
4月前
|
存储 编解码 算法
探索C++与Live555实现RTSP服务器的艺术(二)
探索C++与Live555实现RTSP服务器的艺术
211 1
|
SQL 消息中间件 弹性计算
从ECS到C++软件工程师
从ECS到C++软件工程师
195 0
从ECS到C++软件工程师
|
1月前
|
机器学习/深度学习 编解码 人工智能
阿里云gpu云服务器租用价格:最新收费标准与活动价格及热门实例解析
随着人工智能、大数据和深度学习等领域的快速发展,GPU服务器的需求日益增长。阿里云的GPU服务器凭借强大的计算能力和灵活的资源配置,成为众多用户的首选。很多用户比较关心gpu云服务器的收费标准与活动价格情况,目前计算型gn6v实例云服务器一周价格为2138.27元/1周起,月付价格为3830.00元/1个月起;计算型gn7i实例云服务器一周价格为1793.30元/1周起,月付价格为3213.99元/1个月起;计算型 gn6i实例云服务器一周价格为942.11元/1周起,月付价格为1694.00元/1个月起。本文为大家整理汇总了gpu云服务器的最新收费标准与活动价格情况,以供参考。
阿里云gpu云服务器租用价格:最新收费标准与活动价格及热门实例解析
|
10天前
|
Cloud Native Java 编译器
将基于x86架构平台的应用迁移到阿里云倚天实例云服务器参考
随着云计算技术的不断发展,云服务商们不断推出高性能、高可用的云服务器实例,以满足企业日益增长的计算需求。阿里云推出的倚天实例,凭借其基于ARM架构的倚天710处理器,提供了卓越的计算能力和能效比,特别适用于云原生、高性能计算等场景。然而,有的用户需要将传统基于x86平台的应用迁移到倚天实例上,本文将介绍如何将基于x86架构平台的应用迁移到阿里云倚天实例的服务器上,帮助开发者和企业用户顺利完成迁移工作,享受更高效、更经济的云服务。
将基于x86架构平台的应用迁移到阿里云倚天实例云服务器参考
|
8天前
|
编解码 前端开发 安全
通过阿里云的活动购买云服务器时如何选择实例、带宽、云盘
在我们选购阿里云服务器的过程中,不管是新用户还是老用户通常都是通过阿里云的活动去买了,一是价格更加实惠,二是活动中的云服务器配置比较丰富,足可以满足大部分用户的需求,但是面对琳琅满目的云服务器实例、带宽和云盘选项,如何选择更适合自己,成为许多用户比较关注的问题。本文将介绍如何在阿里云的活动中选择合适的云服务器实例、带宽和云盘,以供参考和选择。
通过阿里云的活动购买云服务器时如何选择实例、带宽、云盘
|
6天前
|
弹性计算 运维 安全
阿里云轻量应用服务器和经济型e实例区别及选择参考
目前在阿里云的活动中,轻量应用服务器2核2G3M带宽价格为82元1年,2核2G3M带宽的经济型e实例云服务器价格99元1年,对于云服务器配置和性能要求不是很高的阿里云用户来说,这两款服务器配置和价格都差不多,阿里云轻量应用服务器和ECS云服务器让用户二选一,很多用户不清楚如何选择,本文来说说轻量应用服务器和经济型e实例的区别及选择参考。
阿里云轻量应用服务器和经济型e实例区别及选择参考
|
7天前
|
机器学习/深度学习 存储 人工智能
阿里云GPU云服务器实例规格gn6v、gn7i、gn6i实例性能及区别和选择参考
阿里云的GPU云服务器产品线在深度学习、科学计算、图形渲染等多个领域展现出强大的计算能力和广泛的应用价值。本文将详细介绍阿里云GPU云服务器中的gn6v、gn7i、gn6i三个实例规格族的性能特点、区别及选择参考,帮助用户根据自身需求选择合适的GPU云服务器实例。
阿里云GPU云服务器实例规格gn6v、gn7i、gn6i实例性能及区别和选择参考
|
17天前
|
存储 弹性计算 运维
阿里云日常运维-购买服务器
这篇文章是关于如何在阿里云购买和配置云服务器ECS的教程。
55 6
阿里云日常运维-购买服务器

热门文章

最新文章