linux 安装sftp及使用sftp工具类上传和下载

news2024/12/28 22:14:08

一、centos7 安装sftp

1.安装 OpenSSH 服务:

sudo yum install openssh-server

2.启动 SSH 服务,并设置为开机启动:

sudo systemctl start sshd
sudo systemctl enable sshd

3.创建一个新用户,用于SFTP连接(替换your_username为你想要的用户名),设置密码(替换sftpadmin为你想要的密码):

sudo adduser your_username
sudo passwd sftpadmin

注意:如果输入密码提示未知的用户,则把sftpadmin换成指定的用户,然后再设置新密码
在这里插入图片描述

4.创建一个目录,用于SFTP的根目录(替换/path/to/sftp_root为你想要的目录路径):

sudo mkdir -p /path/to/sftp_root
sudo chown root:root /path/to/sftp_root
sudo chmod 755 /path/to/sftp_root

5.创建用户的SFTP目录:

sudo mkdir -p /path/to/sftp_root/your_username
sudo chown your_username:your_username /path/to/sftp_root/your_username
sudo chmod 755 /path/to/sftp_root/your_username

6.重启 SSH 服务以应用更改:

sudo systemctl restart sshd

现在,用户 your_username 可以通过 SFTP 连接到服务器,只需要使用他们的常规用户名和密码。确保为用户设置了合适的权限和所需的目录结构。

二、java sftp工具类

1.工具类

(1)引入manven

<dependency>
    <groupId>com.github.mwiede</groupId>
    <artifactId>jsch</artifactId>
    <version>0.2.18</version>
</dependency>

(2)sftp工具类

package com.dcqq.common.utils;

import cn.hutool.core.codec.Base64;
import com.google.common.base.Joiner;
import com.jcraft.jsch.Channel;
import com.jcraft.jsch.ChannelSftp;
import com.jcraft.jsch.JSch;
import com.jcraft.jsch.Session;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.RandomUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.time.DateFormatUtils;

import java.io.*;
import java.util.*;

/**
 * SFTP工具类
 */
@Slf4j
public class SftpUtils {
    private static Session sshSession = null;
    private static String sftpRootDir = "/home/sftp/upload/";//根目录
    private final static String host = "10.122.156.10";//ip
    private final static Integer port = 22;//端口
    private final static String username = "sftp_user";//用户名
    private final static String password = "sftp1122";//密码
    private final static int MAX_TIMES = 3;//重试次数

    public static ChannelSftp connect() {
        // 重试次数
        int count = 0;
        ChannelSftp sftp = SftpUtils.connect(host, port, username, password);
        while (sftp == null && count < MAX_TIMES) {
            count++;
            log.error("第[" + count + "]次重试连接FTP服务器:" + host);
            SftpUtils.sleep(RandomUtils.nextInt(3, 600));
            sftp = SftpUtils.connect(host, port, username, password);
        }
        if (sftp == null) {
            log.error("获取FTP连接失败:" + host);
        }
        return sftp;
    }

    /**
     * 获取ChannelSftp
     */
    public static ChannelSftp connect(String host, Integer port, String username, String password) {
        ChannelSftp sftp = null;
        try {
            JSch jsch = new JSch();
            jsch.getSession(username, host, port);
            sshSession = jsch.getSession(username, host, port);
            sshSession.setPassword(password);
            Properties sshConfig = new Properties();
            sshConfig.put("StrictHostKeyChecking", "no");
            sshSession.setConfig(sshConfig);
            sshSession.connect();
            Channel channel = sshSession.openChannel("sftp");
            channel.connect();
            sftp = (ChannelSftp) channel;
            SftpUtils.mkdirs(sftpRootDir, sftp);
            SftpUtils.cd(sftpRootDir, sftp);
//            System.out.println("连接成功");
            return sftp;
        } catch (Exception e) {
//            e.printStackTrace();
            log.error("连接FTP失败:" + host + ",异常信息:" + e);

        }
        return null;
    }

    public static boolean upload(File file, String directory, String fileName) {
        InputStream inputStream = null;
        try {
            inputStream = new FileInputStream(file);
            return upload(inputStream, directory, fileName);
        } catch (Exception e) {
            e.printStackTrace();
            log.error("FTP文件上传失败:" + directory + "/" + fileName + ",异常标题:" + e.getMessage());
        } finally {
            CloseUtils.closeQuietly(inputStream);
        }
        return false;
    }

