Spring Boot整合Minio实现文件上传

Spring Boot整合Minio后,前端的文件上传有两种方式:

  1. 文件上传到后端,由后端保存到Minio

  2. 这种方式好处是完全由后端集中管理,可以很好的做到、身份验证、权限控制、文件与处理等,并且可以做一些额外的业务逻辑,比如生成缩略图、提取元数据等。

  3. 缺点也很明显:

  4. 延迟时间高了,本来花费上传一次文件的时间,现在多了后端保存到Minio的时间

  5. 后端资源占用,后端本来可以只处理业务请求,现在还要负责文件流,增加了性能压力

  6. 单点故障,Minio即便做了集群,但是如果后端服务器故障,也会导致Minio不可用

  7. 所以,实际上我们不会把文件传到后端,而是直接传给Minio,其实这也符合OSS服务的使用方式。

  8. 文件向后端申请上传凭证,然后直接上传到Minio

  9. 为了避免Minio被攻击,我们需要结合后端,让后端生成并返回一个有时效的上传凭证,前端拿着这个凭证才能去上传,通过这种方式,我们可以做到一定程度的权限控制,本文要分享的就是这种方式。

环境准备

  • 部署好的Minio环境:http://mylocalhost:9001

Spring Boot整合Minio

简单过一下整合方式把。

先引入Minio依赖

# pom.xml 
<dependency> 
  <groupId>io.minio</group Id> 
<artifactId>minio</artifact Id> 
<version>7.1.0</version> 
</dependency>

然后定义配置信息

# application.yml
minio:
endpoint: http://mylocalhost:9001 
accessKey: minio 
secretKey: minio123 
bucket: demo

定义一个属性类

@Component 
@ConfigurationProperties(prefix = "minio")
public class MinioProperties { 
  /** * 对象存储服务的URL 
  */ 
  private String endpoint; 
  /** 
  * Access key就像用户ID,可以唯一标识你的账户 
  */ 
  private String accessKey;
  /** 
  * Secret key是你账户的密码
  */
  private String secretKey; 
  /**
  * 默认文件桶
  */ 
  private String bucket; ...
}

定义Minio配置类

@Configuration 
public class MinioConfig { 
  @Bean
  public MinioClient minioClient(MinioProperties properties){ 
    try { 
      MinioClient.Builder builder = MinioClient.builder(); 
      builder.endpoint(properties.getEndpoint());
      if (StringUtils.hasLength(properties.getAccessKey()) && StringUtils.hasLength(properties.getSecretKey())) { 
        builder.credentials(properties.getAccessKey(),properties.getSecretKey()); 
       } 
       return builder.build(); 
      } 
       catch (Exception e){ 
         return null;
       }
  }
}

现在启动服务即可。

上传凭证

写一个接口,返回上传凭证:

@RequestMapping(value = "/presign", method = {RequestMethod.POST}) 
public Map<String, String> presign(@RequestBody PresignParam presignParam) {
  //如果前端不指定桶,那么给一个默认的
  if (StringUtils.isEmpty(presignParam.getBucket())) {
    presignParam.setBucket("demo"); 
  } 

  // 前端不指定文件名称,就给一个UUID 
  if (StringUtils.isEmpty(presignParam.getFilename())) { 
    presignParam.setFilename(UUID.randomUUID().toString());
  } 

  // 如果想要以子目录的方式保存,就在前面加上斜杠来表示 
  //      presignParam.setFilename("/2023/" + presignParam.getFilename());

  // 设置凭证过期时间 
  ZonedDateTime expirationDate = ZonedDateTime.now().plusMinutes(10); 

  // 创建一个凭证 
  PostPolicy policy = new PostPolicy(presignParam.getBucket(), presignParam.getFilename(), expirationDate); 
  // 限制文件大小,单位是字节byte,也就是说可以设置如:只允许10M以内的文件上传 
  // policy.setContentRange(1, 10 * 1024); 
  // 限制上传文件请求的ContentType 
  // policy.setContentType("image/png");

  try {
  // 生成凭证并返回
    final Map<String, String> map = minioClient.presignedPostPolicy(policy);
    for (Map.Entry<String, String> entry : map.entrySet()) { 
      System.out.println(entry.getKey() + " = " + entry.getValue()); 
    } 
    return map; 
  } catch (MinioException | InvalidKeyException | IOException | NoSuchAlgorithmException e) { 
    e.printStackTrace();
  }
  return null;
}

