FFMPEG基于内存的转码实例——输入输出视频均在内存

简介: 我在6月份写了篇文章《FFMPEG基于内存的转码实例》,讲如何把视频转码后放到内存,然后通过网络发送出去。但该文章只完成了一半,即输入的数据依然是从磁盘文件中读取。在实际应用中,有很多数据是放到内存的,比如播放从服务器接收到的视频,就是在内存中的。

我在6月份写了篇文章《FFMPEG基于内存的转码实例》,讲如何把视频转码后放到内存,然后通过网络发送出去。但该文章只完成了一半,即输入的数据依然是从磁盘文件中读取。在实际应用中,有很多数据是放到内存的,比如播放从服务器接收到的视频,就是在内存中的。时隔2个月,项目终于完成了,虽然在收尾阶段会花费大量时间,但也算空闲了点。于是就继续完善。

本文中,假定的使用场合是,有一个已经放到内存的视频,需要将它转码成另一种封装格式,还是放到内存中。由于是测试,首先将视频从文件中读取到内存,最后会将转换好的视频写入到另一个文件以检查是否正常。当然,限于能力,代码不可能适合于所有情况,但却可以清楚演示出自定义的IO输入输出的用法。

技术要点简述如下:

1、用户自定义的操作

对于内存的操作使用结构体封装:

typedef struct AVIOBufferContext {
    unsigned char* ptr;
    int pos;
    int totalSize;
    int realSize;
}AVIOBufferContext;

输入、输出均使用该结构体:

AVIOBufferContext g_avbuffer_in;
AVIOBufferContext g_avbuffer_out;

实现,read、write、seek函数。其中read为读取时使用到的,其它2个是写入内存要使用的。以read为例:

static int my_read(void *opaque, unsigned char *buf, int size)
{
    AVIOBufferContext* op = (AVIOBufferContext*)opaque;
    int len = size;
    if (op->pos + size > op->totalSize)
    {
        len = op->totalSize - op->pos;
    }
    memcpy(buf, op->ptr + op->pos, len);
    if (op->pos + len >= op->realSize)
    op->realSize += len;
    
    op->pos += len;

    return len;
}

实质进行的是读取已有内存的size数据,拷贝到buf中。opaque方便参数传递。注意,在拷贝时要对pos累加。

其它函数类似。

2、输出配置关键代码:

    avio_out =avio_alloc_context((unsigned char *)g_ptr_out, IO_BUFFER_SIZE, 1,
                &g_avbuffer_out, NULL, my_write, my_seek);

    avformat_alloc_output_context2(&ofmt_ctx, NULL, NULL, out_filename);
    if (!ofmt_ctx)
    {
        printf( "Could not create output context\n");
        ret = AVERROR_UNKNOWN;
        goto end;
    }
    ofmt_ctx->pb=avio_out; // 赋值自定义的IO结构体
    ofmt_ctx->flags=AVFMT_FLAG_CUSTOM_IO; // 指定为自定义

这个跟上述提到的文章是一致的。只是多了个自定义的结构体。

3、输入配置关键代码:

    avio_in =avio_alloc_context((unsigned char *)g_ptr_in, IO_BUFFER_SIZE, 0,
                &g_avbuffer_in, my_read, NULL, NULL); 
    if (!avio_in)
    {
        printf( "avio_alloc_context for input failed\n");
        ret = AVERROR_UNKNOWN;
        goto end;
    }
    // 分配输入的AVFormatContext
    ifmt_ctx=avformat_alloc_context();
    if (!ifmt_ctx)
    {
        printf( "Could not create output context\n");
        ret = AVERROR_UNKNOWN;
        goto end;
    }
    ifmt_ctx->pb=avio_in; // 赋值自定义的IO结构体
    ifmt_ctx->flags=AVFMT_FLAG_CUSTOM_IO; // 指定为自定义
    if ((ret = avformat_open_input(&ifmt_ctx, "wtf", NULL, NULL)) < 0)
    {
        printf("Cannot open input file\n");
        return ret;
    }
    if ((ret = avformat_find_stream_info(ifmt_ctx, NULL)) < 0)
    {
        printf("Cannot find stream information\n");
        return ret;
    }

