手摸手系列之前端Vue实现PDF预览及打印的终极解决方案

前言

近期我正在开发一个前后端分离项目,使用了Spring Boot 和 Vue2,借助了国内优秀的框架 jeecg,前端UI库则选择了 ant-design-vue。在项目中,需要实现文件上传功能,同时还要能够在线预览和下载图片和PDF文件,甚至需要在页面上直接打印PDF文件。尽管框架自带了 vue-print-nb-jeecg 组件,但它相对较为简陋,只支持单页打印,无法实现多页打印。经过仔细的权衡和比较后,最终决定采用 vue-pdf print-js 组件来满足需求。

一、先来展示一下最终效果

前端上传文件列表:
在这里插入图片描述

点击PDF文件后展示预览:

在这里插入图片描述

点击打印按钮后效果:

在这里插入图片描述

二、实现步骤及代码

vue-pdf 可以用于在线预览,而 print-js 则提供了更强大的打印功能,支持多种文档类型,包括PDF、HTML、IMAGE和JSON,而且默认情况下是PDF。其实vue-pdf 也可以实现打印功能,但是跟前述的vue-print-nb一样,只能打印页面显示的第一页内容(预览展示没问题)。
Print.js官网👉点我直达

1. 在vue中安装vue-pdfPrint.js
yarn add vue-pdf
...
yarn add print-js
2. 可以全局引入,也可以在需要的文件中引入
 import pdf from 'vue-pdf'import printJS from 'print-js'
3.主要代码
<a-modal :visible="previewVisibleForAll" :footer="null" @cancel="handleCancelAll" :width="800" :maskClosable="maskClosable" :keyboard="keyboard"><img alt="example" style="width: 100%;margin-top:20px" :src="previewFileSrc" v-if="isImage"/><div v-if="isPdf" style="overflow-y: auto;overflow-x: hidden;"><a-button shape="round" icon="file-pdf" @click="handlePrint(printData)" size="small">打印</a-button><div id="printFrom"><pdf ref="pdf" v-for="item in pageTotal":src="previewFileSrc":key="item":page="item"></pdf></div></div></a-modal>

打印按钮执行的方法