上面的示例代码可以知道,我们还可以加一些权限认证,以判断用户是否有以下权限:

  • 上传权限

  • 可上传的文件大小

  • 可上传的文件类型

请求参数类:

public class PresignParam { 
  // 桶名
  private String bucket; 
  // 文件名 
  private String filename; ...
}

这个接口的返回结果是:

bucket: demo 
x-amz-date: 20230831T042351Z 
x-amz-signature: 79cc2ae0baee274d1d47cb29bdd5e99127059033503c2a02f904f0478a73ecac
key: 寂寞的季节.mp4 
x-amz-algorithm: AWS4-HMAC-SHA256
x-amz-credential: minio/20230831/us-east-1/s3/aws4_request 
policy: eyJleHBpcmF0aW9uIjoiMjAyMy0wOC0zMVQwNDozMzo1MS42MzZaIiwiY29uZGl0aW9ucyI6W1siZXEiLCIkYnVja2V0IiwiZGVtbyJdLFsiZXEiLCIka2V5Iiwi5a+C5a+e55qE5a2j6IqCLm1wNCJdLFsiZXEiLCIkeC1hbXotYWxnb3JpdGhtIiwiQVdTNC1ITUFDLVNIQTI1NiJdLFsiZXEiLCIkeC1hbXotY3JlZGVudGlhbCIsIm1pbmlvLzIwMjMwODMxL3VzLWVhc3QtMS9zMy9hd3M0X3JlcXVlc3QiXSxbImVxIiwiJHgtYW16LWRhdGUiLCIyMDIzMDgzMVQwNDIzNTFaIl1dfQ==
  • bucket:表示目标桶

  • x-amz-date:时间戳

  • x-amz-signature:签名

  • key:文件名

  • x-amz-algorithm:签名算法

  • x-amz-credential:认证授权

  • policy:凭证token

前端收到后,将该凭证连同文件流一并上传到Minio服务器:

uploadFile(file, policy) { 
  console.log("准备上传文件:") 
  console.log("file:" + file) 
  console.log("policy:" + policy) 
  var formData = new FormData() 

  formData.append('file', file) 
  formData.append('key', policy['key']) 
  formData.append('x-amz-algorithm', policy['x-amz-algorithm'])
  formData.append('x-amz-credential', policy['x-amz-credential']) 
  formData.append('x-amz-signature', policy['x-amz-signature']) 
  formData.append('x-amz-date', policy['x-amz-date']) 
  formData.append('policy', policy['policy'])

  return new Promise(((resolve, reject) => {
    $.ajax({
      method: 'POST',
      url: 'http://mylocalhost:9001/' + policy['bucket'],
      data: formData, 
      dataType: 'json', 
      contentType: false, // 必须设置为 false,不设置 contentType,让浏览器自动设置
      processData: false, // 必须设置为 false,不对 FormData 进行序列化处理 
      // async: false, // 设置同步,方便等下做分片上传
      xhr: function xhr() {
        //获取原生的xhr对象 
        var xhr = $.ajaxSettings.xhr(); 
        if (xhr.upload) { 
          //添加 progress 事件监听 
          xhr.upload.addEventListener('progress', function (e) {
            //e.loaded 已上传文件字节数 
            //e.total 文件总字节数
            var percentage = parseInt(e.loaded / e.total * 100) 
            vm.uploadResult = percentage + "%" + ":" + policy['key']
          }, false); 
        } 
        return xhr; 
      }, 
      success: function (result) { 
        vm.uploadResult = '文件上传成功:' + policy['key'] 
        resolve(result) 
      }, 
      error: function (e) { 
        reject()
      } 
    }) 
  }))
}

这样就完成了获取上传凭证并上传文件。

分片上传、秒传、断点续传

分片上传

分片上传可以用在大文件上传上,一个100M的文件可以分成10份,每份10M,一共传输10次,这有以下好处:

  • Minio做了集群,用Nginx转发,那么分片上传可以降低单台Minio服务器的性能压力

  • 多线程上传可以加快上传效率

秒传

现在说说秒传,我们上传一个文件之前,可以用工具生成MD5字符串,就好像这样:

3cc1f3c3c2d1a29ecf60ffad4de278c7

然后拼接上文件名:

3cc1f3c3c2d1a29ecf60ffad4de278c7_寂寞的季节.mp4

这时候去向后端申请上传凭证的时候,后端可以先去看看文件是否已存在,如果文件已存在,就不用生成凭证了,直接告诉前端该文件已经上传完毕,由此实现文件秒传。
这样的好处是:

  • 降低Minio服务器压力

  • 响应秒回,用户体验提高

