FFmpeg编译4

news2024/11/26 18:29:42

CPU=x86-64
TOOLCHAIN= N D K / t o o l c h a i n s / x 8 6 6 4 − 4.9 / p r e b u i l t / l i n u x − x 8 6 6 4 S Y S R O O T = NDK/toolchains/x86_64-4.9/prebuilt/linux-x86_64 SYSROOT= NDK/toolchains/x86644.9/prebuilt/linuxx8664SYSROOT=NDK/platforms/android-21/arch- A R C H / C R O S S P R E F I X = ARCH/ CROSS_PREFIX= ARCH/CROSSPREFIX=TOOLCHAIN/bin/x86_64-linux-android-
PREFIX= ( p w d ) / a n d r o i d / (pwd)/android/ (pwd)/android/CPU
OPTIMIZE_CFLAGS=“-march=$CPU -msse4.2 -mpopcnt -m64 -mtune=intel”
build_android

注意:当执行./build_android.sh的时候,出现权限不足,那么一定要给这个文件添加可执行权限

chmod 777 build_android.sh

这样就可以进行编译生成Android下的ffmpeg

如果在编译的时候出现如下问题:

ffbuild/common.mak:60: recipe for target ‘libavformat/udp.o’ failed
make: *** [libavformat/udp.o] Error 1

libavformat/udp.c 第290~295行进行注释

// mreqs.imr_multiaddr.s_addr = ((struct sockaddr_in *)addr)->sin_addr.s_addr;
// if (local_addr)
// mreqs.imr_interface= ((struct sockaddr_in *)local_addr)->sin_addr;
// else
// mreqs.imr_interface.s_addr= INADDR_ANY;
// mreqs.imr_sourceaddr.s_addr = ((struct sockaddr_in *)&sources[i])->sin_addr.s_addr;

这样之后,基本就是ok的,在我们ffmpeg的文件夹下,会生成一个ffmpeg/android/armv7-a的文件目录,里面内容如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

将编译后的ffmpeg移植到Android

准备

  • 在任何地方新建一个jni的文件夹。
  • 将我们刚刚编译的ffmpeg/android/armv7-a/include下的所有文件拷贝进入jni文件夹。
  • jni下新建一个prebuilt的文件夹,将ffmpeg/android/armv7-a/lib下的so文件全部拷贝到prebuilt之下。
  • 然后将ffmpeg/fftools文件下的如下文件拷贝进入jni
  • cmdutils.c
  • cmdutils.h
  • config.h
  • ffmpeg_filter.c
  • ffmpeg_hw.c
  • ffmpeg_opt.c
  • ffmpeg.c
  • ffmpeg.h

修改ffmpeg文件

  • 修改刚刚拷贝的ffmpeg.c文件,找到int main(int argc, char **argv)函数,将其替换为int run(int argc, char **argv)
  • 在修改后的run(int argc, char **argv) 末尾(retrun 之前)加上如上如下代码:

nb_filtergraphs = 0;
progress_avio = NULL;

input_streams = NULL;
nb_input_streams = 0;
input_files = NULL;
nb_input_files = 0;

output_streams = NULL;
nb_output_stream​
s = 0;
output_files = NULL;
nb_output_files = 0;

  • 并在ffmpeg.h文件末尾加上

int run(int argc, char **argv);

  • 再次打开ffmpeg.c文件,注释掉run(int argc, char **argv)下的所有exit_program函数,大致代码如下:

int run(int argc, char **argv)
{
int i, ret;
BenchmarkTimeStamps ti;

init_dynload();

register_exit(ffmpeg_cleanup);

setvbuf(stderr,NULL,_IONBF,0); /* win32 runtime needs this */

av_log_set_flags(AV_LOG_SKIP_REPEATED);
parse_loglevel(argc, argv, options);

if(argc>1 && !strcmp(argv[1], “-d”)){
run_as_daemon=1;
av_log_set_callback(log_callback_null);
argc–;
argv++;
}

#if CONFIG_AVDEVICE
avdevice_register_all();
#endif
avformat_network_init();

show_banner(argc, argv, options);

/* parse options and open all input/output files */
ret = ffmpeg_parse_options(argc, argv);
if (ret < 0);
// exit_program(1);

if (nb_output_files <= 0 && nb_input_files == 0) {
show_usage();
av_log(NULL, AV_LOG_WARNING, “Use -h to get full help or, even better, run ‘man %s’\n”, program_name);
// exit_program(1);
}

/* file converter / grab */
if (nb_output_files <= 0) {
av_log(NULL, AV_LOG_FATAL, “At least one output file must be specified\n”);
// exit_program(1);
}

// if (nb_input_files == 0) {
// av_log(NULL, AV_LOG_FATAL, “At least one input file must be specified\n”);
// exit_program(1);
// }

for (i = 0; i < nb_output_files; i++) {
if (strcmp(output_files[i]->ctx->oformat->name, “rtp”))
want_sdp = 0;
}

current_time = ti = get_benchmark_time_stamps();
if (transcode() < 0);
// exit_program(1);
if (do_benchmark) {
int64_t utime, stime, rtime;
current_time = get_benchmark_time_stamps();
utime = current_time.user_usec - ti.user_usec;
stime = current_time.sys_usec - ti.sys_usec;
rtime = current_time.real_usec - ti.real_usec;
av_log(NULL, AV_LOG_INFO,
“bench: utime=%0.3fs stime=%0.3fs rtime=%0.3fs\n”,
utime / 1000000.0, stime / 1000000.0, rtime / 1000000.0);
}
av_log(NULL, AV_LOG_DEBUG, “%“PRIu64” frames successfully decoded, %“PRIu64” decoding errors\n”,
decode_error_stat[0], decode_error_stat[1]);
if ((decode_error_stat[0] + decode_error_stat[1]) * max_error_rate < decode_error_stat[1]);
// exit_program(69);

// exit_program(received_nb_signals ? 255 : main_return_code);

nb_filtergraphs = 0;
progress_avio = NULL;

input_streams = NULL;
nb_input_streams = 0;
input_files = NULL;
nb_input_files = 0;

output_streams = NULL;
nb_output_streams = 0;
output_files = NULL;
nb_output_files = 0;

return main_return_code;
}

新建ffmpeg-invoke

jni文件下新建一个ffmpeg-invoke.cpp的c++文件,将如下代码写入

#include <jni.h>
#include <string.h>
#include “android/log.h”

#define LOGD(…) __android_log_print(ANDROID_LOG_DEBUG, “ffmpeg-invoke”, VA_ARGS)

extern “C”{
#include “ffmpeg.h”
#include “libavcodec/jni.h”
}

extern “C”
JNIEXPORT jint JNICALL
Java_com_coder_ffmpeg_jni_FFmpegCmd_run(JNIEnv *env, jclass type, jint cmdLen,
jobjectArray cmd) {
//set java vm
JavaVM *jvm = NULL;
env->GetJavaVM(&jvm);
av_jni_set_java_vm(jvm, NULL);

char *argCmd[cmdLen] ;
jstring buf[cmdLen];

for (int i = 0; i < cmdLen; ++i) {
buf[i] = static_cast(env->GetObjectArrayElement(cmd, i));
char *string = const_cast<char *>(env->GetStringUTFChars(buf[i], JNI_FALSE));
argCmd[i] = string;
LOGD(“argCmd=%s”,argCmd[i]);
}

int retCode = run(cmdLen, argCmd);
LOGD(“ffmpeg-invoke: retCode=%d”,retCode);

return retCode;

}

需注意的是 Java_com_coder_ffmpeg_jni_FFmpegCmd_runcom_coder_ffmpeg_jni表示FFmpegCmd所在你Android项目的位置。

新建Android.mk

jni文件夹下,新建Android.mk的文件,将如下代码拷贝,但是需要注意的是将 LOCAL_C_INCLUDES路径替换成你源码所在位置。

