vue分片上传视频并转换为m3u8文件并播放

开发环境:

基于若依开源框架的前后端分离版本的实践,后端java的springboot,前端若依的vue2,做一个分片上传视频并分段播放的功能,因为是小项目,并没有专门准备文件服务器和CDN服务,后端也是套用的若依的上传功能

实现思路:

  • 前端根据视频文件计算出文件md5值
  • 前端按照指定大小截取视频,执行分片上传(可优化,先使用文件MD5检查文件是否已上传)
  • 后端实现接收分片的接口,当已上传分片数等于总分片数时执行合并分片,得到原视频文件
  • 后端使用ffmpeg按照时间进行视频分割,切割时间根据视频清晰度不同而不同,得到m3u8文件和ts文件列表
  • 后端保存视频信息和文件实际保存地址,并提供查询接口
  • 前端使用流播放器播放视频文件

代码实现

1. vue的分片上传

前端分片上传功能按照以下步骤实现:

1.1,先要写一个上传组件,这里使用elementUI的上传组件

:auto-upload 设置的视频直接不解释上传,即选择好本地文件就上传
:before-upload 中需要计算好文件的md5值,然后去后端查看文件是否已被上传
:http-request 中实现具体的分片上传逻辑
:action 虽然设置了上传地址,但是任然是以http-request设置的方法为准,只是不设置会报错

<el-form-item label="视频文件" prop="file" v-if="form.id==null"><el-upload ref="upload":action="uploadUrl":on-error="onError":before-upload="beforeUpload":before-remove="beforeRemove":auto-upload="true":limit="1":http-request="chunkedUpload":on-progress="onProgress"><div style="border: 1px dashed #c0ccda;padding: 1rem;"><i class="el-icon-upload"></i><div class="el-upload__text">将文件拖到此处,或<em>点击上传</em></div></div><div class="el-upload__tip" slot="tip">只能上传mp4文件,且不超过500M</div><el-progress :percentage="uploadPercentage" status="success"></el-progress></el-upload>
</el-form-item>

1.2,上传方法的js

我使用了两个后端接口,
一个是 testUploadVideo 判断文件是否存在,是若依分装的请求
一个是 process.env.VUE_APP_BASE_API + ‘/manage/video/upload’,单独用axios执行上传分片

在这里插入图片描述