断点续传

结合分片上传和秒传的原理,我们可以来做到断点续传。

场景:当我们要上传一个大文件的时候,进度到一半了,这时候网络掉线导致上传失败,网络恢复后又要重新上传,这就很崩溃。

处理方式:大文件也可以分成一个个小文件来上传,这样即便上传到一半网络掉线,恢复上传的时候可以跳过前一半已上传的部分,接着上传后面一半。

文件合并

当我们分片上传后,后端还需要提供接口,来将所有分片数据合并:

@GetMapping("/compose")
 public void merge() {
   List<ComposeSource> sources = new ArrayList<>();
   // 分片数据放到另一个桶里面:slice 
   sources.add(ComposeSource.builder() 
               .bucket("slice") 
               .object("0寂寞的季节.mp4")
               .build()); 
   sources.add(ComposeSource.builder()
               .bucket("slice") 
               .object("1寂寞的季节.mp4")
               .build());
   sources.add(ComposeSource.builder()
               .bucket("slice")
               .object("2寂寞的季节.mp4") 
               .build());
   final ComposeObjectArgs args = ComposeObjectArgs.builder() 
     .bucket("demo") 
     .object("寂寞的季节.mp4")
     .sources(sources)
     .build(); 

   try { 
     minioClient.composeObject(args);
   } 
   catch (MinioException | InvalidKeyException | IOException | NoSuchAlgorithmException e) {
     e.printStackTrace(); 
   } 
 }

上面的示例很简单,因为只做演示说明。
前端需要传的参数是:

  • 分片桶:slice

  • 分片数据数组: 0寂寞的季节.mp4 1寂寞的季节.mp4 2寂寞的季节.mp4

  • 目标桶:demo

然后调用composeObject函数完成合并。

前端示例代码分享

上面就是关于实战经验分享的全部了,因为需要前端配置来使用,所以这里给出我这篇文章的前端示例,很简单的单页面(技术栈就别吐槽了):

<!DOCTYPE html>
  <html lang="en"> 
    <head> 
    <meta charset="UTF-8">
      <title>Title</title> 
   <script src="https://cdn.bootcdn.net/ajax/libs/vue/2.7.14/vue.js">
   </script> <script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.4/jquery.js">
   </script> <script src="https://cdn.bootcdn.net/ajax/libs/spark-md5/3.0.2/spark-md5.min.js"></script> 
</head> 
<body> 
<div id="app"> 

<h1>{{title}}</h1>
<br> 

      <form @submit.prevent="getPolicyForm">

   <label> 
       桶名 
      <input type="text" v-model="policyParams.bucket">

    </label>
<br>
       <label> 
        文件名 
       <input type="text" v-model="policyParams.filename"> 
         </label> 
<br> 
         <button type="submit">获取上传凭证</button>
<br> 
           <div v-for="(val, key) in policy" :key="key">{{ key }}: <span>{{ val }}</span></div>
    </form> 
<br> 
             <form @submit.prevent="uploadFileForm" v-show="policy != null">
               <label> 文件 <input type="file" @change="fileChange">
                 </label> 
<br> 
                 <br>
                 <button type="submit" v-show="file != null">上传文件</button>
</form> --- 
<br> 
                   <div v-show="file != null"> <button @click="sliceEvent">测试文件分片上传</button> | <button @click="sliceComposeEvent">分片文件合并</button> </div>
<br>
              <br> 
            <br> 
              <p>{{uploadResult}}</p> 