// 将此处的路径改为你ffmepg源码所在位置
LOCAL_C_INCLUDES := /home/anjoiner/Documents/AnJoiner/ffmpeg

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)
LOCAL_MODULE := libavdevice
LOCAL_SRC_FILES := prebuilt/libavdevice.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE := libavutil
LOCAL_SRC_FILES := prebuilt/libavutil.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE := libswresample
LOCAL_SRC_FILES := prebuilt/libswresample.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE := libswscale
LOCAL_SRC_FILES := prebuilt/libswscale.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE := libavcodec
LOCAL_SRC_FILES := prebuilt/libavcodec.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE := libavformat
LOCAL_SRC_FILES := prebuilt/libavformat.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE := libavfilter
LOCAL_SRC_FILES := prebuilt/libavfilter.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE := libpostproc
LOCAL_SRC_FILES := prebuilt/libpostproc.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE := ffmpeg-invoke

LOCAL_SRC_FILES :=ffmpeg-invoke.cpp
cmdutils.c
ffmpeg_filter.c
ffmpeg_opt.c
ffmpeg_hw.c
ffmpeg.c

// 将此处的路径改为你ffmepg源码所在位置
LOCAL_C_INCLUDES := /home/anjoiner/Documents/AnJoiner/ffmpeg

LOCAL_LDLIBS := -llog -ljnigraphics -lz -landroid -lm -pthread -L$(SYSROOT)/usr/lib
LOCAL_SHARED_LIBRARIES := libavdevice libavcodec libavfilter libavformat libavutil libswresample libswscale libpostproc

include $(BUILD_SHARED_LIBRARY)

新建Application.mk

jni文件夹下,新建Application.mk并将如下代码拷贝进入

APP_ABI := armeabi-v7a
APP_PLATFORM := android-21
APP_OPTIM := release
APP_STL := stlport_static

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

到此基本上的配置已经完成,然后在jni文件下运行ndk-build

/home/anjoiner/Documents/AnJoiner/build/jni# /home/anjoiner/Documents/AnJoiner/ffmpeg/ndk/ndk-build

当出现如下代码就表示成功了

[armeabi-v7a] SharedLibrary : libffmpeg-invoke.so
[armeabi-v7a] Install : libffmpeg-invoke.so => libs/armeabi-v7a/libffmpeg-invoke.so
[armeabi-v7a] Install : libpostproc.so => libs/armeabi-v7a/libpostproc.so
[armeabi-v7a] Install : libswresample.so => libs/armeabi-v7a/libswresample.so
[armeabi-v7a] Install : libswscale.so => libs/armeabi-v7a/libswscale.so

测试

  • 新建一个支持C++项目,注意包名需是上述ava_com_coder_ffmpeg_jni_FFmpegCmd_run中的com.coder.ffmpeg
  • jni的同级目录下,会生成一个libs的目录,将这个目录下的armeabi-v7a文件拷贝到你所在的Android项目中的app/src/main/jniLibs
  • 新建一个jni的文件目录,在此目录下新建一个FFmpegCmd

/**

  • @author: AnJoiner
  • @datetime: 19-7-30
    */
    public class FFmpegCmd {
    static {
    System.loadLibrary(“avdevice”);
    System.loadLibrary(“avutil”);
    System.loadLibrary(“avcodec”);
    System.loadLibrary(“swresample”);
    System.loadLibrary(“avformat”);
    System.loadLibrary(“swscale”);
    System.loadLibrary(“avfilter”);
    System.loadLibrary(“postproc”);
    System.loadLibrary(“ffmpeg-invoke”);
    }

private static native int run(int cmdLen, String[] cmd);

public static int runCmd(String[] cmd){
return run(cmd.length,cmd);
}
}

拷贝进来run方法名会出现红色,不用管他,

  • 在我们的MainActivity中进行调用,此方法是将音频进行剪切

public class MainActivity extends AppCompatActivity {

// Used to load the ‘native-lib’ library on application startup.
static {
System.loadLibrary(“native-lib”);
}

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
if (ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE)
!= PackageManager.PERMISSION_GRANTED) {
ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
100);
}

// Example of a call to a native method
TextView tv = findViewById(R.id.sample_text);
tv.setText(stringFromJNI());

findViewById(R.id.btn_test).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
ffmpegTest();
}
});
}