    public static boolean upload(byte[] data, String directory, String fileName) {
        ByteArrayInputStream inputStream = null;
        try {
            inputStream = new ByteArrayInputStream(data);
            return upload(inputStream, directory, fileName);
        } finally {
            CloseUtils.closeQuietly(inputStream);
        }
    }

    public static boolean upload(InputStream inputStream, String directory, String fileName) {
        try {
            // 重试次数
            int count = 0;
            boolean success = doUpload(inputStream, directory, fileName);
            while (!success && count < MAX_TIMES) {
                count++;
                log.info("第[{}/{}]次重新上传文件:{}", count, MAX_TIMES, fileName);
                SftpUtils.sleep(RandomUtils.nextInt(1, 300));
                success = doUpload(inputStream, directory, fileName);
            }
            return success;
        } finally {
            CloseUtils.closeQuietly(inputStream);
        }
    }

    private static boolean doUpload(InputStream inputStream, String directory, String fileName) {
        ChannelSftp sftp = SftpUtils.connect();
        if (sftp == null) {
            return false;
        }
        try {
            SftpUtils.mkdirs(directory, sftp);
            sftp.put(inputStream, fileName, ChannelSftp.OVERWRITE);
            return true;
        } catch (Exception e) {
//            throw new RuntimeException(e);
            log.error("FTP文件上传失败:" + directory + "/" + fileName + ",异常标题:" + e.getMessage());
        } finally {
            SftpUtils.disconnect(sshSession);
            SftpUtils.disconnect(sftp);
        }
        return false;
    }

    public static String downloadAsBase64(String fileName) {
        ByteArrayOutputStream outputStream = download(fileName);
        try {
            if (outputStream == null || outputStream.toByteArray().length == 0) {
                return "";
            }
            return Base64.encode(outputStream.toByteArray());
        } finally {
            CloseUtils.closeQuietly(outputStream);
        }
    }

    public static ByteArrayOutputStream download(String fileName) {
        if (StringUtils.isEmpty(fileName)) {
            return null;
        }
        // 重试次数,最多6次
        int count = 0;
        ByteArrayOutputStream outputStream = doDownload(fileName);
        while (outputStream == null && count < MAX_TIMES) {
            count++;
            log.info("第[{}/2]次重新下载文件:{}", count, fileName);
            SftpUtils.sleep(RandomUtils.nextInt(1, 30));
            outputStream = doDownload(fileName);
        }
        return outputStream;
    }

    public static ByteArrayOutputStream doDownload(String fileName) {
        if (StringUtils.isEmpty(fileName)) {
            return null;
        }
        ChannelSftp sftp = SftpUtils.connect();
        try {
            ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
            sftp.get(fileName, outputStream);
            return outputStream;
        } catch (Exception e) {
//            throw new RuntimeException(e);
            log.error("FTP文件下载失败:" + fileName + ",异常标题:" + e.getMessage());
            if ("No such file.".equals(e.getMessage())) {
                return new ByteArrayOutputStream();
            }
        } finally {
            SftpUtils.disconnect(sshSession);
            SftpUtils.disconnect(sftp);
        }
        return null;
    }

    public static ChannelSftp cd(String directory, ChannelSftp sftp) {
        try {
            sftp.cd(directory);
        } catch (Exception e) {
        }
        return sftp;
    }

    public static ChannelSftp mkdir(String directory, ChannelSftp sftp) {
        try {
            sftp.mkdir(directory);
        } catch (Exception e) {
        }
        return sftp;
    }

    public static void disconnect(Session sshSession) {
        if (sshSession != null) {
            try {
                sshSession.disconnect();
            } catch (Exception e) {
            }
        }
    }

    public static void disconnect(ChannelSftp sftp) {
        if (sftp != null) {
            try {
                sftp.disconnect();
            } catch (Exception e) {
            }
        }
    }


    public static void mkdirs(String directory, ChannelSftp sftp) {

        String[] dirs = directory.split("/");
        for (int i = 0; i < dirs.length; i++) {
            String d = dirs[i];
            if (d.length() > 0) {

                SftpUtils.mkdir(d, sftp);
                SftpUtils.cd(d, sftp);
            }
        }
    }

    public static String datePath(String moduleName, Date dateTime) {
        return Joiner.on("/").join(moduleName, DateFormatUtils.format(dateTime, "yyyy/MM/dd"));
    }

    public static String datePath(String moduleName) {
        return Joiner.on("/").join(moduleName, DateFormatUtils.format(System.currentTimeMillis(), "yyyy/MM/dd"));
    }