// data参数
printData: {printable: 'printFrom',header: '',ignore: ['no-print']
},// 执行方法
handlePrint(params) {printJS({printable: params.printable, // 'printFrom', // 标签元素idtype: params.type || 'html',header: params.header, // '表单',targetStyles: ['*'],style: '@page {margin:0 10mm};', // 可选-打印时去掉眉页眉尾ignoreElements: params.ignore || [], // ['no-print']properties: params.properties || null})
}

不同组件,如果文件是图片就预览图片,PDF就预览PDF。

4. 全部代码:
<template><div :id="containerId" style="position: relative"><!--  ---------------------------- begin 图片左右换位置 ------------------------------------- --><div class="movety-container" :style="{top:top+'px',left:left+'px',display:moveDisplay}" style="padding:0 8px;position: absolute;z-index: 91;height: 32px;width: 104px;text-align: center;"><div :id="containerId+'-mover'" :class="showMoverTask?'uploadty-mover-mask':'movety-opt'" style="margin-top: 12px"><a @click="moveLast" style="margin: 0 5px;"><a-icon type="arrow-left" style="color: #fff;font-size: 16px"/></a><a @click="moveNext" style="margin: 0 5px;"><a-icon type="arrow-right" style="color: #fff;font-size: 16px"/></a></div></div><!--  ---------------------------- end 图片左右换位置 ------------------------------------- --><a-uploadname="file":multiple="multiple":action="uploadAction":headers="headers":data="{'biz':bizPath}":fileList="fileList":beforeUpload="doBeforeUpload"@change="handleChange":disabled="disabled":returnUrl="returnUrl":listType="complistType"@preview="handlePreview1":showUploadList="{showRemoveIcon: true,showDownloadIcon: true}":class="{'uploadty-disabled':disabled}"><template><div v-if="isImageComp"><a-icon type="plus" /><div class="ant-upload-text">{{ text }}</div></div><a-button v-else-if="buttonVisible"><a-icon type="upload" />{{ text }}</a-button></template></a-upload><a-modal :visible="previewVisible" :footer="null" @cancel="handleCancel"><img alt="example" style="width: 100%" :src="previewImage" /></a-modal><a-modal :visible="previewVisibleForAll" :footer="null" @cancel="handleCancelAll" :width="800" :maskClosable="maskClosable" :keyboard="keyboard"><img alt="example" style="width: 100%;margin-top:20px" :src="previewFileSrc" v-if="isImage"/><div v-if="isPdf" style="overflow-y: auto;overflow-x: hidden;"><a-button shape="round" icon="file-pdf" @click="handlePrint(printData)" size="small">打印</a-button><div id="printFrom"><pdf ref="pdf" v-for="item in pageTotal":src="previewFileSrc":key="item":page="item"></pdf></div></div></a-modal></div>
</template><script>import Vue from 'vue'import { ACCESS_TOKEN } from "@/store/mutation-types"import { getFileAccessHttpUrl } from '@/api/manage';import pdf from 'vue-pdf'import printJS from 'print-js'const FILE_TYPE_ALL = "all"const FILE_TYPE_IMG = "image"const FILE_TYPE_TXT = "file"const uidGenerator=()=>{return '-'+parseInt(Math.random()*10000+1,10);}const getFileName=(path)=>{if(path.lastIndexOf("\\")>=0){let reg=new RegExp("\\\\","g");path = path.replace(reg,"/");}return path.substring(path.lastIndexOf("/")+1);}export default {name: 'JUpload',components: { pdf },data(){return {printData: {printable: 'printFrom',header: '',ignore: ['no-print']},uploadAction:window._CONFIG['domianURL']+"/sys/common/upload",headers:{},fileList: [],newFileList: [],uploadGoOn:true,previewVisible: false,//---------------------------- begin 图片左右换位置 -------------------------------------previewImage: '',containerId:'',top:'',left:'',moveDisplay:'none',showMoverTask:false,moverHold:false,currentImg:'',//---------------------------- end 图片左右换位置 -------------------------------------previewVisibleForAll:false,pageTotal: null,previewFileSrc:'',isImage:false,isExcel:false,isPdf:false,}},props:{text:{type:String,required:false,default:"点击上传"},fileType:{type:String,required:false,default:FILE_TYPE_ALL},/*这个属性用于控制文件上传的业务路径*/bizPath:{type:String,required:false,default:"temp"},value:{type:[String,Array],required:false},// update-begin- --- author:wangshuai ------ date:20190929 ---- for:Jupload组件增加是否能够点击disabled:{type:Boolean,required:false,default: false},// update-end- --- author:wangshuai ------ date:20190929 ---- for:Jupload组件增加是否能够点击//此属性被废弃了triggerChange:{type: Boolean,required: false,default: false},/*** update -- author:lvdandan -- date:20190219 -- for:Jupload组件增加是否返回url,* true:仅返回url* false:返回fileName filePath fileSize*/returnUrl:{type:Boolean,required:false,default: true},number:{type:Number,required:false,default: 0},buttonVisible:{type:Boolean,required:false,default: true},multiple: {type: Boolean,default: true},beforeUpload: {type: Function},maskClosable: {type: Boolean,default:true,},keyboard: {type: Boolean,default:true,},},watch:{value:{immediate: true,handler() {let val = this.valueif (val instanceof Array) {if(this.returnUrl){this.initFileList(val.join(','))}else{this.initFileListArr(val);}} else {this.initFileList(val)}}}},computed:{isImageComp(){return this.fileType === FILE_TYPE_IMG},complistType(){return this.fileType === FILE_TYPE_IMG?'picture-card':'text'}},created(){const token = Vue.ls.get(ACCESS_TOKEN);//---------------------------- begin 图片左右换位置 -------------------------------------this.headers = {"X-Access-Token":token};this.containerId = 'container-ty-'+new Date().getTime();//---------------------------- end 图片左右换位置 -------------------------------------},methods:{handlePrint(params) {printJS({printable: params.printable, // 'printFrom', // 标签元素idtype: params.type || 'html',header: params.header, // '表单',targetStyles: ['*'],style: '@page {margin:0 10mm};', // 可选-打印时去掉眉页眉尾ignoreElements: params.ignore || [], // ['no-print']properties: params.properties || null})},printPdf() {this.$refs.pdf.print()// window.print()},initFileListArr(val){console.log(val)if(!val || val.length==0){this.fileList = [];return;}let fileList = [];for(var a=0;a<val.length;a++){let url = getFileAccessHttpUrl(val[a].filePath);fileList.push({uid:uidGenerator(),name:val[a].fileName,status: 'done',url: url,response:{status:"history",message:val[a].filePath}})}this.fileList = fileListconsole.log(this.fileList)},initFileList(paths){if(!paths || paths.length==0){//return [];// update-begin- --- author:os_chengtgen ------ date:20190729 ---- for:issues:326,Jupload组件初始化bugthis.fileList = [];return;// update-end- --- author:os_chengtgen ------ date:20190729 ---- for:issues:326,Jupload组件初始化bug}let fileList = [];let arr = paths.split(",")for(var a=0;a<arr.length;a++){let url = getFileAccessHttpUrl(arr[a]);fileList.push({uid:uidGenerator(),name:getFileName(arr[a]),status: 'done',url: url,response:{status:"history",message:arr[a]}})}this.fileList = fileList},handlePathChange(){let uploadFiles = this.fileListlet path = ''if(!uploadFiles || uploadFiles.length==0){path = ''}let arr = [];for(var a=0;a<uploadFiles.length;a++){// update-begin-author:lvdandan date:20200603 for:【TESTA-514】【开源issue】多个文件同时上传时,控制台报错if(uploadFiles[a].status === 'done' ) {arr.push(uploadFiles[a].response.message)}else{return;}// update-end-author:lvdandan date:20200603 for:【TESTA-514】【开源issue】多个文件同时上传时,控制台报错}if(arr.length>0){path = arr.join(",")}this.$emit('change', path);},doBeforeUpload(file){this.uploadGoOn=truevar fileType = file.type;if(this.fileType===FILE_TYPE_IMG){if(fileType.indexOf('image')<0){this.$message.warning('请上传图片');this.uploadGoOn=falsereturn false;}}// 文件大小限定在600K以下const isLt2M = file.size / 1024 / 1024 < 10;if (!isLt2M){this.$message.warning('请确保上传的文件小于10MB!');this.fileList = []this.uploadGoOn=falsereturn false;}// 扩展 beforeUpload 验证if (typeof this.beforeUpload === 'function') {return this.beforeUpload(file)}return true},handleChange(info) {console.log("--文件列表改变--")if(!info.file.status && this.uploadGoOn === false){info.fileList.pop();}let fileList = info.fileListif(info.file.status==='done'){if(this.number>0){fileList = fileList.slice(-this.number);}if(info.file.response.success){fileList = fileList.map((file) => {if (file.response) {let reUrl = file.response.message;file.url = getFileAccessHttpUrl(reUrl);}return file;});}//this.$message.success(`${info.file.name} 上传成功!`);}else if (info.file.status === 'error') {this.$message.error(`${info.file.name} 上传失败.`);}else if(info.file.status === 'removed'){this.handleDelete(info.file)}this.fileList = fileListif(info.file.status==='done' || info.file.status === 'removed'){//returnUrl为true时仅返回文件路径if(this.returnUrl){this.handlePathChange()}else{//returnUrl为false时返回文件名称、文件路径及文件大小this.newFileList = [];for(var a=0;a<fileList.length;a++){// update-begin-author:lvdandan date:20200603 for:【TESTA-514】【开源issue】多个文件同时上传时,控制台报错if(fileList[a].status === 'done' ) {var fileJson = {fileName:fileList[a].name,filePath:fileList[a].response.message,fileSize:fileList[a].size};this.newFileList.push(fileJson);}else{return;}// update-end-author:lvdandan date:20200603 for:【TESTA-514】【开源issue】多个文件同时上传时,控制台报错}this.$emit('change', this.newFileList);}}},handleDelete(file){//如有需要新增 删除逻辑console.log(file)},// handlePreview(file){//   console.log('file')//   console.log(file)//   if(this.fileType === FILE_TYPE_IMG){//     this.previewImage = file.url || file.thumbUrl;//     this.previewVisible = true;//   }else{//     if(file.name.endsWith('pdf') || file.name.endsWith('PDF')) {//       let viewPath = window._CONFIG['domianURL'].replace('9999', '15550').replace('/jeecg-boot', '') + '/' + (file.url.replace(window._CONFIG['staticDomainURL'] + "/", ''))//       console.log(viewPath)//       window.open(viewPath,"_blank")//     this.isPdf = true//       this.previewFileSrc = file.url//     }//     // else {//TODO:重新打开页面//     //   location.href=file.url//     // }//   }// },// 获取pdf总页数getTotal() {// 多页pdf的src中不能直接使用后端获取的pdf地址 否则会按页数请求多次数据// 需要使用下述方法的返回值作为urlthis.previewFileSrc = pdf.createLoadingTask(this.previewFileSrc)// 获取页码this.previewFileSrc.promise.then(pdf => this.pageTotal = pdf.numPages).catch(error => {})},handlePreview1(file){if(this.fileType === FILE_TYPE_IMG){this.previewImage = file.url || file.thumbUrl;this.previewVisible = true;}else{// 判断当前文件类型this.previewFileSrc = file.url || file.thumbUrl; // "http://localhost:9999/sys/common/static/orderPaymentInfo/网约区域复习题_1694585302732.pdf"let fileTypeLocal = this.matchFileType(file.name);this.isImage = false;this.isPdf = false;if(fileTypeLocal == 'image') {this.previewVisibleForAll = true;this.isImage = true;} else if(fileTypeLocal == 'pdf') {this.previewVisibleForAll = true;this.isPdf = true;this.getTotal()} else {location.href=file.url}}},matchFileType(fileName) {// 后缀获取let suffix = '';// 获取类型结果let result = '';if (!fileName) return false;try {// 截取文件后缀suffix = fileName.substr(fileName.lastIndexOf('.') + 1, fileName.length)// 文件后缀转小写,方便匹配suffix = suffix.toLowerCase()} catch (err) {suffix = '';}// fileName无后缀返回 falseif (!suffix) {result = false;return result;}let fileTypeList = [// 图片类型{'typeName': 'image', 'types': ['png', 'jpg', 'jpeg', 'bmp', 'gif']},// 文本类型{'typeName': 'txt', 'types': ['txt']},// excel类型{'typeName': 'excel', 'types': ['xls', 'xlsx']},{'typeName': 'word', 'types': ['doc', 'docx']},{'typeName': 'pdf', 'types': ['pdf']},{'typeName': 'ppt', 'types': ['ppt']},// 视频类型{'typeName': 'video', 'types': ['mp4', 'm2v', 'mkv']},// 音频{'typeName': 'radio', 'types': ['mp3', 'wav', 'wmv']}]// let fileTypeList = ['image', 'txt', 'excel', 'word', 'pdf', 'video', 'radio']for (let i = 0; i < fileTypeList.length; i++) {const fileTypeItem = fileTypeList[i]const typeName = fileTypeItem.typeNameconst types = fileTypeItem.typesconsole.log(fileTypeItem);result = types.some(function (item) {return item === suffix;});if (result) {return typeName}}return 'other'},handleCancel(){this.previewVisible = false;},handleCancelAll(){this.previewVisibleForAll = false;this.isImage = false;this.isPdf = false;},//---------------------------- begin 图片左右换位置 -------------------------------------moveLast(){//console.log(ev)//console.log(this.fileList)//console.log(this.currentImg)let index = this.getIndexByUrl();if(index==0){this.$message.warn('未知的操作')}else{let curr = this.fileList[index].url;let last = this.fileList[index-1].url;let arr =[]for(let i=0;i<this.fileList.length;i++){if(i==index-1){arr.push(curr)}else if(i==index){arr.push(last)}else{arr.push(this.fileList[i].url)}}this.currentImg = lastthis.$emit('change',arr.join(','))}},moveNext(){let index = this.getIndexByUrl();if(index==this.fileList.length-1){this.$message.warn('已到最后~')}else{let curr = this.fileList[index].url;let next = this.fileList[index+1].url;let arr =[]for(let i=0;i<this.fileList.length;i++){if(i==index+1){arr.push(curr)}else if(i==index){arr.push(next)}else{arr.push(this.fileList[i].url)}}this.currentImg = nextthis.$emit('change',arr.join(','))}},getIndexByUrl(){for(let i=0;i<this.fileList.length;i++){if(this.fileList[i].url === this.currentImg || encodeURI(this.fileList[i].url) === this.currentImg){return i;}}return -1;}},mounted(){const moverObj = document.getElementById(this.containerId+'-mover');if(moverObj){moverObj.addEventListener('mouseover',()=>{this.moverHold = truethis.moveDisplay = 'block';});moverObj.addEventListener('mouseout',()=>{this.moverHold = falsethis.moveDisplay = 'none';});}let picList = document.getElementById(this.containerId)?document.getElementById(this.containerId).getElementsByClassName('ant-upload-list-picture-card'):[];if(picList && picList.length>0){picList[0].addEventListener('mouseover',(ev)=>{ev = ev || window.event;let target = ev.target || ev.srcElement;if('ant-upload-list-item-info' == target.className){this.showMoverTask=falselet item = target.parentElementthis.left = item.offsetLeftthis.top=item.offsetTop+item.offsetHeight-50;this.moveDisplay = 'block';this.currentImg = target.getElementsByTagName('img')[0].src}});picList[0].addEventListener('mouseout',(ev)=>{ev = ev || window.event;let target = ev.target || ev.srcElement;//console.log('移除',target)if('ant-upload-list-item-info' == target.className){this.showMoverTask=truesetTimeout(()=>{if(this.moverHold === false)this.moveDisplay = 'none';},100)}if('ant-upload-list-item ant-upload-list-item-done' == target.className || 'ant-upload-list ant-upload-list-picture-card'== target.className){this.moveDisplay = 'none';}})//---------------------------- end 图片左右换位置 -------------------------------------}},model: {prop: 'value',event: 'change'}}
</script><style lang="less">
.uploadty-disabled{.ant-upload-list-item {.anticon-close{display: none;}.anticon-delete{display: none;}}
}//---------------------------- begin 图片左右换位置 -------------------------------------.uploadty-mover-mask{background-color: rgba(0, 0, 0, 0.5);opacity: .8;color: #fff;height: 28px;line-height: 28px;}//---------------------------- end 图片左右换位置 -------------------------------------
</style>
总结

除了以上两个组件库1+1的方式,还有百度前端大神开发的vue-office组件库,而且优点也很明显:

  • 使用简单,对新手友好,只传递一个文件地址,就可实现预览。
  • 提供多种文件的一站式预览解决方案,解决常见的docx、excel、pdf三种文件的预览。
  • 预览效果也好,不只是对内容预览,也要支持样式。

预览的效果确实超级棒,可惜的是不支持打印功能,不能满足需求,可惜了。
有需要的可以去看vue-office的演示效果:
Vue框架演示效果
非Vue框架文件预览

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

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

相关文章

[golang 流媒体在线直播系统] 4.真实RTMP推流摄像头把摄像头拍摄的信息发送到腾讯云流媒体服务器实现直播

用RTMP推流摄像头把摄像头拍摄的信息发送到腾讯云流媒体服务器实现直播,该功能适用范围广,比如:幼儿园直播、农场视频直播, 一.准备工作 要实现上面的功能,需要准备如下设备: 推流摄像机&#xff08;监控&#xff09; 流媒体直播服务器(腾讯云流媒体服务器,自己搭建的流媒体服务…

MySQL 面试题——MySQL 基础

目录 1.什么是 MySQL&#xff1f;有什么优点&#xff1f;2.MySQL 中的 DDL 与 DML 是分别指什么&#xff1f;3.✨数据类型 varchar 与 char 有什么区别&#xff1f;4.数据类型 BLOB 与 TEXT 有什么区别&#xff1f;5.DATETIME 和 TIMESTAMP 的异同&#xff1f;6.✨MySQL 中 IN …

Json-Jackson和FastJson

狂神&#xff1a; 测试Jackson 纯Java解决日期格式化 设置ObjectMapper FastJson&#xff1a; 知乎&#xff1a;Jackson使用指南 1、常见配置 方式一&#xff1a;yml配置 spring.jackson.date-format指定日期格式&#xff0c;比如yyyy-MM-dd HH:mm:ss&#xff0c;或者具体的…

线程池(重点)

1.线程池的三大方法 package com.kuang.pool;import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; //Executors工具类 三大方法 //使用线程池&#xff0c;创建线程 public class Demo01 {public static void main(String[] args) { // …

Python + Jmeter 实现自动化性能压测

Step01: Python脚本开发 文件路径&#xff1a;D://wl//testproject//Fone-grpc//project1//test_client.py Python 脚本作用&#xff1a; 通过 grpc 调用底层 c 的接口&#xff0c;做数据库的数据插入与查询操作&#xff0c;然后将返回的结果进行拼接与输出。 2.代码里面将…

LeNet-5

目录 一、知识点 二、代码 三、查看卷积层的feature map 1. 查看每层信息 ​2. show_featureMap.py 背景&#xff1a;LeNet-5是一个经典的CNN&#xff0c;由Yann LeCun在1998年提出&#xff0c;旨在解决手写数字识别问题。 一、知识点 1. iter()next() iter()&#xff1a;…

阿里云无影云电脑介绍_云办公_使用_价格和优势说明

什么是阿里云无影云电脑&#xff1f;无影云电脑&#xff08;原云桌面&#xff09;是一种快速构建、高效管理桌面办公环境&#xff0c;无影云电脑可用于远程办公、多分支机构、安全OA、短期使用、专业制图等使用场景&#xff0c;阿里云百科分享无影云桌面的详细介绍、租用价格、…

【查缺补漏 女娲补天】2023平安

秋招了&#xff0c;只根据自己的情况记录&#xff0c;大概率不会很全。标题是我觉得的重点。既搬砖也搬博客。 Telnet协议 远程登录和管理网路设备的标准协议TCP传输层之上&#xff1a;应用层工作模型&#xff1a;C/S模式&#xff08;client/server&#xff09;服务端端口号默…

Python 数独求解器

文章目录 使用回溯算法在Python中解决数独总结 Sudoku&#xff08;数独&#xff09;是一种基于逻辑的数字填充谜题游戏&#xff0c;最受喜爱的是那些热爱逻辑和推理的人。解决数独谜题有助于提高集中注意力和逻辑思维能力。 本文介绍了如何使用Python解决数独谜题。 使用回溯算…

车联网远程监控管理提升车辆调度效率,实现高效运营

随着智慧城市建设与物联网技术发展&#xff0c;车辆使用4G工业路由器网络实现车联网&#xff0c;并对车上视频监控、GPS定位以及温湿度传感器等信息进行数据采集和实时传输。这些数据的采集和监测将通过4G网络上传到管理平台&#xff0c;为车辆调度和运行效率的优化提供了有力的…

计算机组成原理——基础入门总结(一)

本帖更新一些关于计算机组成原理的重点内容。由于博主考研时并不会考这门课&#xff0c;但是考虑到操作系统中又很多重要晦涩的概念涉及很多诸如内存、存储器、磁盘、cpu乃至各种寄存器的知识&#xff0c;此处挑选一些核心的内容总结复盘一遍——实现声明&#xff1a;本帖的内容…

【微信小程序】外卖点餐效果展示

概述 外卖点餐效果展示&#xff0c;左右布局&#xff0c;快速点餐&#xff0c;商家信息展示等...程序是模仿人家的&#xff0c;所以界面没做什么调整&#xff0c;功能是没啥问题&#xff0c;可以正常使用... 详细 直接看效果图&#xff1a; 可以把这个点餐这个功能分为5部分…

第13篇:ESP32 idf wifi联网使用SNTP同步网络时间LCD ST7920液晶屏显示

第1篇:Arduino与ESP32开发板的安装方法 第2篇:ESP32 helloword第一个程序示范点亮板载LED 第3篇:vscode搭建esp32 arduino开发环境 第4篇:vscodeplatformio搭建esp32 arduino开发环境 ​​​​​​第5篇:doit_esp32_devkit_v1使用pmw呼吸灯实验 第6篇:ESP32连接无源喇叭播…

看完这篇 教你玩转渗透测试靶机Vulnhub——Grotesque:2

Vulnhub靶机Grotesque&#xff1a;1.0.1渗透测试详解 Vulnhub靶机介绍&#xff1a;Vulnhub靶机下载&#xff1a;Vulnhub靶机安装&#xff1a;①&#xff1a;信息收集&#xff1a;②&#xff1a;暴力破解&#xff1a;③&#xff1a;SSH登入&#xff1a;④&#xff1a;提权&#…

day21算法

常见的七种查找算法&#xff1a; ​ 数据结构是数据存储的方式&#xff0c;算法是数据计算的方式。所以在开发中&#xff0c;算法和数据结构息息相关。今天的讲义中会涉及部分数据结构的专业名词&#xff0c;如果各位铁粉有疑惑&#xff0c;可以先看一下哥们后面录制的数据结构…

MapReduce YARN 的部署

1、部署说明 Hadoop HDFS分布式文件系统&#xff0c;我们会启动&#xff1a; NameNode进程作为管理节点DataNode进程作为工作节点SecondaryNamenode作为辅助 同理&#xff0c;Hadoop YARN分布式资源调度&#xff0c;会启动&#xff1a;ResourceManager进程作为管理节点NodeM…

uni-app 实现自定义按 A~Z 排序的通讯录(字母索引导航)

创建 convertPinyin.js 文件 convertPinyin.js 将下面的内容复制粘贴到其中 const pinyin (function() {let Pinyin function(ops) {this.initialize(ops);},options {checkPolyphone: false,charcase: "default"};Pinyin.fn Pinyin.prototype {init: functi…

JavaScript:二进制数组【笔记】

二进制数组【ArrayBuffer对象、Type的Array视图和DataView视图】JavaScript操作二进制数据的一个接口。 这些接口原本是和WebGL有关【WebGL是浏览器与显卡之间的通信接口】&#xff0c;为了满足JavaScript与显卡之间大量、实时数据交换&#xff0c;那么JavaScript和显卡之间的…

一款非常容易上手的报表工具,简单操作实现BI炫酷界面数据展示,驱动支持众多不同类型的数据库,可视化神器,免开源了

一款非常容易上手的报表工具&#xff0c;简单操作实现BI炫酷界面数据展示&#xff0c;驱动支持众多不同类型的数据库&#xff0c;可视化神器&#xff0c;免开源了。 在互联网数据大爆炸的这几年&#xff0c;各类数据处理、数据可视化的需求使得 GitHub 上诞生了一大批高质量的…

flink 端到端一致性

背景 我们经常会混淆flink提供的状态一致性保证和数据端到端一致性保证的关系&#xff0c;总以为他们表达的是同一个意思&#xff0c;事实上&#xff0c;他们不是一个含义&#xff0c;flink只能保证其维护的内部状态的一致性&#xff0c;而数据端到端的一致性需要数据源&#…