/**

  • A native method that is implemented by the ‘native-lib’ native library,
  • which is packaged with this application.
    */
    public native String stringFromJNI();

private void ffmpegTest() {
new Thread() {
@Override
public void run() {
long startTime = System.currentTimeMillis();
String input =
Environment.getExternalStorageDirectory().getPath() + File.separator +
“DCIM” + File.separator + “test.mp3”;
String output =
Environment.getExternalStorageDirectory().getPath() + File.separator +
“DCIM” + File.separator + “output.mp3”;

String cmd = “ffmpeg -y -i %s -vn -acodec copy -ss %s -t %s %s”;
String result = String.format(cmd, input, “00:00:30”, “00:00:40”, output);
FFmpegCmd.runCmd(result.split(" "));
Log.d(“FFmpegTest”, “run: 耗时:” + (System.currentTimeMillis() - startTime));
}
}.start();

}

如果上述步骤不出错,基本没什么问题,在资料的参考下,我也弄了好几天,但幸好结果还是好的~

使用 NDK-r20 编译FFmpeg

前面使用过老版本NDK-r14b编译FFmpeg,其实质是通过gcc去进行编译,但是最新的NDK版本中已经不使用gcc去编译,而是使用clang去进行编译.这里贴上最新配置文件代码

  • 下载最新版本NDK-r20
  • 下载FFmpeg-4.2.1

#!/bin/bash

清空上次的编译

make clean
#你自己的NDK路径.
export NDK=/home/anjoiner/Android/Sdk/ndk-bundle
TOOLCHAIN=$NDK/toolchains/llvm/prebuilt/linux-x86_64/
API=29