    public static String fileFullPath(String filePath, String fileName) {
        return Joiner.on("/").join(filePath, fileName);
    }

    /**
     * 删除文件
     *
     * @param directory  要删除文件所在目录
     * @param deleteFile 要删除的文件
     * @throws Exception
     */
    public void delete(String directory, String deleteFile) throws Exception {
        ChannelSftp sftp = SftpUtils.connect();
        if (sftp != null) {
            sftp.cd(directory);
            sftp.rm(deleteFile);
        }
    }

    /**
     * 列出目录下的文件
     *
     * @param directory 要列出的目录
     * @return list 文件名列表
     * @throws Exception
     */
    @SuppressWarnings("unchecked")
    public static List<String> listFiles(String directory) throws Exception {
        ChannelSftp sftp = SftpUtils.connect();
        Vector fileList;
        List<String> fileNameList = new ArrayList<String>();
        fileList = sftp.ls(directory);
        Iterator it = fileList.iterator();
        while (it.hasNext()) {
            String fileName = ((ChannelSftp.LsEntry) it.next()).getFilename();
            if (".".equals(fileName) || "..".equals(fileName)) {
                continue;
            }
            fileNameList.add(fileName);
        }
        return fileNameList;
    }

    /**
     * @param millis *         the length of time to sleep in milliseconds
     */
    public static void sleep(long millis) {
        try {
            Thread.sleep(millis);
        } catch (Exception e) {
        }
    }

    public static void main(String[] args) throws Exception {
        //this.delete("/userReport/1/2022/ //删除
        //显示目录
        /*List<String> list = listFiles(sftpRootDir);
        for (String s : list) {
            System.out.println(s);
        }*/

        //上传文件
        File reportFile = new File("C:\\图片\\1.png");
        upload(reportFile, "upload", "1.png");

    }
}

(3)相关代码

CloseUtils

package com.dcqq.common.utils;

import java.io.Closeable;
import java.util.zip.ZipInputStream;

/**
 * 关闭文件流
 *
 */
public class CloseUtils {

    /**
     * 关闭文件流
     *
     * @param closeables
     */
    public static void closeQuietly(Closeable... closeables) {
        if (closeables != null && closeables.length > 0) {

            for (Closeable closeable : closeables) {

                if (closeable != null) {
                    try {
                        closeable.close();
                    } catch (Exception e) {}
                }
            }
        }
    }

    public static void closeEntryQuietly(ZipInputStream... closeables) {
        if (closeables != null && closeables.length > 0) {
            for (ZipInputStream closeable : closeables) {
                if (closeable != null) {
                    try {
                        closeable.closeEntry();
                    } catch (Exception e) { }
                }
            }
        }
    }
}

2.使用

(1)controller层,CommFileManagerController 公共文件管理层

package com.dcqq.common;

import com.dcqq.system.service.IFileService;
import com.dcqq.web.controller.common.CommonController;
import io.swagger.annotations.Api;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * 文件下载
 */
@RestController
@Api(tags = "文件下载")
@RequestMapping("/fileManager")
public class CommFileManagerController {
    private static final Logger log = LoggerFactory.getLogger(CommFileManagerController .class);
    
    @Autowired
    private IFileService fileService;
    @Autowired
    private ServerConfig serverConfig;

	/**
     * 文件上传
     */
    @ApiOperation(value = "文件上传", notes = "文件上传")
    @PostMapping("/uploadFile")
    public AjaxResult uploadFile(MultipartFile file) {
        FileForm form = new FileForm();
        form.setUrl(serverConfig.getUrl());
        return fileService.uploadFile(file, form);
    }

    /**
     * 文件读取
     * @param request
     * @param response
     * @throws IOException
     *  访问示例:https://test.tt.com.cn/prod-api/fileManager/upload/2024/06/18/1_20240618103039A005.png
     */
    @GetMapping(value = "/upload/**")
    public void upload(HttpServletRequest request, HttpServletResponse response){
        log.info("============upload");
        fileService.download(request,response);
    }
}

(2)service业务层

IFileService

package com.dcqq.system.service;

import com.dcqq.common.core.domain.AjaxResult;
import com.dcqq.form.FileForm;
import org.springframework.web.multipart.MultipartFile;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * 文件
 * 
 * @author dcqq
 * @date 2024-04-03
 */
public interface IFileService {

    /**
     * 文件上传
     * @param file
     * @return
     */
    AjaxResult uploadFile(MultipartFile file, FileForm form);
    