对于avio_alloc_context的赋值和输出一样,只是没有了write和seek。对于输入所用的AVFormatContext变量,用avformat_alloc_context来分配。由于是读内存的数据,因此avformat_open_input就不用指定文件名了。

我在代码中尽量加了注释,下面是代码:

[cpp]  view plain  copy
 
    1. /** 
    2. 他山之石,学习为主,版权所无,翻版不究,有错无责 
    3.                   Late Lee  2015.08 
    4. 基于内存的格式封装测试(从内存视频转换到另一片内存视频) 
    5. 使用 
    6. ./a.out a.avi a.mkv 
    7.  
    8. 支持的: 
    9. avi mkv mp4 flv ts ... 
    10.  
    11. 参考: 
    12. http://blog.csdn.net/leixiaohua1020/article/details/25422685 
    13.  
    14. log 
    15. 新版本出现: 
    16. Using AVStream.codec.time_base as a timebase hint to the muxer is  
    17. deprecated. Set AVStream.time_base instead. 
    18.  
    19. test passed!! 
    20.  
    21. mp4->avi failed 
    22. 出现: 
    23. H.264 bitstream malformed, no startcode found, use the h264_mp4toannexb bitstream filter  
    24. 解决见: 
    25. http://blog.chinaunix.net/uid-11344913-id-4432752.html 
    26. 官方解释: 
    27. https://www.ffmpeg.org/ffmpeg-bitstream-filters.html#h264_005fmp4toannexb 
    28.  
    29.  
    30. ts -> avi passed 
    31.  
    32. 其它: 
    33.  
    34. 1、传递给ffmpeg的avio_alloc_context中的内存p和大小size,可以使用32768。 
    35. 如果转换后的数据保存在内存p1,这个内存p1一定要和前面所说的p不同。因为 
    36. 在自定义的write中的buf参数,就是p,所以要拷贝到其它内存。 
    37. 如定义p为32768,但定义p1为50MB,可以转换50MB的视频 
    38. 测试: 
    39. p为32768时,需调用write 1351次 
    40. 2倍大小时,调用write 679次 
    41. p越大,调用次数最少,内存消耗越大 
    42. (用time测试,时间上没什么变化,前者为4.7s,后者为4.6s,但理论上内存大一点好) 
    43.  
    44. 2、优化: 
    45.    转换功能接口封装为类,把write、seek等和内存有关的操作放到类外部实现, 
    46.    再传递到该类中,该类没有内存管理更好一些。 
    47.  
    48. todo 
    49.    重复读文件,如何做? 
    50. */  
    51.   
    52. #include <stdio.h>  
    53. #include <stdlib.h>  
    54. #include <unistd.h>  
    55.   
    56. #include <sys/types.h>  
    57. #include <sys/stat.h>  
    58. #include <fcntl.h>  
    59.   
    60. extern "C" {  
    61. #include "libavcodec/avcodec.h"  
    62. #include "libavformat/avformat.h"  
    63. #include "libswscale/swscale.h"  
    64. }  
    65.   
    66. #include "file_utils.h"  
    67.   
    68. #ifndef min  
    69. #define min(a,b) ((a) > (b) ? (b) : (a))  
    70. #endif  
    71.   
    72. #define _LL_DEBUG_  
    73.   
    74. // low level debug  
    75. #ifdef _LL_DEBUG_  
    76.     #define debug(fmt, ...) printf(fmt, ##__VA_ARGS__)  
    77.     #define LL_DEBUG(fmt, ...) printf("[DEBUG %s().%d @ %s]: " fmt, \  
    78.     __func__, __LINE__, P_SRC, ##__VA_ARGS__)  
    79. #else  
    80.     #define debug(fmt, ...)  
    81.     #define LL_DEBUG(fmt, ...)  
    82. #endif  
    83.   
    84. #define DEFAULT_MEM (10*1024*1024)  
    85.   
    86. //参考file协议的内存,使用大小32768,大一点也可以  
    87. #define IO_BUFFER_SIZE (32768*1)  
    88.   
    89. typedef struct AVIOBufferContext {  
    90.     unsigned char* ptr;  
    91.     int pos;  
    92.     int totalSize;  
    93.     int realSize;  
    94. }AVIOBufferContext;  
    95.   
    96. // note 这两个是用户视频数据,  
    97. // g_avbuffer_in为已经读取的视频  
    98. // g_avbuffer_out是ffmpeg转换后的视频,直接将该内存写入文件即可  
    99. AVIOBufferContext g_avbuffer_in;  
    100. AVIOBufferContext g_avbuffer_out;  
    101.   
    102. // note这两个是FFMPEG内部使用的IO内存,与AVIOBufferContext的ptr不同  
    103. // 在测试时,发现直接定义为数组,会有错误,故使用malloc  
    104. static char *g_ptr_in = NULL;  
    105. static char *g_ptr_out = NULL;  
    106.   
    107. // 每次read_frame时,就会调用到这个函数,该函数从g_avbuffer_in读数据  
    108. static int my_read(void *opaque, unsigned char *buf, int size)  
    109. {  
    110.     AVIOBufferContext* op = (AVIOBufferContext*)opaque;  
    111.     int len = size;  
    112.     if (op->pos + size > op->totalSize)  
    113.     {  
    114.         len = op->totalSize - op->pos;  
    115.     }  
    116.     memcpy(buf, op->ptr + op->pos, len);  
    117.     if (op->pos + len >= op->realSize)  
    118.     op->realSize += len;  
    119.       
    120.     op->pos += len;  
    121.   
    122.     return len;  
    123. }  
    124.   
    125. static int my_write(void *opaque, unsigned char *buf, int size)  
    126. {  
    127.     AVIOBufferContext* op = (AVIOBufferContext*)opaque;  
    128.     if (op->pos + size > op->totalSize)  
    129.     {  
    130.         // 重新申请  
    131.         // 根据数值逐步加大  
    132.         int newTotalLen = op->totalSize*sizeof(char) * 3 / 2;  
    133.         unsigned char* ptr = (unsigned char*)av_realloc(op->ptr, newTotalLen);  
    134.         if (ptr == NULL)  
    135.         {  
    136.             // todo 是否在此处释放内存?  
    137.             return -1;  
    138.         }  
    139.         debug("org ptr: %p new ptr: %p size: %d(%0.fMB) ", op->ptr, ptr,   
    140.                     newTotalLen, newTotalLen/1024.0/1024.0);  
    141.         op->totalSize = newTotalLen;  
    142.         op->ptr = ptr;  
    143.         debug(" realloc!!!!!!!!!!!!!!!!!!!!!!!\n");  
    144.     }  
    145.     memcpy(op->ptr + op->pos, buf, size);  
    146.   
    147.     if (op->pos + size >= op->realSize)  
    148.         op->realSize += size;  
    149.   
    150.     //static int cnt = 1;  
    151.     //debug("%d write %p %p pos: %d len: %d\n", cnt++, op->ptr, buf, op->pos, size);  
    152.       
    153.     op->pos += size;  
    154.   
    155.     return 0;  
    156. }  
    157.   
    158. static int64_t my_seek(void *opaque, int64_t offset, int whence)  
    159. {  
    160.     AVIOBufferContext* op = (AVIOBufferContext*)opaque;  
    161.     int64_t new_pos = 0; // 可以为负数  
    162.     int64_t fake_pos = 0;  
    163.   
    164.     switch (whence)  
    165.     {  
    166.         case SEEK_SET:  
    167.             new_pos = offset;  
    168.             break;  
    169.         case SEEK_CUR:  
    170.             new_pos = op->pos + offset;  
    171.             break;  
    172.         case SEEK_END: // 此处可能有问题  
    173.             new_pos = op->totalSize + offset;  
    174.             break;  
    175.         default:  
    176.             return -1;  
    177.     }  
    178.       
    179.     fake_pos = min(new_pos, op->totalSize);  
    180.     if (fake_pos != op->pos)  
    181.     {  
    182.         op->pos = fake_pos;  
    183.     }  
    184.     //debug("seek pos: %d(%d)\n", offset, op->pos);  
    185.     return new_pos;  
    186. }  
    187.   
    188. int remuxer_mem_read(int argc, char* argv[])  
    189. {  
    190.     //输入对应一个AVFormatContext,输出对应一个AVFormatContext  
    191.     AVFormatContext *ifmt_ctx = NULL, *ofmt_ctx = NULL;  
    192.     AVIOContext *avio_in = NULL, *avio_out = NULL;  
    193.     const char *in_filename = NULL, *out_filename = NULL;  
    194.     AVPacket pkt;  
    195.       
    196.     int ret = 0;  
    197.   
    198.     if (argc < 3)  
    199.     {  
    200.         printf("usage: %s [input file] [output file]\n", argv[0]);  
    201.         printf("eg %s foo.avi bar.ts\n", argv[0]);  
    202.         return -1;  
    203.     }  
    204.   
    205.     in_filename  = argv[1];  
    206.     out_filename = argv[2];  
    207.   
    208.     memset(&g_avbuffer_in, '\0', sizeof(AVIOBufferContext));  
    209.     memset(&g_avbuffer_out, '\0', sizeof(AVIOBufferContext));  
    210.   
    211.     read_file(in_filename, (char**)&g_avbuffer_in.ptr, &g_avbuffer_in.totalSize);  
    212.       
    213.     // 分配输出视频数据空间  
    214.     g_avbuffer_out.ptr = (unsigned char*)av_realloc(NULL, DEFAULT_MEM*sizeof(char));  // new  
    215.     if (g_avbuffer_out.ptr == NULL)  
    216.     {  
    217.         debug("alloc output mem failed.\n");  
    218.         return -1;  
    219.     }  
    220.     g_avbuffer_out.totalSize = DEFAULT_MEM;  
    221.     memset(g_avbuffer_out.ptr, '\0', g_avbuffer_out.totalSize);  
    222.       
    223.     g_ptr_in = (char*)malloc(IO_BUFFER_SIZE*sizeof(char));  
    224.     g_ptr_out = (char*)malloc(IO_BUFFER_SIZE*sizeof(char));  
    225.   
    226.     // 初始化  
    227.     av_register_all();  
    228.   
    229.     // 输出相关  
    230.     // note 要指定IO内存,还在指定自定义的操作函数,这里有write和seek  
    231.     avio_out =avio_alloc_context((unsigned char *)g_ptr_out, IO_BUFFER_SIZE, 1,  
    232.                 &g_avbuffer_out, NULL, my_write, my_seek);   
    233.     if (!avio_out)  
    234.     {  
    235.         printf( "avio_alloc_context failed\n");  
    236.         ret = AVERROR_UNKNOWN;  
    237.         goto end;  
    238.     }  
    239.     // 分配AVFormatContext  
    240.     // 为方便起见,使用out_filename来根据输出文件扩展名来判断格式  
    241.     // 如果要使用如“avi”、“mp4”等指定,赋值给第3个参数即可  
    242.     // 注意该函数会分配AVOutputFormat  
    243.     avformat_alloc_output_context2(&ofmt_ctx, NULL, NULL, out_filename);  
    244.     if (!ofmt_ctx)  
    245.     {  
    246.         printf( "Could not create output context\n");  
    247.         ret = AVERROR_UNKNOWN;  
    248.         goto end;  
    249.     }  
    250.     ofmt_ctx->pb=avio_out; // 赋值自定义的IO结构体  
    251.     ofmt_ctx->flags=AVFMT_FLAG_CUSTOM_IO; // 指定为自定义  
    252.   
    253.     debug("guess format: %s(%s) flag: %d\n", ofmt_ctx->oformat->name,   
    254.             ofmt_ctx->oformat->long_name, ofmt_ctx->oformat->flags);  
    255.   
    256.   
    257.     //  输入相关  
    258.     // 分配自定义的AVIOContext 要区别于输出的buffer  
    259.     // 由于数据已经在内存中,所以指定read即可,不用write和seek  
    260.     avio_in =avio_alloc_context((unsigned char *)g_ptr_in, IO_BUFFER_SIZE, 0,  
    261.                 &g_avbuffer_in, my_read, NULL, NULL);   
    262.     if (!avio_in)  
    263.     {  
    264.         printf( "avio_alloc_context for input failed\n");  
    265.         ret = AVERROR_UNKNOWN;  
    266.         goto end;  
    267.     }  
    268.     // 分配输入的AVFormatContext  
    269.     ifmt_ctx=avformat_alloc_context();  
    270.     if (!ifmt_ctx)  
    271.     {  
    272.         printf( "Could not create output context\n");  
    273.         ret = AVERROR_UNKNOWN;  
    274.         goto end;  
    275.     }  
    276.     ifmt_ctx->pb=avio_in; // 赋值自定义的IO结构体  
    277.     ifmt_ctx->flags=AVFMT_FLAG_CUSTOM_IO; // 指定为自定义  
    278.   
    279.     // 注:第二个参数本来是文件名,但基于内存,不再有意义,随便用字符串  
    280.     if ((ret = avformat_open_input(&ifmt_ctx, "wtf", NULL, NULL)) < 0)  
    281.     {  
    282.         printf("Cannot open input file\n");  
    283.         return ret;  
    284.     }  
    285.     if ((ret = avformat_find_stream_info(ifmt_ctx, NULL)) < 0)  
    286.     {  
    287.         printf("Cannot find stream information\n");  
    288.         return ret;  
    289.     }  
    290.   
    291.     // 复制所有的stream  
    292.     for (int i = 0; i < (int)(ifmt_ctx->nb_streams); i++)  
    293.     {  
    294.         //根据输入流创建输出流  
    295.         AVStream *in_stream = ifmt_ctx->streams[i];  
    296.         AVStream *out_stream = avformat_new_stream(ofmt_ctx, in_stream->codec->codec);  
    297.         if (!out_stream)  
    298.         {  
    299.             printf( "Failed allocating output stream\n");  
    300.             ret = AVERROR_UNKNOWN;  
    301.             goto end;  
    302.         }  
    303.         //复制AVCodecContext的设置  
    304.         ret = avcodec_copy_context(out_stream->codec, in_stream->codec);  
    305.         if (ret < 0)  
    306.         {  
    307.             printf( "Failed to copy context from input to output stream codec context\n");  
    308.             goto end;  
    309.         }  
    310.         out_stream->codec->codec_tag = 0;  
    311.         if (ofmt_ctx->oformat->flags & AVFMT_GLOBALHEADER)  
    312.             out_stream->codec->flags |= CODEC_FLAG_GLOBAL_HEADER;  
    313.     }  
    314.     //输出一下格式------------------  
    315.     printf("output format:\n");  
    316.     av_dump_format(ofmt_ctx, 0, out_filename, 1);  
    317.   
    318.     // 写文件头  
    319.     ret = avformat_write_header(ofmt_ctx, NULL);  
    320.     if (ret < 0)  
    321.     {  
    322.         printf( "Error occurred when opening output file\n");  
    323.         goto end;  
    324.     }  
    325.   
    326.     // 帧  
    327.     while (1)  
    328.     {  
    329.         AVStream *in_stream, *out_stream;  
    330.         //获取一个AVPacket  
    331.         ret = av_read_frame(ifmt_ctx, &pkt);  
    332.         if (ret < 0)  
    333.         {  
    334.             printf("av_read_frame failed or end of stream.\n");  
    335.             break;  
    336.         }  
    337.         in_stream  = ifmt_ctx->streams[pkt.stream_index];  
    338.         out_stream = ofmt_ctx->streams[pkt.stream_index];  
    339.   
    340.         //转换PTS/DTS  
    341.         pkt.pts = av_rescale_q_rnd(pkt.pts, in_stream->time_base,   
    342.             out_stream->time_base, (AVRounding)(AV_ROUND_NEAR_INF|AV_ROUND_PASS_MINMAX));  
    343.         pkt.dts = av_rescale_q_rnd(pkt.dts, in_stream->time_base,  
    344.             out_stream->time_base, (AVRounding)(AV_ROUND_NEAR_INF|AV_ROUND_PASS_MINMAX));  
    345.         pkt.duration = av_rescale_q(pkt.duration, in_stream->time_base, out_stream->time_base);  
    346.         pkt.pos = -1;  
    347.   
    348.         // 写入一帧  
    349.         ret = av_interleaved_write_frame(ofmt_ctx, &pkt);  
    350.         if (ret < 0) {  
    351.             printf( "Error muxing packet\n");  
    352.             break;  
    353.         }  
    354.         av_free_packet(&pkt);  
    355.     }  
    356.   
    357.     //写文件尾(Write file trailer)  
    358.     printf("--------write trailer------------\n");  
    359.     av_write_trailer(ofmt_ctx);  
    360.   
    361.     // 把输出的视频写到文件中  
    362.     printf("write to file: %s %p %d\n", out_filename, g_avbuffer_out.ptr, g_avbuffer_out.realSize);  
    363.     write_file(out_filename, (char*)g_avbuffer_out.ptr, g_avbuffer_out.realSize, 1);  
    364.   
    365. end:  
    366.     if (avio_in != NULL)  av_freep(avio_in);   
    367.     if (avio_out != NULL) av_freep(avio_out);  
    368.     //if (g_ptr_in != NULL) free(g_ptr_in);  
    369.     if (g_ptr_out != NULL) free(g_ptr_out);  
    370.   
    371.     // 该函数会释放用户自定义的IO buffer,上面不再释放,否则会corrupted double-linked list  
    372.     avformat_close_input(&ifmt_ctx);  
    373.     avformat_free_context(ofmt_ctx);  
    374.   
    375.     if (g_avbuffer_in.ptr != NULL) free(g_avbuffer_in.ptr);  
    376.     if (g_avbuffer_out.ptr != NULL) free(g_avbuffer_out.ptr);  
    377.   
    378.     return ret;  
    379. }  