<ul> <!-- <li v-for="item in sliceUploadResult">{{ item }}</li>--> <li v-for="(item, index) in sliceUploadResult" :key="index">{{ item }}</li> </ul> <br> </div> 
       <script> var vm = new Vue({
          el: "#app", data() { 
              return { title: "Minio测试" 
                 // 请求凭证参数 , policyParams: { bucket: null , filename: null } 
                 // 请求到的凭证 , policy: null 
                 // 待上传文件 , file: null
                 // 上传文件参数 , uploadParams: { file: null } 
                 // 分片上传参数 , sliceParams: { bucket: "" , filename: "" , file: null } , slicePolicys: [] , sliceCount: 0
                 // 上传结果回调 , uploadResult: null 
                 // 分片上传结果回调 , sliceUploadResult: null };
          }, methods: { getPolicyForm() {
           this.policyParams.bucket = "demo" this.policyParams.filename = "寂寞的季节.mp4" this.requestPolicy(this.policyParams)
             }, requestPolicy(params) { 
                 return new Promise(((resolve, reject) => { 
                   $.ajax({ type: "POST", url: "http://localhost:8888/presign", contentType: "application/json", data: JSON.stringify(params),
                     // async: false, success: function (result) { console.log(result) vm.policy = result; resolve(result) }, error: function (e) { reject() }
    });
   }))
  }, 
             fileChange(event) { 
               const file = event.target.files[0] this.file = file }, 
                 uploadFileForm() { this.uploadFile(this.file, this.policy) }, 
                   uploadFile(file, policy) { 
                     console.log("准备上传文件:") console.log("file:" + file) console.log("policy:" + policy) var formData = new FormData() formData.append('file', file) 
                     formData.append('key', policy['key']) 
                     formData.append('x-amz-algorithm', policy['x-amz-algorithm']) 
                     formData.append('x-amz-credential', policy['x-amz-credential']) 
                     formData.append('x-amz-signature', policy['x-amz-signature'])
                     formData.append('x-amz-date', policy['x-amz-date']) 
                     formData.append('policy', policy['policy']) 
                     return new
                     Promise(((resolve, reject) => { 
                       $.ajax({ method: 'POST', url: 'http://mylocalhost:9001/' + policy['bucket'], data: 
                               formData, dataType: 'json', contentType: false, 
                               // 必须设置为 false,不设置 contentType,让浏览器自动设置 processData: false,
                               // 必须设置为 false,不对 FormData 进行序列化处理 // async: false, 
                               // 设置同步,方便等下做分片上传 xhr: function xhr() { //获取原生的xhr对象 var xhr = $.ajaxSettings.xhr(); if (xhr.upload) { 
                               //添加 progress 事件监听 xhr.upload.addEventListener('progress', function (e) {
                               //e.loaded 已上传文件字节数 //e.total 文件总字节数 var percentage = parseInt(e.loaded / e.total * 100) vm.uploadResult = percentage + "%" + ":" + policy['key'] }, false);
                              } return xhr; 
                              }, success: function (result) { vm.uploadResult = '文件上传成功:' + policy['key'] resolve(result) }, 
                       error: function (e) { reject() } 
                      })
          }))
}, 
  sliceEvent() { 
    // 获取文件 var file = this.file
    // 设置分片大小:5MB var chunkSize = 5 * 1024 * 1024 
    // 计算总共有多少个分片 var totalChunk = Math.ceil(file.size / chunkSize)
    // 数组存放所有分片 var chunks = [] 
    // 遍历所有分片 for (var i = 0; i < totalChunk; i++) { 
    // 利用slice获取分片 var start = i * chunkSize var end = Math.min(file.size, start + chunkSize) var blob = file.slice(start, end)
    // 添加分片到数组 chunks.push(blob) } console.log(totalChunk) this.sliceUploadResult = Array(totalChunk).fill(0) for (let i = 0; i < chunks.length; i++) { var file = chunks[i]; this.calculateMD5(file) .then((md5) => { console.log(md5);
    // 输出计算出的 MD5 值 }) .catch((error) => { console.error(error); 
    // 处理错误 }); } return
    // 创建序号 var index = 0; 
    // 循环上传分片 while (index < totalChunk) { console.log('------------------------------') 
    params = { "bucket": "slice", "filename": index + "寂寞的季节.mp4" } var policyPromise = this.requestPolicy(params); (function (index) { var file = chunks[index] 
    policyPromise.then(function (result) { var filename = result['key'] console.log('准备上传文件:', filename, ',序号为:', index) vm.uploadFile(file, result).then(function (result) { console.log('上传完成:' + filename) vm.sliceUploadResult[index] = ('分片文件上传成功:' + filename) }) }) })(index) index++ } }, sliceComposeEvent() { var parmas = {} $.ajax({ method: 'POST', url: 'http://localhost:8888/compose', data: formData, dataType: 'json', contentType: false, 
    // 必须设置为 false,不设置 contentType,让浏览器自动设置 processData: false, 
    // 必须设置为 false,不对 FormData 进行序列化处理 
    // async: false, 
    // 设置同步,方便等下做分片上传 xhr: function xhr() { 
    //获取原生的xhr对象 var xhr = $.ajaxSettings.xhr(); if (xhr.upload) { 
    //添加 progress 事件监听 xhr.upload.addEventListener('progress', function (e) { 
    //e.loaded 已上传文件字节数 //e.total 文件总字节数 var percentage = parseInt(e.loaded / e.total * 100) vm.uploadResult = percentage + "%" + ":" + policy['key'] }, false);
    } return xhr; },
      success: function (result) { vm.uploadResult = '文件上传成功:' + policy['key'] resolve(result) },
        error: function (e) { reject() }
}) 
}, 
  calculateMD5(file) { return new Promise((resolve, reject) => { const reader = new FileReader(); 
            // 读取文件内容 reader.readAsArrayBuffer(file); reader.onload = () => { const spark = new SparkMD5.ArrayBuffer(); spark.append(reader.result); 
            // 将文件内容添加到 MD5 计算器中 const md5 = spark.end();
            // 计算 MD5 值 resolve(md5); }; 
                        reader.onerror = (error) => { reject(error); 
                     }; 
              });
         }
}, mounted() { 
}, created() {
}, 
});
</script> 
</body> 
</html>

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