    /**
     * 文件下载
     * @param request
     * @param response
     */
    void download(HttpServletRequest request, HttpServletResponse response);
}

FileServiceImpl

package com.dcqq.common;

import cn.hutool.core.io.file.FileNameUtil;
import cn.hutool.core.net.URLDecoder;
import com.dcqq.common.core.domain.AjaxResult;
import com.dcqq.common.utils.*;
import com.dcqq.common.utils.file.MimeTypeUtils;
import com.dcqq.form.FileForm;
import com.dcqq.system.service.IFileService;
import org.apache.commons.io.FilenameUtils;
import org.springframework.stereotype.Service;
import org.springframework.web.multipart.MultipartFile;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.Objects;

/**
 * 文件
 * 
 * @author dcqq
 * @date 2024-04-03
 */
@Service
public class FileServiceImpl implements IFileService {

    public AjaxResult uploadFile(MultipartFile file, FileForm form) {
        try {
            String url = form.getUrl()+"/prod-api"+"/fileManager/upload";
            String dir=DateUtils.datePath();
            
            //获取文件后缀名
			String extension = FilenameUtils.getExtension(file.getOriginalFilename());
	        if (StringUtils.isEmpty(extension)) {
	            extension = MimeTypeUtils.getExtension(Objects.requireNonNull(file.getContentType()));
	        }

			//获取文件名+后缀
			String fileName = file.getOriginalFilename()+"."+extension;
			
            SftpUtils.upload(file.getBytes(), dir, fileName);
            
            AjaxResult ajax = AjaxResult.success();
			
			url=url+dir+fileName;

            ajax.put("url", url);
            ajax.put("newFileName", fileName);
            ajax.put("originalFilename", file.getOriginalFilename());
            return ajax;
        } catch (Exception e) {
            return AjaxResult.error(e.getMessage());
        }
    }

    @Override
    public void download(HttpServletRequest request, HttpServletResponse response) {
        try {
            String requestURI = URLDecoder.decode(request.getRequestURI(), StandardCharsets.UTF_8);
            // ContentType,即告诉客户端所发送的数据属于什么类型
            String ext = FileNameUtil.extName(requestURI);
            MediaTypeEnum mediaType = MediaTypeEnum.byCode(ext);
            response.setContentType(mediaType.getName());
            String ftpPath = StringUtils.substringAfter(requestURI, "/download/");
            ByteArrayOutputStream out = SftpUtils.download(ftpPath);
            response.getOutputStream().write(out.toByteArray());
            response.flushBuffer();
            CloseUtils.closeQuietly(out, response.getOutputStream());
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }
}

(3)相关代码

MediaTypeEnum

package com.dcqq.common.utils;

import lombok.Getter;
import org.springframework.http.MediaType;

import java.util.HashMap;
import java.util.Map;

/**
 * 媒体类型
 */
@Getter
public enum MediaTypeEnum {
    PNG("png", MediaType.IMAGE_PNG_VALUE),
    JPG("jpg", "image/jpg"),
    JPEG("jpeg", MediaType.IMAGE_JPEG_VALUE),
    PDF("pdf", MediaType.APPLICATION_PDF_VALUE),
    ZIP("zip", "application/zip"),
    DOCX("docx", MediaType.APPLICATION_OCTET_STREAM_VALUE),
    STREAM("other", MediaType.APPLICATION_OCTET_STREAM_VALUE),
    XML("xml", MediaType.APPLICATION_XML_VALUE),
    JSON("json", MediaType.APPLICATION_JSON_VALUE)
    ;
    private String code;
    private String name;

    private static Map<String, MediaTypeEnum> cacheEnum = new HashMap<>();

    static {
        for(MediaTypeEnum dictEnum:MediaTypeEnum.values()){
            cacheEnum.put(dictEnum.code, dictEnum);
        }
    }
    public static MediaTypeEnum byCode(String code){
        if(cacheEnum.containsKey(code)){
            return cacheEnum.get(code);
        }
        return STREAM;
    }

    MediaTypeEnum(String code, String name){
        this.code = code;
        this.name = name;
    }
}

ServerConfig

package com.dcqq.framework.config;

import javax.servlet.http.HttpServletRequest;

import org.springframework.stereotype.Component;
import com.dcqq.common.utils.ServletUtils;

/**
 * 服务相关配置
 *
 * @author ruoyi
 */
@Component
public class ServerConfig {
    /**
     * 获取完整的请求路径,包括:域名,端口,上下文访问路径
     *
     * @return 服务地址
     */
    public String getUrl() {
        HttpServletRequest request = ServletUtils.getRequest();
        return getDomain(request);
    }