<script>
import { addVideo, getVideo, testUploadVideo, updateVideo } from '@/api/manage/video'
import SparkMD5 from 'spark-md5'
import axios from 'axios'export default {name: 'videoWin',data() {return {uploadUrl: process.env.VUE_APP_BASE_API + '/manage/video/upload', //文件上传的路径uploadPromises: [], // 记录并发上传分片的线程uploadPercentage:0 //上传进度}},,methods: {beforeUpload: async function(file) {// 在上传之前获取视频的宽高和分辨率const video = document.createElement('video')video.src = URL.createObjectURL(file)video.preload = 'metadata'const loadedMetadata = new Promise(resolve => {video.onloadedmetadata = () => {window.URL.revokeObjectURL(video.src)const width = video.videoWidthconst height = video.videoHeightconsole.log('视频宽高:', width, height)this.form.width = widththis.form.height = heightresolve();}});// 等待视频的宽高和分辨率获取完成await loadedMetadata;// 计算文件的md5值const reader = new FileReader()const md5Promise = new Promise(resolve => {reader.onload = () => {const spark = new SparkMD5.ArrayBuffer()spark.append(reader.result)const md5 = spark.end(false)this.form.identifier = md5 // 将MD5值存储到form中resolve(md5);}});reader.readAsArrayBuffer(file); // 读取文件内容并计算MD5值const md5 = await md5Promise;// 检查文件是否已被上传const response = await testUploadVideo(md5);console.log("判断文件是否存在", response)if (response.msg === "文件已存在,秒传成功") {console.log("文件已存在")// 取消上传this.$refs.upload.abort(file);return false;} else {return true;}},chunkedUpload({ file }) {const totalSize = file.sizeconst chunkCount = Math.ceil(totalSize / (5 * 1024 * 1024)) // 每个分片5MB// 创建分片上传请求数组// 上传分片for (let i = 0; i < chunkCount; i++) {const start = i * (5 * 1024 * 1024)const end = Math.min((i + 1) * (5 * 1024 * 1024), totalSize)const chunk = file.slice(start, end)const formData = new FormData()formData.append('file', chunk)formData.append('filename', file.name)formData.append('totalChunks', chunkCount)formData.append('chunkNumber', i)formData.append('identifier', this.form.identifier) // 添加文件的MD5值作为参数// 发送分片上传请求const source = axios.CancelToken.source() // 创建cancelTokenconst uploadPromise = this.uploadChunk(formData, source.token, (progressEvent) => {console.log('更新进度', progressEvent)this.uploadPercentage = Math.round((progressEvent.loaded / progressEvent.total) * 100) // 更新进度条的值;}).catch(error => {console.error('分片上传失败', error)// 弹出告警消息this.$message({type: 'error',message: '视频上传失败!'})})this.uploadPromises.push({ promise: uploadPromise, source }) // 保存cancelToken}// 等待所有分片上传完成return Promise.all(this.uploadPromises).then(responses => {console.log('分片上传完成', responses)}).catch(error => {console.error('分片上传失败', error)})},/**更新进度*/onProgress(event, file) {this.uploadPercentage = Math.floor((event.loaded / event.total) * 100);},/**上传分片*/uploadChunk(formData, onProgress) {return axios.post(process.env.VUE_APP_BASE_API + '/manage/video/upload', formData, {onUploadProgress: onProgress // 添加进度回调}).then(response => {console.log('分片上传成功', response.data)})},/**上传分片失败*/onError(error, file, fileList) {console.error('上传失败', error)},// 取消上传请求beforeRemove(file, fileList) {this.form.identifier = nullreturn true}}
}
</script>

2. 后端接口实现

2.1 控制层代码

@RestController
@RequestMapping("/manage/video")
@CrossOrigin // 允许跨域
public class ManageVideoController extends BaseController {@Autowiredprivate IManageVideoService manageVideoService;/*** 上传分片前校验文件是否存在** @return*/@GetMapping("/preUpload")public AjaxResult preUpload(@RequestParam("fileMd5") String fileMd5) {return manageVideoService.checkExists(fileMd5);}/*** 上传分片** @return*/@PostMapping("/upload")public AjaxResult fragmentation(@ModelAttribute UploadPO uploadPO) {return manageVideoService.uploadChunk(uploadPO);}
}

2.1 服务层代码

接收到分片上传文件后经历以下步骤:

  1. 再次校验是否文件已存在,不存在就保存临时分片文件;
  2. 校验已上传分片数是否等于总分篇数,如果是则合并;
  3. 将临时文件合并和源mp4文件;
  4. 获取视频的时长和大小,因为ffmpeg不支持按照大小拆分,如果只是按照固定时长拆分,20s可能是2M也可能是34M,无法达到拆分视频以缩短预览视频等待时间的目的;
  5. 执行视频拆分,生成playlist.m3u8和一系列ts文件
  6. 重写m3u8文件的ts地址,1是因为若依开发环境和线上环境的指定前缀不一致,2是因为本地开发没开nginx转发静态资源,线上也没开文件服务
@Overridepublic AjaxResult checkExists(String fileMd5) {String fileUploadDir = RuoYiConfig.getProfile() + "/video";//判断文件是否已被上传String videoFile = fileUploadDir + "/" + fileMd5 + ".mp4";File file = new File(videoFile);if (file.exists()) {return AjaxResult.success("文件已存在,秒传成功");}return AjaxResult.success();}@Overridepublic AjaxResult uploadChunk(UploadPO uploadPO) {String fileUploadTempDir = RuoYiConfig.getProfile() + "/videotmp";String fileUploadDir = RuoYiConfig.getProfile() + "/video";// 获得文件分片数据MultipartFile fileData = uploadPO.getFile();// 分片第几片int index = uploadPO.getChunkNumber();//总分片数int totalChunk = uploadPO.getTotalChunks();// 文件md5标识String fileMd5 = uploadPO.getIdentifier();//判断文件是否已被上传String videoFile = fileUploadDir + "/" + fileMd5 + ".mp4";File file = new File(videoFile);if (file.exists()) {return AjaxResult.success("文件已存在,秒传成功");}String newName = fileMd5 + index + ".tem";File uploadFile = new File(fileUploadTempDir + "/" + fileMd5, newName);if (!uploadFile.getParentFile().exists()) {uploadFile.getParentFile().mkdirs();}try {fileData.transferTo(uploadFile);// 判断总分片数是否等于当前目录下的分片文件数量int currentChunkCount = getChunkCount(fileUploadTempDir + "/" + fileMd5);if (totalChunk == currentChunkCount) {// 调用合并方法merge(fileMd5, fileUploadTempDir, fileUploadDir);//根据运行环境分别调用ffmpegString os = System.getProperty("os.name").toLowerCase();String m3u8Dir = fileUploadDir + "/" + fileMd5;File m3u8FileDir = new File(m3u8Dir);if (!m3u8FileDir.exists()) {m3u8FileDir.mkdirs();}//计算视频总时长和视频大小,确定视频的分段时长String mp4File = fileUploadDir + "/" + fileMd5 + ".mp4";//每个2M分片的毫秒数long duration = getTsDuration(mp4File);// 异步执行视频拆分if (os.contains("win")) {mp4ToM3u8ForWindow(fileMd5, mp4File, m3u8Dir, duration);} else {mp4ToM3u8ForLinux(fileMd5, mp4File, m3u8Dir, duration);}}//执行成功返回 urlreturn AjaxResult.success();} catch (IOException | InterruptedException e) {log.error("上传视频失败:{}", e.toString());FileUtil.del(fileUploadTempDir + "/" + fileMd5); //删除临时文件FileUtil.del(videoFile); //删除视频源文件FileUtil.del(fileUploadDir + "/" + fileMd5); //删除分段ts视频return AjaxResult.error(502, "上传视频失败");} catch (EncoderException e) {log.error("视频切割时计算分段时长失败:{}", e.toString());FileUtil.del(fileUploadTempDir + "/" + fileMd5); //删除临时文件FileUtil.del(videoFile); //删除视频源文件FileUtil.del(fileUploadDir + "/" + fileMd5); //删除分段ts视频return AjaxResult.error(502, "上传视频失败");}}/*** 获取当前目录下的分片文件数量** @param directoryPath* @return*/private int getChunkCount(String directoryPath) {File directory = new File(directoryPath);if (!directory.exists() || !directory.isDirectory()) {return 0;}File[] files = directory.listFiles((dir, name) -> name.endsWith(".tem"));return files != null ? files.length : 0;}/*** 合并分片** @param uuid* @return*/public void merge(String uuid, String fileUploadTempDir, String fileUploadDir) throws IOException {File dirFile = new File(fileUploadTempDir + "/" + uuid);//分片上传的文件已经位于同一个文件夹下,方便寻找和遍历(当文件数大于十的时候记得排序用冒泡排序确保顺序是正确的)String[] fileNames = dirFile.list();Arrays.sort(fileNames, (o1, o2) -> {int i1 = Integer.parseInt(o1.substring(o1.indexOf(uuid) + uuid.length()).split("\\.tem")[0]);int i2 = Integer.parseInt(o2.substring(o2.indexOf(uuid) + uuid.length()).split("\\.tem")[0]);return i1 - i2;});//创建空的合并文件,以未见md5为文件名File targetFile = new File(fileUploadDir, uuid + ".mp4");if (!targetFile.getParentFile().exists()) {targetFile.getParentFile().mkdirs();}RandomAccessFile writeFile = new RandomAccessFile(targetFile, "rw");long position = 0;for (String fileName : fileNames) {System.out.println(fileName);File sourceFile = new File(fileUploadTempDir + "/" + uuid, fileName);RandomAccessFile readFile = new RandomAccessFile(sourceFile, "rw");int chunksize = 1024 * 3;byte[] buf = new byte[chunksize];writeFile.seek(position);int byteCount;while ((byteCount = readFile.read(buf)) != -1) {if (byteCount != chunksize) {byte[] tempBytes = new byte[byteCount];System.arraycopy(buf, 0, tempBytes, 0, byteCount);buf = tempBytes;}writeFile.write(buf);position = position + byteCount;}readFile.close();}writeFile.close();cn.hutool.core.io.FileUtil.del(dirFile);}/*** 视频拆分** @param inputFilePath   D:/home/dxhh/uploadPath/video/md5.mp4* @param outputDirectory D:/home/dxhh/uploadPath/video/md5*/@Asyncpublic void mp4ToM3u8ForWindow(String fileMd5, String inputFilePath, String outputDirectory, long ms) throws IOException {File uploadFile = new File(outputDirectory);if (!uploadFile.exists()) {uploadFile.mkdirs();}Path outputDirPath = Paths.get(outputDirectory);//我的ffmpeg.exe放在 项目的/resources/script目录下Path resourcePath = Paths.get("./script/ffmpeg.exe");FFmpeg.atPath(resourcePath.getParent()).addInput(UrlInput.fromPath(Paths.get(inputFilePath))).addOutput(UrlOutput.toPath(outputDirPath.resolve("output_%03d.ts"))).addArguments("-f", "segment").addArguments("-segment_time", ms + "ms") // 分片时长为30s.addArguments("-segment_list", outputDirPath.resolve("playlist.m3u8").toString()).addArguments("-c:v", "copy") // 优化视频编码参数.addArguments("-c:a", "copy") // 优化音频编码参数.execute();// 修改生成的m3u8文件,将ts链接替换为完整URLupdateM3u8File(fileMd5, outputDirectory);}/*** 视频拆分** @param fileMd5         adw1dwdadadwdadasd* @param inputFilePath   /home/dxhh/uploadPath/video/md5.mp4* @param outputDirectory /home/dxhh/uploadPath/video/md5* @throws IOException* @throws InterruptedException*/public void mp4ToM3u8ForLinux(String fileMd5, String inputFilePath, String outputDirectory, long ms) throws IOException, InterruptedException {String command = "ffmpeg -i " + inputFilePath + " -c copy -map 0 -f segment -segment_time " + ms + "ms -segment_list " + outputDirectory + "/playlist.m3u8 " + outputDirectory + "/output_%03d.ts";//ffmpeg -i /home/dxhh/uploadPath/video/md5.mp4 -c copy -map 0 -f segment -segment_time 1236ms -segment_list /home/dxhh/uploadPath/video/md5/playlist.m3u8 /home/dxhh/uploadPath/video/md5/output_%03d.tslog.info("视频分割脚本:{}", command);ProcessBuilder builder = new ProcessBuilder(command.split(" "));builder.redirectErrorStream(true);Process process = builder.start();BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream()));String line;while ((line = reader.readLine()) != null) {System.out.println(line);}int exitCode = process.waitFor();if (exitCode == 0) {System.out.println("FFmpeg command executed successfully");updateM3u8File(fileMd5, outputDirectory);} else {System.out.println("FFmpeg command failed with exit code " + exitCode);}}private void updateM3u8File(String fileMd5, String outputDirectory) throws IOException {String m3u8FilePath = outputDirectory + "/playlist.m3u8";List<String> lines = Files.readAllLines(Paths.get(m3u8FilePath));List<String> newLines = new ArrayList<>();for (String line : lines) {if (line.endsWith(".ts")) {if ("dev".equals(active)) {newLines.add("/dev-api/profile/video/" + fileMd5 + "/" + line);} else {newLines.add("/stage-api/profile/video/" + fileMd5 + "/" + line);}} else {newLines.add(line);}}Files.write(Paths.get(m3u8FilePath), newLines);}public long getTsDuration(String filePath) throws EncoderException {int targetSize = 2 * 1024 * 1024; // 2MBFile videoFile = new File(filePath);long fileSize = videoFile.length();Encoder encoder = new Encoder();MultimediaInfo multimediaInfo = encoder.getInfo(videoFile);long duration = multimediaInfo.getDuration();System.out.println("Duration: " + duration + " ms");System.out.println("File size: " + fileSize + " bytes");// Calculate target duration for a 2MB videolong targetDuration = (duration * targetSize) / fileSize;System.out.println("Target duration for a 2MB video: " + targetDuration + " ms");return targetDuration;}

获取视频时长需要用到jave工具包,想上传资源的提示已存在,应该可以在csdn搜到;
还需要ffmpeg软件,如果是windows环境运行,只需要调用本地的ffmpeg.exe就好,如果是在linux运行,需要安装ffmpeg;

   <!--视频切割--><dependency><groupId>com.github.kokorin.jaffree</groupId><artifactId>jaffree</artifactId><version>2023.09.10</version></dependency><dependency><groupId>it.sauronsoftware.jave</groupId><artifactId>jave2</artifactId><version>1.0.2</version><scope>system</scope><systemPath>${project.basedir}/lib/jave-1.0.2.jar</systemPath></dependency>

2.3 linux中安装ffmpeg

  1. 下载 ffmpeg 工具包并解压
wget http://www.ffmpeg.org/releases/ffmpeg-4.2.tar.gz
tar -zxvf ffmpeg-4.2.tar.gz
  1. 进入工具包文件夹并进行安装,将 ffmpeg 安装至 / usr/local/ffmpeg 下
 cd ffmpeg-4.2./configure --prefix=/usr/local/ffmpeg
./configure --prefix=/usr/local/ffmpeg --enable-openssl --disable-x86asm
make && make install

注意:若出现以下报错,请跳至第五步,待第五步安装成功后再返回第二步。
在这里插入图片描述

  1. 配置环境变量,使其 ffmpeg 命令生效
 #利用vi编辑环境变量
vi /etc/profile#在最后位置处添加环境变量,点击i进入编辑模式,esc键可退出编辑模式
export PATH=$PATH:/usr/local/ffmpeg/bin#退出编辑模式后,:wq 保存退出
#刷新资源,使其生效
source /etc/profile
  1. 查看 ffmpeg 版本,验证是否安装成功
ffmpeg -version

若出现以下内容,则安装成功。

在这里插入图片描述

  1. 若第二步出现图片中的错误信息,则需要安装 yasm

记得退出 ffmpeg 工具包文件夹,cd … 返回上一层

 #下载yasm工具包
wget http://www.tortall.net/projects/yasm/releases/yasm-1.3.0.tar.gz#解压
tar -zxvf yasm-1.3.0.tar.gz#进入工具包文件夹并开始安装
cd yasm-1.3.0
./configure
make && make install

安装完成后直接返回第二步即可,此时命令就不会报错了。

2.4 视频资源地址

因为是基于若依框架开发的,其实只要上传的的时候是往 RuoYiConfig.getProfile() 这个指定配置目录保存文件,都是能直接访问不需要额外开发,这里就简单过一下
若依的自定义参数配置类从yml文件读取用户配置

@Component
@ConfigurationProperties(prefix = "xxx")
public class RuoYiConfig {/*** 上传路径 /home/user/xxxx/upload*/private static String profile;
}

在通用配置定义一个静态资源路由前缀

/*** 通用常量定义** @author li.dh*/
public class CommonConstant {/*** 资源映射路径 前缀*/public static final String RESOURCE_PREFIX = "/profile";
}

在mvc配置中添加静态资源的转发映射,将/profile前缀的请求转发到RuoYiConfig.getProfile()路径下

@Configuration
public class ResourcesConfig implements WebMvcConfigurer {@Overridepublic void addResourceHandlers(ResourceHandlerRegistry registry) {/** 本地文件上传路径 */registry.addResourceHandler(CommonConstant.RESOURCE_PREFIX + "/**").addResourceLocations("file:" + RuoYiConfig.getProfile() + "/");/** swagger配置 */registry.addResourceHandler("/swagger-ui/**").addResourceLocations("classpath:/META-INF/resources/webjars/springfox-swagger-ui/").setCacheControl(CacheControl.maxAge(5, TimeUnit.HOURS).cachePublic());}
}

3. vue播放流视频

我的需求是在列表上点击视频弹出播放弹窗

<!-- 播放视频 --><el-dialog :title="title" :visible.sync="open" width="800px" append-to-body @close="open=false"><video-player class="video-player vjs-custom-skin"ref="videoPlayer":playsinline="true":options="playerOptions"></video-player></el-dialog>
import 'video.js/dist/video-js.css'data(){return {// 弹出层标题title: '',m3u8Url: '',// 是否显示弹出层open: false,playerOptions: {playbackRates: [0.5, 1.0, 1.5, 2.0], // 可选的播放速度autoplay: true, // 如果为true,浏览器准备好时开始回放。muted: false, // 默认情况下将会消除任何音频。loop: false, // 是否视频一结束就重新开始。preload: 'auto', // 建议浏览器在<video>加载元素后是否应该开始下载视频数据。auto浏览器选择最佳行为,立即开始加载视频(如果浏览器支持)language: 'zh-CN',aspectRatio: '16:9', // 将播放器置于流畅模式,并在计算播放器的动态大小时使用该值。值应该代表一个比例 - 用冒号分隔的两个数字(例如"16:9"或"4:3")fluid: true, // 当true时,Video.js player将拥有流体大小。换句话说,它将按比例缩放以适应其容器。sources: [{type: 'application/x-mpegURL', // 类型src: this.m3u8Url}],poster: '', // 封面地址notSupportedMessage: '此视频暂无法播放,请稍后再试', // 允许覆盖Video.js无法播放媒体源时显示的默认信息。controlBar: {timeDivider: true, // 当前时间和持续时间的分隔符durationDisplay: true, // 显示持续时间remainingTimeDisplay: false, // 是否显示剩余时间功能fullscreenToggle: true // 是否显示全屏按钮}}}
},
methods: {openVideo(picurl, url, title) {this.title = titlelet videourl = process.env.VUE_APP_BASE_API + urllet imgurl = process.env.VUE_APP_BASE_API + picurl// console.log("视频地址:" , videourl)this.m3u8Url = videourlthis.playerOptions.sources[0].src = videourl // 重新加载视频this.playerOptions.poster = imgurl // 封面// this.$refs.videoPlayer.play() // 播放视频this.open = true}
}

4. 实现效果

在这里插入图片描述

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

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

相关文章

2023NewStarCTF

目录 一、阳光开朗大男孩 二、大怨种 三、2-分析 四、键盘侠 五、滴滴滴 六、Include? 七、medium_sql 八、POP Gadget 九、OtenkiGirl 一、阳光开朗大男孩 1.题目给出了secret.txt和flag.txt两个文件&#xff0c;secret.txt内容如下&#xff1a; 法治自由公正爱国…

【Redis】list常用命令内部编码使用场景

文章目录 前置知识列表类型的特点 命令LPUSHLPUSHXRPUSHRPUSHXLRANGELPOPRPOPLINDEXLREMLINSERTLTRIMLSETLLEN 阻塞版本命令BLPOPBRPOP 命令总结内部编码测试内部编码 使用场景消息队列分频道的消息队列 模拟栈和队列 前置知识 列表类型是⽤来存储多个有序的字符串&#xff0c…

第一次实操Python+robotframework接口自动化测试

目前我们需要考虑的是如何实现关键字驱动实现接口自动化输出&#xff0c;通过关键字的封装实现一定意义上的脚本与用例的脱离&#xff01; robot framework 的安装不过多说明&#xff0c;网上资料比较太多~ 实例&#xff1a;&#xff01;&#xff01;&#xff01;&#xff01…

AI:80-基于深度学习的医学图像分割与病变识别

🚀 本文选自专栏:人工智能领域200例教程专栏 从基础到实践,深入学习。无论你是初学者还是经验丰富的老手,对于本专栏案例和项目实践都有参考学习意义。 ✨✨✨ 每一个案例都附带有在本地跑过的代码,详细讲解供大家学习,希望可以帮到大家。欢迎订阅支持,正在不断更新中,…

Flink SQL 表值聚合函数(Table Aggregate Function)详解

使用场景&#xff1a; 表值聚合函数即 UDTAF&#xff0c;这个函数⽬前只能在 Table API 中使⽤&#xff0c;不能在 SQL API 中使⽤。 函数功能&#xff1a; 在 SQL 表达式中&#xff0c;如果想对数据先分组再进⾏聚合取值&#xff1a; select max(xxx) from source_table gr…

2022年06月 Python(五级)真题解析#中国电子学会#全国青少年软件编程等级考试

Python等级考试(1~6级)全部真题・点这里 一、单选题(共25题,每题2分,共50分) 第1题 Python中 print(“八进制{: o}”.format(12)) 正确的输出结果是?( ) A: 八进制:O B: 八进制:O14 C: 八进制14O D: 八进制14 答案:D 字符串的format()格式。 第2题 下列的程…

pychon/PIL/opencv/json学习过程中遇到的问题

1. 使用PIL.Image读取图片 注意&#xff1a;pytorch中对图像预处理是transforms的输入必须是PIL格式的文件&#xff0c;使用cv2读取的图片就按照第二条的代码处理&#xff08;3通道合并、归一化处理&#xff09; from PIL import Image img Image.open("test1.jpg"…

2023 年最新企业微信官方会话机器人开发详细教程(更新中)

目标是开发一个简易机器人&#xff0c;能接收消息并作出回复。 获取企业 ID 企业信息页面链接地址&#xff1a;https://work.weixin.qq.com/wework_admin/frame#profile 自建企业微信机器人 配置机器人应用详情 功能配置 接收消息服务器配置 配置消息服务器配置 配置环境变量…

如何利用 cpolar 内网穿透技术实现 U8 用友 ERP 异地访问

文章目录 前言1. 服务器本机安装U8并调试设置2. 用友U8借助cpolar实现企业远程办公2.1 在被控端电脑上&#xff0c;点击开始菜单栏&#xff0c;打开设置——系统2.2 找到远程桌面2.3 启用远程桌面 3. 安装cpolar内网穿透3.1 注册cpolar账号3.2 下载cpolar客户端 4. 获取远程桌面…

【MATLAB源码-第75期】基于模拟退火算法(SA)的栅格路径规划,输出做短路径图和适应度曲线。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 模拟退火算法是一种启发式优化算法&#xff0c;通常用于解决组合优化问题&#xff0c;例如旅行商问题和图着色问题。它模拟了固体材料在退火过程中逐渐冷却达到稳定状态的行为&#xff0c;以寻找问题的全局最优解。 以下是模…

Hadoop原理,HDFS架构,MapReduce原理

Hadoop原理&#xff0c;HDFS架构&#xff0c;MapReduce原理 2022找工作是学历、能力和运气的超强结合体&#xff0c;遇到寒冬&#xff0c;大厂不招人&#xff0c;可能很多算法学生都得去找开发&#xff0c;测开 测开的话&#xff0c;你就得学数据库&#xff0c;sql&#xff0c…

postswigger 靶场(CSRF)攻略-- 1.没有防御措施的 CSRF 漏洞

靶场地址&#xff1a; What is CSRF (Cross-site request forgery)? Tutorial & Examples | Web Security Academy (portswigger.net)https://portswigger.net/web-security/csrf 没有防御措施的 CSRF 漏洞 题目中已告知易受攻击的是电子邮件的更改功能&#xff0c;而目…

Oracle Primavera Unifier 23.10 新特征

根据官方的说法&#xff0c;Unifier 23.7 ~ 23.9 更多为对功能bug的修复&#xff0c;以下将对23.10进行重点介绍 Cost Sheets Cost Sheets Support Conditional Formatting Conditional formatting of table data is now supported in cost sheets with features such as ce…

基于注解的声明式事务

1.什么是事务 数据库事务(transaction)是访问并可能操作各种数据项的一个数据库操作序列&#xff0c;这些操作要么全部执行要么全部不执行&#xff0c;是一个不可分割的工作单位。事务由事务开始与事务结束之间执行的全部数据库操作组成。 2.事务的特性 A&#xff1a;原子性(A…

【FPGA】十进制计数器 | 实现 4-bit 2421 十进制计数器 | 有限状态机(FSM)

目录 Ⅰ. 实践说明 0x00 十进制计数器 0x01 有限状态机&#xff08;FSM&#xff09; Ⅱ. 实践部分 0x00 4-bit 2421 十进制计数器 Ⅰ. 实践说明 0x00 十进制计数器 十进制计数器是一种以十进制运算的计数器&#xff0c;从 0 数到 9&#xff0c;然后返回 0 状态。由于它需…

青少年编程学习 等级考试 蓝桥杯/NOC/GESP等比赛资料合集

一、博主愚见 在当今信息技术高速发展的时代&#xff0c;编程已经成为了一种必备的技能。随着社会对于科技人才的需求不断增加&#xff0c;青少年编程学习正逐渐成为一种趋势。为了更好地帮助青少年学习编程&#xff0c;提升他们的技能和素质&#xff0c;博主结合自身多年从事青…

如何使用CORS和CSP保护前端应用程序安全

前端应用在提供无缝用户体验方面起着核心作用。在当今互联网的环境中&#xff0c;第三方集成和API的普及使得确保强大的安全性至关重要。安全漏洞可能导致数据盗窃、未经授权访问以及品牌声誉受损。本文将向您展示如何使用CORS和CSP为您的网页增加安全性。 嗨&#xff0c;大家好…

大数据可视化数据大屏可视化模板【可视化项目案例-05】

🎉🎊🎉 你的技术旅程将在这里启航! 🚀🚀 本文选自专栏:可视化技术专栏100例 可视化技术专栏100例,包括但不限于大屏可视化、图表可视化等等。订阅专栏用户在文章底部可下载对应案例源码以供大家深入的学习研究。 🎓 每一个案例都会提供完整代码和详细的讲解,不…

基于安卓android微信小程序的校园互助平台

项目介绍 随着社会的发展&#xff0c;社会的方方面面都在利用信息化时代的优势。互联网的优势和普及使得各种系统的开发成为必需。 本文以实际运用为开发背景&#xff0c;运用软件工程原理和开发方法&#xff0c;它主要是采用java语言技术和mysql数据库来完成对系统的设计。整…

Unity中Shader的间接光的产生Meta Pass

文章目录 前言Unity中Shader的间接光的产生Meta Pass&#xff0c;这也是属于全局光照 GI 的内容。主要实现像现实生活中&#xff0c;光线照到有颜色的物体后&#xff0c;该物体有反射出该颜色的光的效果。 一、我们先使用Unity自带的Shader看看间接光效果1、先按照如下设置搭建…