from:http://blog.csdn.net/subfate/article/details/48001433

目录
相关文章
|
5月前
|
编解码 NoSQL Java
使用Spring Boot + Redis 队列实现视频文件上传及FFmpeg转码的技术分享
【8月更文挑战第30天】在当前的互联网应用中,视频内容的处理与分发已成为不可或缺的一部分。对于视频平台而言,高效、稳定地处理用户上传的视频文件,并对其进行转码以适应不同设备的播放需求,是提升用户体验的关键。本文将围绕使用Spring Boot结合Redis队列技术来实现视频文件上传及FFmpeg转码的过程,分享一系列技术干货。
263 3
|
10天前
|
存储 缓存 资源调度
阿里云服务器经济型、通用算力型、计算型、通用型、内存型实例区别与选择指南
在我们通过阿里云的活动选购云服务器的时候会发现,相同配置的云服务器往往有多个不同的实例可选,而且价格差别也比较大,这会是因为不同实例规格的由于采用的处理器不同,底层架构也有所不同(例如X86 计算架构与Arm 计算架构),因此不同实例的云服务器其性能与适用场景是有所不同。本文将详细解析阿里云的经济型、通用算力型、计算型、通用型和内存型实例的性能特点及适用场景,帮助用户根据自己的业务需求做出明智的选择。
|
6月前
|
存储 安全 数据库
阿里云服务器计算型、通用型、内存型主要实例规格性能特点和适用场景汇总
阿里云服务器ECS计算型、通用型、内存型规格族属于独享型云服务器,在高负载不会出现计算资源争夺现象,因为每一个vCPU都对应一个Intel ® Xeon ®处理器核心的超线程,具有性能稳定且资源独享的特点。本文为大家整理汇总了阿里云服务器ECS计算型、通用型、内存型主要实例规格族具体实例规格有哪些,各个实例规格的性能特点和主要适用场景。
阿里云服务器计算型、通用型、内存型主要实例规格性能特点和适用场景汇总
|
1月前
|
存储 分布式计算 安全
阿里云服务器经济型e、通用算力型u1、计算型c8i、通用型g8i、内存型r8i实例介绍与选择参考
在阿里云现在的活动中,可选的云服务器实例规格主要有经济型e、通用算力型u1、计算型c8i、通用型g8i、内存型r8i实例,虽然阿里云在活动中提供了多种不同规格的云服务器实例,以满足不同用户和应用场景的需求。但是有的用户并不清楚他们的性能如何,应该如何选择。本文将详细介绍阿里云服务器中的经济型e、通用算力型u1、计算型c8i、通用型g8i、内存型r8i实例的性能、适用场景及选择参考,帮助用户根据自身需求做出合适的选择。
|
5月前
|
编解码 Linux
CentOS安装ffmpeg并转码视频为mp4
CentOS安装ffmpeg并转码视频为mp4
168 0
|
2月前
|
编解码 监控 网络协议
如何使用FFmpeg实现RTSP推送H.264和H.265(HEVC)编码视频
本文详细介绍了如何使用FFmpeg实现RTSP推送H.264和H.265(HEVC)编码视频。内容涵盖环境搭建、编码配置、服务器端与客户端实现等方面,适合视频监控系统和直播平台等应用场景。通过具体命令和示例代码,帮助读者快速上手并实现目标。
388 6
|
2月前
|
存储 缓存 安全
阿里云服务器内存型r7、r8a、r8y、r8i实例区别及选择参考
随着阿里云2024年金秋云创季的开始,目前在阿里云的活动中,属于内存型实例规格的云服务器有内存型r7、内存型r8a、内存型r8y和内存型r8i这几个实例规格,相比于活动内的经济型e和通用算力型u1等实例规格来说,这些实例规格等性能更强,虽然这几个实例规格的云服务器通常处理器与内存的配比为都是1:8,但是他们在处理器、存储、网络、安全等方面等性能并不是一样的,所以他们的适用场景也有着不同。本文为大家介绍内存型r7、r8a、r8y、r8i实例的性能、适用场景的区别以及选择参考。
|
3月前
|
Java 数据安全/隐私保护
Java ffmpeg 实现视频加文字/图片水印功能
【10月更文挑战第22天】在 Java 中使用 FFmpeg 实现视频加文字或图片水印功能,需先安装 FFmpeg 并添加依赖(如 JavaCV)。通过构建 FFmpeg 命令行参数,使用 `drawtext` 滤镜添加文字水印,或使用 `overlay` 滤镜添加图片水印。示例代码展示了如何使用 JavaCV 实现文字水印。
210 1
|
4月前
|
分布式计算 大数据 数据挖掘
阿里云服务器计算型c8i、通用型g8i、内存型r8i实例测评与价格参考
阿里云服务器计算型c8i、通用型g8i、内存型r8i实例是阿里云的第八代云服务器实例规格,是除了计算型c7和c8y、通用型g7与g8y、内存型r7与r8y之外同样深受用户喜欢的云服务器实例规格。本文将详细介绍阿里云第八代云服务器中的计算型c8i、通用型g8i、以及内存型r8i实例,包括它们的技术特性、适用场景以及最新的活动价格信息。
阿里云服务器计算型c8i、通用型g8i、内存型r8i实例测评与价格参考
|
3月前
|
计算机视觉 Python
FFMPEG学习笔记(一): 提取视频的纯音频及无声视频
本文介绍了如何使用FFmpeg工具从视频中提取纯音频和无声视频。提供了具体的命令行操作,例如使用`ffmpeg -i input.mp4 -vn -c:a libmp3lame output.mp3`来提取音频,以及`ffmpeg -i input.mp4 -c:v copy -an output.mp4`来提取无声视频。此外,还包含了一个Python脚本,用于批量处理视频文件,自动提取音频和生成无声视频。
105 1