    public static String getDomain(HttpServletRequest request) {
        StringBuffer url = request.getRequestURL();
        String contextPath = request.getServletContext().getContextPath();
        return url.delete(url.length() - request.getRequestURI().length(), url.length()).append(contextPath).toString();
    }
}

FileForm

package com.dcqq.form;

import io.swagger.annotations.ApiModelProperty;
import lombok.Data;

@Data
public class FileForm {
    @ApiModelProperty("请求路径")
    private String url;
}

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

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

相关文章

Linux:多线程中的互斥与同步

多线程 线程互斥互斥锁互斥锁实现的原理封装原生线程库封装互斥锁 死锁避免死锁的四种方法 线程同步条件变量 线程互斥 在多线程中&#xff0c;如果存在有一个全局变量&#xff0c;那么这个全局变量会被所有执行流所共享。但是&#xff0c;资源共享就会存在一种问题&#xff1…

Ilya出走记:SSI的超级安全革命

图片&#xff5c;OpenAI官网 ©自象限原创 作者丨罗辑、程心 和OpenAI分道扬镳以后&#xff0c;Ilya“神秘而伟大”的事业终于揭开了面纱。 6月20日&#xff0c;前OpenAI核心创始人 Ilya Stuskever&#xff0c;在官宣离职一个月后&#xff0c;Ilya在社交媒体平台公开了…

opencascade AIS_InteractiveContext源码学习2

AIS_InteractiveContext 前言 交互上下文&#xff08;Interactive Context&#xff09;允许您在一个或多个视图器中管理交互对象的图形行为和选择。类方法使这一操作非常透明。需要记住的是&#xff0c;对于已经被交互上下文识别的交互对象&#xff0c;必须使用上下文方法进行…

神经网络学习5-非线性激活

非线性激活&#xff0c;即 这是最常用的 inplaceTrue 原位操作 改变变量本身的值&#xff0c;就是是否输入时若原本有值&#xff0c;是否更换 该函数就是表示&#xff1a;输入小于零时输出0&#xff0c;大于零时保持不变 代码如下&#xff1a; import torch from torch imp…

芋道源码 yudao-cloud 、Boot 文档,开发指南 看全部,破解[芋道快速开发平台 Boot + Cloud]

1、文档全部保存本地部署查看&#xff0c;真香 文档已抓取最新版本&#xff0c;2024.06.21。【唯一遗憾&#xff0c;表结构到2024.04月&#xff0c;已被限制放到知识星球】会员中心&#xff0c;支付中心&#xff0c;CRM&#xff0c;ERP&#xff0c;商城&#xff0c;公众号运行…

利氪科技拿下C轮超级融资,国产智能底盘黑马奔向黄金时代

“智能驾驶遗珠&#xff0c;国产替代富矿。” 这是海通证券在最近一期研报中&#xff0c;描述线控底盘产业的用语。它很巧妙地点明了&#xff0c;这个藏在车身之下的部分&#xff0c;拥有何种特征——稳坐技术体系的核心点位&#xff0c;拥有前景广阔的市场。 事实上&#xf…

生成式AI与开发者:威胁还是机遇?

近期&#xff0c;围绕生成式人工智能&#xff08;AI&#xff09;是否能取代程序员的讨论达到了前所未有的高度。百度的创始人李彦宏甚至预言&#xff0c;未来可能不再需要程序员这一职业。这个话题让很多开发者&#xff0c;包括有几年开发经验的我&#xff0c;感到不安。我记得…

【ArcGIS微课1000例】0120:ArcGIS批量修改符号的样式(轮廓)

ArcGIS可以批量修改符号的样式,如样式、填充颜色、轮廓等等。 文章目录 一、加载实验数据二、土地利用符号化三、批量修改符号样式四、注意事项一、加载实验数据 订阅专栏后,从私信查收专栏配套的完整实验数据包,打开0120.rar中的土地利用数据,如下图所示: 查看属性表: …

Python Web实战:Python+Django+MySQL实现基于Web版的增删改查

项目实战 1.创建项目(sms) File->New Project->Django 稍等片刻&#xff0c;项目的目录结构如下图 项目创建后确认是否已安装Django和mysqlclient解释器&#xff0c;如何确认&#xff1f;file->Settings 如果没有请在Terminal终端输入以下命令完成安装 pip instal…

德璞资本:科技股波动解析,三巫日与日元效应下的市场走向

摘要 近期&#xff0c;美国科技股的表现令人担忧&#xff0c;标普500指数在科技股的拖累下出现下跌。亚洲股市也受到影响&#xff0c;特别是日本和韩国股市。随着期权到期日的临近&#xff0c;市场面临更大的波动风险。本文将详细分析科技股失去动能的原因、三巫日的影响及未来…

elementui组件库实现电影选座面板demo

<!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>Cinema Seat Selection</title><!-- 引入E…

华为重磅官宣:超9亿台、5000个头部应用已加入鸿蒙生态!人形机器人现身 专注AI芯片!英伟达挑战者Cerebras要上市了

内容提要 华为表示&#xff0c;盘古大模型5.0加持&#xff0c;小艺能力全新升级。小艺智能体与导航条融为一体&#xff0c;无处不在&#xff0c;随时召唤。只需将文字、图片、文档“投喂”小艺&#xff0c;即可便捷高效处理文字、识别图像、分析文档。 正文 据华为终端官方微…

策略模式:applicationContext.getBeansOfType()方法

applicationContext.getBeansOfType() 一般用来获取某个接口的所有实例Bean 方法定义如下&#xff1a; 入参一般是接口&#xff0c;即interface。响应是个Map结构&#xff0c;key bean在容器中的名称&#xff0c;value bean实列 开发步骤&#xff1a; 1.定义接口 2.定义…

“锟斤拷,烫烫烫,屯屯屯”的由来

在程序开发过程中&#xff0c;调试是不可或缺的一环。调试不仅可以帮助开发者发现错误&#xff0c;还能提供程序运行时的内部状态信息。然而&#xff0c;在调试过程中&#xff0c;开发者有时会遇到一些奇怪的字符。这些乱码通常是由内存状态的特殊标记&#xff0c;或者字符集不…

如何通过准确预测需求来减少PMC成为“夹心饼干”的风险?

在瞬息万变的商业环境中&#xff0c;产品物料控制&#xff08;PMC&#xff09;部门时常扮演着“夹心饼干”的角色&#xff0c;既要满足市场快速变化的需求&#xff0c;又要协调供应商、生产线等多方利益。如何减少这种风险&#xff0c;让PMC部门从“夹心困境”中脱颖而出&#…

【Git】 -- Part1 -- 基础操作

1. Git简介 Git 是一个开源的分布式版本控制系统&#xff0c;由 Linus Torvalds 于 2005 年开发&#xff0c;主要用于源代码管理。Git 允许多名开发者共同合作处理同一个项目&#xff0c;跟踪每个文件的修改&#xff0c;并且在必要时回滚到之前的版本。 Linus Torvalds是Linux…

使用 Kubernetes 部署 MinIO 和 Trino

Trino&#xff08;以前称为 Presto&#xff09;是一个 SQL 查询引擎&#xff0c;而不是 SQL 数据库。Trino 避开了 SQL 数据库的存储组件&#xff0c;只专注于一件事 - 超快的 SQL 查询。Trino 只是一个查询引擎&#xff0c;不存储数据。相反&#xff0c;Trino与各种数据库交互…

怎么添加网页到桌面快捷方式?

推荐用过最棒的学习网站&#xff01;https://offernow.cn 添加网页到桌面快捷方式&#xff1f; 很简单&#xff0c;仅需要两步&#xff0c;接下来以chrome浏览器为例。 第一步 在想要保存的网页右上角点击设置。 第二步 保存并分享-创建快捷方式&#xff0c;保存到桌面即可…

rancher快照备份至S3

巧用rancher的S3快照备份功能&#xff0c;快速实现集群复制、集群转移、完全崩溃后的极限修复 1.进入集群管理&#xff0c;在对应的集群菜单后&#xff0c;点击编辑配置 2.选择ETCD&#xff0c;启用&#xff0c;Backup Snapshots to S3选项 并填入你的minio 3 配置成功后 手…

【Linux基础IO】深入理解缓冲区

缓冲区在文件操作的过程中是比较重要的&#xff0c;理解缓冲区向文件刷新内容的原理可以更好的帮助我们更深层的理解操作系统内核对文件的操作。 FILE 因为IO相关函数与系统调用接口对应&#xff0c;并且库函数封装系统调用&#xff0c;所以本质上&#xff0c;访问文件都是通过…