目标检测文献阅读-DETR:使用Transformer进行端到端目标检测

目录 摘要 Abstract 1 引言 2 DETR结构 2.1 Backbone 2.2 Encoder 2.3 Decoder 2.4 FFN 3 目标检测集合预测损失 3.1 二分图匹配损失 3.2 损失函数 总结 摘要 本周阅读的论文题目是《End-to-End Object Detection with Transformers》(使用Transformer进行端到端目…

服务器双网卡NCCL通过交换机通信

1、NCCL变量设置 export CUDA_DEVICE_MAX_CONNECTIONS1 export NCCL_SOCKET_IFNAMEeno2 export NCCL_IB_DISABLE0 #export NCCL_NETIB export NCCL_IB_HCAmlx5_0,mlx5_1 export NCCL_IB_GID_INDEX3 export NCCL_DEBUGINFOGPUS_PER_NODE4MASTER_ADDR192.168.1.2 MASTER_PORT600…

B树及其Java实现详解

文章目录 B树及其Java实现详解一、引言二、B树的结构与性质1、节点结构2、性质 三、B树的操作1、插入操作1.1、插入过程 2、删除操作2.1、删除过程 3、搜索操作 四、B树的Java实现1、节点类实现2、B树类实现 五、使用示例六、总结 B树及其Java实现详解 一、引言 B树是一种多路…

数据分析思维(八):分析方法——RFM分析方法

数据分析并非只是简单的数据分析工具三板斧——Excel、SQL、Python&#xff0c;更重要的是数据分析思维。没有数据分析思维和业务知识&#xff0c;就算拿到一堆数据&#xff0c;也不知道如何下手。 推荐书本《数据分析思维——分析方法和业务知识》&#xff0c;本文内容就是提取…

微信小程序用的SSL证书有什么要求吗?

微信小程序主要建立在手机端使用&#xff0c;然而手机又涉及到各种系统及版本&#xff0c;所以对SSL证书也有要求&#xff0c;如果要小程序可以安全有效的访问需要满足以下要求&#xff1a; 1、原厂SSL证书&#xff08;原厂封&#xff09;。 2、DV单域名或者DV通配符。 3、兼…

手动安装 Maven 依赖到本地仓库

文章目录 手动安装 Maven 依赖到本地仓库1. 下载所需的 JAR 文件2. 安装 JAR 文件到本地仓库3. 验证安装4. 在项目中使用该依赖 手动安装 Maven 依赖到本地仓库 遇到的问题&#xff1a; idea导入一个新的工程&#xff0c;发现pom文件中的一些依赖死活下载不下来&#xff0c;这…

VSCode Live Server 插件安装和使用

VSCode Live Server是一个由Ritwick Dey开发的Visual Studio Code扩展插件&#xff0c;它提供了一个带有实时重载功能的本地开发服务器。在VSCode中安装和使用Live Server插件进行实时预览和调试Web应用程序。这将大大提高前端开发效率&#xff0c;使网页设计和开发变得更为流畅…

UART串口数据分析

串口基础知识详细介绍&#xff1a; 该链接详细介绍了串并行、单双工、同异步、连接方式 https://blog.csdn.net/weixin_43386810/article/details/127156063 该文章将介绍串口数据的电平变化、波特率计算、脉宽计算以及数据传输量的计算。 捕获工具&#xff1a;逻辑分析仪&…

Internet协议原理

文章目录 考试说明Chapter 0: 本书介绍Chapter 1: Introduction And Overview 【第1章&#xff1a;引言与概述】Chapter 2: Overview Of Underlying Network Technologies 【第2章&#xff1a;底层网络技术的回顾】Chapter 3: Internetworking Concept And Architectural Model…

DeepSeek-V3 通俗详解:从诞生到优势,以及与 GPT-4o 的对比

1. DeepSeek 的前世今生 1.1 什么是 DeepSeek&#xff1f; DeepSeek 是一家专注于人工智能技术研发的公司&#xff0c;致力于打造高性能、低成本的 AI 模型。它的目标是让 AI 技术更加普惠&#xff0c;让更多人能够用上强大的 AI 工具。 1.2 DeepSeek-V3 的诞生 DeepSeek-V…

linux之自动挂载

如果想要实现自动挂载&#xff0c;应该挂在客户端&#xff01;&#xff01;&#xff01;&#xff01;&#xff01; 客户端&#xff1a; [rootlocalhost ~]# yum install nfs-utils -y &#xff08;下载软件&#xff09; [rootlocalhost ~]# systemctl start nfs-utils.servic…

RHCSA知识点汇总

第0章&#xff1a;Linux基础入门 0.1 什么是计算机 计算机的组成&#xff1a; 控制器&#xff1a;是整个计算机的中枢神经&#xff0c;根据程序要求进行控制&#xff0c;协调计算机各部分工作及内存与外设的访问等。 输入设备&#xff1a;将文字、数据、程序和控制命令等信…

交响曲-24-3-单细胞CNV分析及聚类

CNV概述 小于1kb是常见的插入、移位、缺失等的变异 人体内包含<10% 的正常CNV&#xff0c;我们的染色体数是两倍体&#xff0c;正常情况下&#xff0c;只有一条染色体表达&#xff0c;另一条沉默&#xff0c;当表达的那条染色体发生CNV之后&#xff0c;表达数量就会成倍增加…

【Linux-多线程】POSIX信号量-基于环形队列生产消费模型

POSIX信号量 POSIX信号量和System V信号量作用相同&#xff0c;都是用于同步操作&#xff0c;达到无冲突的访问共享资源的目的。但POSIX可以用于线程间同步 1.快速认识信号量接口 POSIX信号量分为两种类型&#xff1a; 命名信号量&#xff08;Named Semaphores&#xff09;&…

Linux下文件操作相关接口

文章目录 一 文件是什么普通数据文件 二 文件是谁打开的进程用户 三 进程打开文件的相关的接口c语言标准库相关文件接口1. fopen 函数2. fread 函数3. fwrite 函数4. fclose 函数5. fseek 函数 linux系统调用接口1. open 系统调用2. creat 系统调用3. read 系统调用4. write 系…

UE蓝图节点备忘录

获取索引为0的玩家 获取视图缩放 反投影屏幕到世界 获取屏幕上的鼠标位置 对指定的物体类型进行射线检测 判断物体是否有实现某个接口 上面节点的完整应用 通过PlayerControlle获取相机相关数据 从相机处发射射线撞击物体从而获取物体信息 抽屉推拉功能 节点说明 ##门的旋转开关…

玩机搞机基本常识-------列举安卓机型一些不常用的adb联机命令

前面分享过很多 常用的adb命令&#xff0c;今天分享一些不经常使用的adb指令。以作备用 1---查看当前手机所有app包名 adb shell pm list package 2--查看当前机型所有apk包安装位置 adb shell pm list package -f 3--- 清除指定应用程序数据【例如清除浏览器应用的数据】 …

LeetCode【剑指offer】系列(字符串篇)

剑指offer05.替换空格 题目链接 题目&#xff1a;假定一段路径记作字符串path&#xff0c;其中以 “.” 作为分隔符。现需将路径加密&#xff0c;加密方法为将path中的分隔符替换为空格" "&#xff0c;请返回加密后的字符串。 思路&#xff1a;遍历即可。 通过代…

idea java.lang.OutOfMemoryError: GC overhead limit exceeded

Idea build项目直接报错 java: GC overhead limit exceeded java.lang.OutOfMemoryError: GC overhead limit exceeded 设置 编译器 原先heap size 设置的是 700M , 改成 2048M即可

aws(学习笔记第二十二课) 复杂的lambda应用程序(python zip打包)

aws(学习笔记第二十二课) 开发复杂的lambda应用程序(python的zip包) 学习内容&#xff1a; 练习使用CloudShell开发复杂lambda应用程序(python) 1. 练习使用CloudShell CloudShell使用背景 复杂的python的lambda程序会有许多依赖的包&#xff0c;如果不提前准备好这些python的…