function build_android
{
echo “Compiling FFmpeg for C P U " . / c o n f i g u r e   − − p r e f i x = CPU" ./configure \ --prefix= CPU"./configure prefix=PREFIX
–enable-neon
–enable-hwaccels
–enable-gpl
–enable-postproc
–enable-shared
–enable-jni
–enable-mediacodec
–enable-decoder=h264_mediacodec
–disable-static
–disable-doc
–enable-ffmpeg
–disable-ffplay
–disable-ffprobe
–enable-avdevice
–disable-doc
–disable-symver
–cross-prefix= C R O S S P R E F I X   − − t a r g e t − o s = a n d r o i d   − − a r c h = CROSS_PREFIX \ --target-os=android \ --arch= CROSSPREFIX targetos=android arch=ARCH
–cpu= C P U   − − c c = CPU \ --cc= CPU cc=CC
–cxx= C X X − − e n a b l e − c r o s s − c o m p i l e   − − s y s r o o t = CXX --enable-cross-compile \ --sysroot= CXXenablecrosscompile sysroot=SYSROOT
–extra-cflags=”-Os -fpic O P T I M I Z E C F L A G S "   − − e x t r a − l d f l a g s = " OPTIMIZE_CFLAGS" \ --extra-ldflags=" OPTIMIZECFLAGS" extraldflags="ADDI_LDFLAGS"
$ADDITIONAL_CONFIGURE_FLAG
make clean

最后

小编这些年深知大多数初中级Android工程师,想要提升自己,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

资料⬅专栏获取
= C X X − − e n a b l e − c r o s s − c o m p i l e   − − s y s r o o t = CXX --enable-cross-compile \ --sysroot= CXXenablecrosscompile sysroot=SYSROOT
–extra-cflags=“-Os -fpic O P T I M I Z E C F L A G S "   − − e x t r a − l d f l a g s = " OPTIMIZE_CFLAGS" \ --extra-ldflags=" OPTIMIZECFLAGS" extraldflags="ADDI_LDFLAGS”
$ADDITIONAL_CONFIGURE_FLAG
make clean

最后

小编这些年深知大多数初中级Android工程师,想要提升自己,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。

[外链图片转存中…(img-qhQAkwk2-1719088014991)]一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

资料⬅专栏获取

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

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

相关文章

2024-06-23 操作系统实验5——模拟页式存储管理

文章目录 一、实验目的二、实验内容三、实验过程四、结果测试五、实验总结和说明 补录与分享本科实验&#xff0c;以示纪念。 一、实验目的 通过编写和调试请求页式存储管理的模拟程序以加深对请求页式存储管理方案的理解。 二、实验内容 页面淘汰算法可采用FIFO置换算法&a…

React+TS前台项目实战(十五)-- 全局常用组件Table封装

文章目录 前言Table组件1. 功能分析2. 代码详细注释3. 使用方式4. 效果展示 总结 前言 在这篇文章中&#xff0c;我们将对本系列项目中常用的表格组件Table进行自定义封装&#xff0c;以提高性能并适应项目需求。后期也可进行修改和扩展&#xff0c;以满足项目的需求。 Table组…

Windows 11 安装hp 1020 plus 打印机驱动 (Ubuntu 20.04.3 LTS 部署cups局域网共享打印服务器)

1 win11 下载HP laserjet 1020 plus驱动,可以官网下载哦 链接下载 2 手动添加hp laserjet 1020驱动: 控制面板-->查看设备和打印机-->打印机和扫描仪-->添加设备-->我需要的打印机不在列表中-->通过手动添加-->按名称选择共享打印机 如果找不到&#xff0…

论文《Tree Decomposed Graph Neural Network》笔记

【TDGNN】本文提出了一种树分解方法来解决不同层邻域之间的特征平滑问题&#xff0c;增加了网络层配置的灵活性。通过图扩散过程表征了多跳依赖性&#xff08;multi-hop dependency&#xff09;&#xff0c;构建了TDGNN模型&#xff0c;该模型可以灵活地结合大感受场的信息&…

简易部署的设备日志采集工具

永久免费: Gitee下载 最新版本 使用说明: Moretl 企业级采集文件工具 优势: A. 开箱即用. 解压直接运行.不需额外安装. B. 批管理设备. 设备配置均在后台管理. C. 无人值守 客户端自启动,自更新. D. 稳定安全. 架构简单,内存占用小,通过授权访问.

OPENCV中0x00007FFE5F35F39C发生异常

原因&#xff1a;读取图片时已经为灰度图像&#xff0c;又进行了一次灰度处理cvtColor 解决方法&#xff1a;如上图所示&#xff0c;将cv::imread的第二个参数改为cv::IMREAD_COLOR&#xff1b;或者保留cv::IMREAD_GRAYSCALE&#xff0c;删去后面的cv::cvtColor

高通安卓12-OTA 升级

1.OTA介绍 OTA 英文全称是 Over-the-Air Technology&#xff0c;即空间下载技术的意思。 OTA 升级是 Android 系统提供的标准软件升级方式。它功能强大&#xff0c;可以无损失升级系统&#xff0c;主 要通过网络[例如 WIFI、3G]自动下载 OTA 升级包、自动升级&#xff0c;但…

仿饿了么加入购物车旋转控件 - 自带闪转腾挪动画 的按钮

, mWidth - mCircleWidth, mHeight - mCircleWidth); canvas.drawRoundRect(rectF, mHintBgRoundValue, mHintBgRoundValue, mHintPaint); //前景文字 mHintPaint.setColor(mHintFgColor); // 计算Baseline绘制的起点X轴坐标 int baseX (int) (mWidth / 2 - mHintPaint.m…

新闻管理与推荐系统Python+Django+协同过滤推荐算法+管理系统

一、介绍 新闻管理与推荐系统。本系统使用Python作为主要开发语言开发的一个新闻管理与推荐的网站平台。 网站前端界面采用HTML、CSS、BootStrap等技术搭建界面。后端采用Django框架处理用户的逻辑请求&#xff0c;并将用户的相关行为数据保存在数据库中。通过Ajax技术实现前后…

mayavi pyqt 实例

目录 安装&#xff1a; 示例代码&#xff1a; 生成3d检测框&#xff1a; 安装&#xff1a; pip install pyqt5 mayavi traits traitsui 示例代码&#xff1a; import sys from PyQt5.QtWidgets import QApplication, QMainWindow, QVBoxLayout, QWidget, QPushButton from …

区块链中nonce是什么,什么作用

目录 区块链中nonce是什么,什么作用 区块链中nonce是什么,什么作用 Nonce在以太坊中是一个用于确保交易顺序性和唯一性的重要参数。以下是对Nonce的详细解释: 定义 Nonce是一个scalar值,它等于从该地址发送的交易数量,或在具有关联代码的账户的情况下,由该账户创建的合…

【Flutter 专题】112 图解自定义 ACEPieWidget 饼状图 (一)

类别选项球&#xff1b;切割绘制饼状图&#xff1b;饼状图中绘制文字&#xff1b; 1. 类别选项球 对于两侧不同颜色类别选项卡&#xff0c;仅需要简单设置一下 Container 的 decoration 装饰器即可&#xff0c;只是方便用户查看饼状图分类而已&#xff1b; return Container…

不用写一行代码,deepseek结合腾讯云语音识别来批量转录Mp3音频

首先&#xff0c;打开window系统中的cmd命令行工具&#xff0c;或者powershell&#xff0c;安装腾讯云tencentcloud的Python库 pip install -i https://mirrors.tencent.com/pypi/simple/ --upgrade tencentcloud-sdk-python 然后&#xff0c;开通腾讯云的对象存储COS服务&…

【小沐学AI】Python实现语音识别(Whisper-Web)

文章目录 1、简介2、下载2.1 openai-whisper2.2 whisper-web 结语 1、简介 https://openai.com/index/whisper/ Whisper 是一种自动语音识别 &#xff08;ASR&#xff09; 系统&#xff0c;经过 680,000 小时的多语言和多任务监督数据的训练&#xff0c;从网络上收集。我们表…

【大数据 复习】第8章 Hadoop架构再探讨

一、概念 1.Hadoop1.0的核心组件&#xff08;仅指MapReduce和HDFS&#xff0c;不包括Hadoop生态系统内的Pig、Hive、HBase等其他组件&#xff09;&#xff0c;主要存在以下不足&#xff1a; &#xff08;1&#xff09;抽象层次低&#xff0c;需人工编码 &#xff08;2&#xf…

Docker常用命令与实战示例

docker 1. 安装2. 常用命令3. 存储4. 网络5. redis主从复制示例6. wordpress示例7. DockerFile8. 一键安装超多中间件&#xff08;compose&#xff09; 1. 安装 以centOS系统为例 # 移除旧版本docker sudo yum remove docker \docker-client \docker-client-latest \docker-c…

AI时代的音乐革命:创作更简单,灵魂在哪里?

#AI在创造还是毁掉音乐# 我是李涛&#xff0c;一名音乐创作者&#xff0c;最近一直在思考一个问题&#xff1a;AI到底是在创造音乐&#xff0c;还是在毁掉音乐&#xff1f; 几个月前&#xff0c;我第一次接触到AI音乐创作工具。它让我震惊&#xff0c;只需要输入几个关键词&a…

数据结构7---图

一、定义 对于图的定义&#xff0c;我们需要明确几个注意的地方:一线性表中我们把数据元素叫元素&#xff0c;树中叫结点&#xff0c;在途中数据元素我们则称之为顶点(Vertex)。 对于图的定义&#xff0c;我们需要明确几个注意的地方: 线性表中我们把数据元素叫元素&#xf…

实现文件分片合并功能并使用Github Actions自动编译Release

一、编译IOS镜像 1.1 编译 起因是公司电脑使用的Win11 23H2的预览版&#xff0c;这个预览版系统的生命周期只到2024-09-18&#xff0c;到期后就会强制每两小时重启。这是Windows强制升级系统的一种手段。 虽然公司里的台式电脑目前用不到&#xff0c;但是里面还保留许多旧项…

Jenkins定时构建自动化(一):Jenkins下载安装配置

目录 ​编辑 一、jdk下载安装 1. 已下载安装jdk 2. 未下载安装jdk 二、jenkins安装 1. .war包安装 三、获取IP地址 四、jenkins网页配置 一、jdk下载安装 1. 已下载安装jdk &#xff08;1&#xff09;查询jdk版本命令&#xff1a;java -version &#xff08;2&#xff09;…