Springboot整合Jsch-Sftp

背景

  开发一个基于jsch的sftp工具类,方便在以后的项目中使用。写代码的过程记录下来,作为备忘录。。。

Maven依赖

  • springboot依赖
<parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>2.7.0</version>
</parent>
  • jsch依赖
 <dependency>
    <groupId>com.jcraft</groupId>
    <artifactId>jsch</artifactId>
    <version>0.1.55</version>
</dependency>
  • 完整依赖
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>org.example</groupId>
    <artifactId>jsch-sftp</artifactId>
    <version>1.0-SNAPSHOT</version>

    <properties>
        <maven.compiler.source>8</maven.compiler.source>
        <maven.compiler.target>8</maven.compiler.target>
    </properties>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.7.0</version>
    </parent>

    <dependencies>

        <!-- web应用相关 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <!-- 单元测试 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
        </dependency>

        <!-- jsch SFTP -->
        <dependency>
            <groupId>com.jcraft</groupId>
            <artifactId>jsch</artifactId>
            <version>0.1.55</version>
        </dependency>

        <!-- lombok -->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>

    </dependencies>

</project>

项目结构

在这里插入图片描述

Sftp连接工具类

  • JschSftpRun: 项目启动类
package cn.com.soulfox;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;

/**
 * 启动类
 *
 * @author xxxx
 * @create 2024/7/5 11:50
 */
@SpringBootApplication
public class JschSftpRun {

    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(JschSftpRun.class, args);
    }
}

  • JschSftpConneciton: 创建sftp连接工具类
      用于创建sftp连接,类中提供了创建sftp连接的方法,另外还提供了在创建sftp连接失败后,重新尝试创建连接的方法
package cn.com.soulfox.jsch;

import com.jcraft.jsch.*;
import lombok.extern.slf4j.Slf4j;

/**
 * @author xxxx
 * @create 2024/7/4 12:15
 */
@Slf4j
public class JschSftpConneciton {

    private static final String SFTP_PROTOCOL = "sftp";
    private static final Integer DEFAULT_RETRY_COUNT_5 = 5;
    private static final Long SLEEP_ITME_1000 = 1000L;//每次重试的时间间隔

    private String host;
    private Integer port;
    private String username;
    private String password;
    private Integer sessionTimeout;
    private Integer connectionTimeout;

    public JschSftpConneciton setHost(String host) {
        this.host = host;
        return this;
    }

    public JschSftpConneciton setPort(Integer port) {
        this.port = port;
        return this;
    }

    public JschSftpConneciton setUsername(String username) {
        this.username = username;
        return this;
    }

    public JschSftpConneciton setPassword(String password) {
        this.password = password;
        return this;
    }

    public JschSftpConneciton setSessionTimeout(Integer sessionTimeout) {
        this.sessionTimeout = sessionTimeout;
        return this;
    }

    public JschSftpConneciton setConnectionTimeout(Integer connectionTimeout) {
        this.connectionTimeout = connectionTimeout;
        return this;
    }

    /**
     * 返回SftpWrapper对象,是为了方便释放Session,Channel,ChannelSftp资源
     * SFTP连接服务器
     */
    public SftpWrapper connect() throws  Exception {
        JSch jsch = new JSch();
        Session session = null;
        Channel channel = null;
        ChannelSftp sftp = null;
        try {

            session = jsch.getSession(username, host, port);
            if (session == null) {
                throw new JSchException("create session error");
            }
            //设置登陆主机的密码
            session.setPassword(password);
            //设置第一次登陆的时候提示,可选值:(ask | yes | no)
            session.setConfig("StrictHostKeyChecking", "no");
            //设置登陆超时时间
            session.connect(sessionTimeout);
            session.sendKeepAliveMsg();
            //创建sftp通信通道
            channel = session.openChannel(SFTP_PROTOCOL);
            channel.connect(connectionTimeout);
            sftp = (ChannelSftp) channel;

            return new SftpWrapper(session, channel, sftp);
        }catch (JSchException e){
            log.error("SFTP连接异常:", e);
            if (sftp != null ) {
                sftp.disconnect();
            }
            if (channel != null ) {
                channel.disconnect();
            }
            if (session != null ) {
                session.disconnect();
            }
            throw e;
        } catch (Exception e1) {
            log.error("SFTP连接异常:", e1);
            if (sftp != null ) {
                sftp.disconnect();
            }
            if (channel != null ) {
                channel.disconnect();
            }
            if (session != null ) {
                session.disconnect();
            }
            throw e1;
        }
    }

    /**
     * 获取sftp连接,获取连接失败时会重试,
     * 默认重试次数:5次
     * @return
     * @throws Exception
     */
    public SftpWrapper connectWithRetry() throws Exception {
        return connectWithRetry(DEFAULT_RETRY_COUNT_5);
    }
    public SftpWrapper connectWithRetry(int retryCount) throws Exception {
        //最大重试次数
        int maxRetryCount = DEFAULT_RETRY_COUNT_5;
        if(retryCount > 0){
            maxRetryCount = retryCount;
        }

        int retry = 0;
        Throwable t = null;
        do {
            try {
                SftpWrapper channelSftpWrapper = this.connect();
                if(channelSftpWrapper != null){
                    t = null;
                    return channelSftpWrapper;
                }
            } catch (Exception e) {
                t = e;
            }

            try {
                Thread.sleep(SLEEP_ITME_1000);//休息1秒
            } catch (InterruptedException e) {
            }
        }while (retry++ < maxRetryCount);
        if(t != null){
            throw new Exception(t);
        }
        return null;
    }

}

  • SftpWrapper: sftp连接对象包装类,属性包括Session,Channel,ChannelSftp
      执行sftp操作有ChannelSftp就可以了,使用Sftp包装类,是为了方便关闭资源
package cn.com.soulfox.jsch;

import com.jcraft.jsch.Channel;
import com.jcraft.jsch.ChannelSftp;
import com.jcraft.jsch.Session;
import lombok.AllArgsConstructor;
import lombok.Data;

/**
 * SftpWrapper类简单包装ChannelSftl对象,方便关闭资源
 * @author xxxx
 * @create 2024/7/4 14:26
 */
@AllArgsConstructor//生成全字段,构造方法
@Data //生成 getter,setter方法
public class SftpWrapper {
    private Session session = null;
    private Channel channel = null;
    private ChannelSftp sftp = null;

    /**
     * 关闭资源
     */
    public void disconnect() {
        if (sftp != null && sftp.isConnected()) {
            sftp.disconnect();
        }
        if (channel != null && channel.isConnected()) {
            channel.disconnect();
        }
        if (session != null && session.isConnected()) {
            session.disconnect();
        }
    }
}

  • JschSftpConfig: jsch sftp配置类
      初始化 jsch sftp
package cn.com.soulfox.config;

import cn.com.soulfox.jsch.JschSftpConneciton;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Lazy;

/**
 * @author xxxx
 * @create 2024/7/4 12:12
 */
@Configuration
public class JschSftpConfig {

    @Value("${jsch.sftp.host}")
    private String host;
    @Value("${jsch.sftp.port:22}")
    private Integer port;
    @Value("${jsch.sftp.username}")
    private String username;
    @Value("${jsch.sftp.password}")
    private String password;

    @Value("${jsch.sftp.session-timeout:60000}")
    private Integer sessionTimeout;//单位毫秒
    @Value("${jsch.sftp.connect-timeout:5000}")
    private Integer connectTimeout;//单位毫秒
    
    @Bean
    @Lazy
    public JschSftpConneciton jschSftpConneciton(){
        return new JschSftpConneciton()
        		.setHost(host)
                .setPort(port)
                .setUsername(username)
                .setPassword(password)
                .setSessionTimeout(sessionTimeout)
                .setConnectionTimeout(connectTimeout);
    }

}

sftp连接信息配置在文件application.yml里
jsch:
sftp:
host: 172.168.xxx.xx
port: 1221 #默认22,1221公司自己定义的,这里要配置正确
username: xxxx #远程服务器用户名
password: xxxx #远程服务器密码
sftp连接信息

SftpUitl 工具类,提供下载,上传文件功能

package cn.com.soulfox.jsch;

import com.jcraft.jsch.ChannelSftp;
import com.jcraft.jsch.SftpException;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import java.io.*;

/**
 * @author xxxx
 * @create 2024/7/4 14:28
 */
@Component
@Slf4j
public class JschSftpUtil {

    private static JschSftpConneciton jschSftpConneciton;

    @Autowired
    public void setJschSftpConneciton(JschSftpConneciton jschSftpConneciton) {
        JschSftpUtil.jschSftpConneciton = jschSftpConneciton;
    }

    /**
     * 下载文件
     * @param remotePath            远程目录
     * @param downloadFileName      待下载的远程文件名称
     * @param localSavePath         下载文件保存的本地目录
     */
    public static void downloadFile(String remotePath, String downloadFileName, String localSavePath) {

        SftpWrapper sftpWrapper = null;

        try {
            //sftp连接对象
            sftpWrapper = jschSftpConneciton.connectWithRetry();

            //进入远程服务器指定的目录
            sftpWrapper.getSftp().cd(remotePath);

            if (!checkLocalPath(localSavePath)) {
                log.info("本地目录[{}]不存在,且新建失败+++++", localSavePath);
                return;
            }

            String localFile = checkPathEnd(localSavePath) + downloadFileName;
            File outFile = new File(localFile);

            sftpWrapper.getSftp().get(downloadFileName, new FileOutputStream(outFile));
            log.info("从远程目录[{}]下载文件[{}]到本地目录[{}]成功", remotePath, downloadFileName, localSavePath);

        } catch (SftpException e) {
            log.info("从远程目录[{}]下载文件[{}]到本地目录[{}]失败", remotePath, downloadFileName, localSavePath);
            log.error("下载文件失败: ", e);
        } catch (Exception e) {
            log.info("从远程目录[{}]下载文件[{}]到本地目录[{}]失败", remotePath, downloadFileName, localSavePath);
            log.error("下载文件失败: ", e);
        } finally {
            if (sftpWrapper != null) {
                sftpWrapper.disconnect();
            }
        }


    }

    /**
     *
     * @param localDir              保存上传文件的本地目录
     * @param uploadFileName        上传文件名称
     * @param remoteSaveDir         保存上传文件的远程目录, 建议使用绝对路径
     *                              如果使用相对路径,建议基准目录使用sfpt登录后所在的目录
     *                              这个目录,使用channelSftp.goHome()可以获取
     */
    public static void uploadFile(String localDir, String uploadFileName, String remoteSaveDir) {

        String uploadFilePath = checkPathEnd(localDir) + uploadFileName;
        File uploadFile = new File(uploadFilePath);
        uploadFile(uploadFile, remoteSaveDir);

    }

    /**
     * 上传文件
     * @param uploadFilePath        本地文件的路径
     * @param remoteSaveDir         保存上传文件的远程目录, 建议使用绝对路径
     *                              如果使用相对路径,建议基准目录使用sfpt登录后所在的目录
     *                              这个目录,使用channelSftp.goHome()可以获取
     */
    public static void uploadFile(String uploadFilePath, String remoteSaveDir) {
        File uploadFile = new File(uploadFilePath);
        uploadFile(uploadFile, remoteSaveDir);
    }

    /**
     * 上传文件
     * @param uploadFile        上传文件的File对象
     * @param remoteSavePath    保存上传文件的远程目录, 建议使用绝对路径
     *                          如果使用相对路径,建议基准目录使用sfpt登录后所在的目录
     *                          这个目录,使用channelSftp.goHome()可以获取
     */
    public static void uploadFile(File uploadFile, String remoteSavePath) {
        if(uploadFile == null ){
            log.info("本地文件对象不存在++++");
            return;
        }
        if(!uploadFile.exists()){
            log.info("本地文件[{}]不存在", uploadFile.getAbsoluteFile());
            return;
        }
        InputStream is = null;
        try {
            is = new FileInputStream(uploadFile);
        } catch (FileNotFoundException e) {
            log.info("获取本地文件[{}]的文件流失败", uploadFile.getAbsoluteFile());
            log.error("获取文件流失败: ", e);

            if(is != null){
                try {
                    is.close();
                } catch (IOException ioException) {
                }
            }
            return;
        }

        SftpWrapper sftpWrapper = null;
        try {
            //sftp连接对象
            sftpWrapper = jschSftpConneciton.connectWithRetry();

            //检查远程目录,不存在则创建
            createLinuxRemoteDirs(sftpWrapper.getSftp(), remoteSavePath);

            //进入用户home目录,
            sftpWrapper.getSftp().cd(sftpWrapper.getSftp().getHome());
            //保证当前目录在上传文件保存的目录
            sftpWrapper.getSftp().cd(remoteSavePath);

            //上传文件
            sftpWrapper.getSftp().put(is, uploadFile.getName());

        } catch (SftpException e) {
            log.info("上传本地文件[{}]到远程目录[{}]失败", uploadFile.getAbsoluteFile(), remoteSavePath);
            log.error("上传本地文件失败: ", e);
        } catch (Exception e) {
            log.info("上传本地文件[{}]到远程目录[{}]失败", uploadFile.getAbsoluteFile(), remoteSavePath);
            log.error("上传本地文件失败: ", e);
        } finally {
            if (sftpWrapper != null) {
                sftpWrapper.disconnect();
            }
            if(is != null){
                try {
                    is.close();
                } catch (IOException e) {
                }
            }
        }
    }

    /**
     * 检查目录结是否以目录分隔符结尾
     * 如果不是,则加上目录分隔符结尾
     *
     * @param localPath 本地目录
     * @return
     */
    private static String checkPathEnd(String localPath) {

        if (localPath.endsWith("/") || localPath.endsWith("\\")) {
            return localPath;
        }

        return localPath + File.separator;
    }

    /**
     * 检查本地目录是否存在,不存在就创建。
     * 为了防止其他线程已创建目录,加上了重试代码
     *
     * @param localPath 本地目录
     * @return
     */
    private static boolean checkLocalPath(String localPath) {

        int maxRetryCount = 5;
        int retry = 0;

        do {
            File localDir = new File(localPath);
            if (localDir.exists()) {
                return true;
            }
            boolean result = localDir.mkdirs();
            if (result) {
                return true;
            }

            try {
                Thread.sleep(1000L);//休息1秒
            } catch (InterruptedException e) {
            }
        } while (retry++ < maxRetryCount);

        return false;
    }

    /**
     * 检查和创建[ linux系统 ]的远程多级目录,
     * 外部调用, 单纯的创建远程目录,不作其他操作
     * @param remoteDir     远程目录
     * @throws SftpException
     */
    public static void createLinuxRemoteDirs( String remoteDir) throws SftpException {
        SftpWrapper sftpWrapper = null;
        try {
            sftpWrapper = jschSftpConneciton.connectWithRetry();
            createLinuxRemoteDirs(sftpWrapper.getSftp(), remoteDir);

        } catch (SftpException e) {
            log.info("创建Linux远程目录[{}]失败", remoteDir);
            log.error("创建Linux远程目录失败: ", e);
        } catch (Exception e) {
            log.info("创建Linux远程目录[{}]失败", remoteDir);
            log.error("创建Linux远程目录失败: ", e);
        } finally {
            if (sftpWrapper != null) {
                sftpWrapper.disconnect();
            }
        }
    }
    /**
     * 检查和创建[ linux系统 ]的远程多级目录,
     * linux系统目录分隔符是 “/”
     * 内部上传文件的方法调用
     *
     * @param sftpChannel
     * @param remoteDir  远程目录
     * @throws SftpException
     */
    public static void createLinuxRemoteDirs(ChannelSftp sftpChannel, String remoteDir) throws SftpException {

        if(remoteDir == null || "".equals(remoteDir)){
            log.info("待创建的远程目录为空++++++++");
            return;
        }
        String[] dirs = remoteDir.split("/");;

        if(dirs == null || dirs.length == 0){
            log.info("拆分目录[{}]失败,没有获取到目录数组", remoteDir);
            return;
        }

        //进入用户home目录,保证初始目录正确
        sftpChannel.cd(sftpChannel.getHome());

        if( dirs.length == 1){
            //只有一层目录,直接处理
            try {
                sftpChannel.cd(dirs[0]);
            } catch (SftpException e) {
                if (e.id == ChannelSftp.SSH_FX_NO_SUCH_FILE) {
                    log.info("开始创建远程目录[{}]", dirs[0]);
                    sftpChannel.mkdir(dirs[0]);
                } else {
                    throw e;
                }
            }
            return;
        }

        StringBuilder sb = new StringBuilder();

        //处理第一个元素
        if(remoteDir.startsWith(".")){
            //相对路径,把缺少的路径补上
            sb.append(sftpChannel.getHome()).append("/");
        }else if(remoteDir.startsWith("/")){
            //绝对路径,把"/"放到目录开头
            sb.append("/");
        }else {
            //既不是”/“开头,也不是”.“开头
            //属于相对路径的一种情况
            try {
                //先处理第一层目录
                sftpChannel.cd(dirs[0]);
            } catch (SftpException e) {
                if (e.id == ChannelSftp.SSH_FX_NO_SUCH_FILE) {
                    log.info("开始创建远程目录[{}]", dirs[0]);
                    sftpChannel.mkdir(dirs[0]);
                } else {
                    throw e;
                }
            }
            //把已处理的目录加上
            sb.append(sftpChannel.getHome()).append("/").append(dirs[0]).append("/");
        }


        //从第二个元素开始创建不存在的目录
        for (int i = 1; i < dirs.length; i++) {
            String dir = dirs[i];
            if (dir.isEmpty() ) {
                //跳过空字符串
                continue;
            }
            sb.append(dir + "/");
            try {
                sftpChannel.cd(sb.toString());
            } catch (SftpException e) {
                if (e.id == ChannelSftp.SSH_FX_NO_SUCH_FILE) {
                    log.info("开始创建远程目录[{}]", sb.toString());
                    sftpChannel.mkdir(sb.toString());
                } else {
                    throw e;
                }
            }
        }
//        for (String dir : dirs) {
//            if (dir.isEmpty() || dir.contains(".")) {
//                //跳过空字符串,和"."字符串
//                continue;
//            }
//            sb.append(dir + "/");
//            try {
//                sftpChannel.cd(sb.toString());
//            } catch (SftpException e) {
//                if (e.id == ChannelSftp.SSH_FX_NO_SUCH_FILE) {
//                    sftpChannel.mkdir(sb.toString());
//                } else {
//                    throw e;
//                }
//            }
//        }
    }
}

单元测试

  • JschSftpConfigTest: 测试类
package cn.com.soulfox.config;

import cn.com.soulfox.JschSftpRun;
import cn.com.soulfox.jsch.JschSftpConneciton;
import cn.com.soulfox.jsch.JschSftpUtil;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;

/**
 * @author xxxx
 * @create 2024/7/5 12:57
 */
@RunWith(SpringRunner.class)
@SpringBootTest(classes = JschSftpRun.class,
        webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
public class JschSftpConfigTest {


    @Autowired
    private JschSftpConneciton sftpConneciton;

    /**
     * 下载文件
     */
    @Test
    public void testDownload(){
        //从远程服务器上的下载 /home/jiyh/ 目录下的 testabc.txt
        //下载文件保存到 d:\jiyh
        String remotePath = "/home/jiyh/";
        String downloadFileName = "testabc.txt";
        String localSavePath = "d:\\jiyh";
        JschSftpUtil.downloadFile(remotePath, downloadFileName, localSavePath);
    }

    /**
     * 上传文件
     */
    @Test
    public void testUpload(){
        //上传传本地 d:\jiyh 目录下的 test123.txt文件
        //到远程 /home/jiyh/test/test 目录
        //目录不存在,会自动创建
        JschSftpUtil.uploadFile("d:\\jiyh", "test123.txt", "/home/jiyh/test/test");

    }
}

  • 在远程服务器准备测试文件 文件内容随便,这里我准备的内容为“112dadfdefee”
    在这里插入图片描述

  • 测试下载功能
    测试结果:
    在这里插入图片描述

  • 测试上传功能 :在本地 d:/jiyh 目录准备测试文件“test123.txt”,内容为“dfdfdfdfdaerwrt”
    测试结果
    在这里插入图片描述

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

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

相关文章

《Windows API每日一练》8.5 listbox控件

列表框是将一批文本字符串显示在一个具有滚动功能的方框中的控件。通过发送消息到列表框的窗口过程&#xff0c;程序可以添加或删除列表中的字符串。当列表框中的一个项目被选中时&#xff0c;列表框控件便发送 WM_COMMAND消息到其父窗口。然后父窗口确定哪个项目被选中。 本节…

镭速助力全球高速传输,实现点对点高效传输打破时差壁垒!

在当今全球化的背景下&#xff0c;跨国界的文件协作已成为许多跨国企业日常工作的一部分。但是&#xff0c;时差和网络不稳定等因素常常影响团队间的有效交流与合作。特别是在大文件传输环节&#xff0c;传统的在线传输方法在接收方离线或网络条件不佳时显得力不从心&#xff0…

关于原型和原型链的学习和实践

在前端面试中&#xff0c;原型和原型链始终是一个避不开的问题&#xff0c;今天就弄明白! 原型和原型链 对象的创建方式工厂模式构造函数模式原型模式 原型和原型链实践 对象的创建方式 原型和原型链都是关于对象的内容&#xff0c;先来看一下JavaScript中对象的构建方式。 工…

什么是C#

C#是一种面向对象的语言与c语言不同 C语言是面向过程的编程 C#运行于.NETFramework和.NETCore之上的高级语言 C#是由C和C衍生而来的一种语言 在C#中不建议使用指针 什么叫面向对象 是一种编程范式&#xff0c;它将现实世界中的事物抽象为对象&#xff0c;并通过对象之间的…

深度学习的数学PDF

链接: https://pan.baidu.com/s/1_jScZ7dcyAWGqbrad6bbCQ?pwd9gj9 提取码: 9gj9 复制这段内容后打开百度网盘手机App&#xff0c;操作更方便哦

【计算机网络仿真】b站湖科大教书匠思科Packet Tracer——实验17 开放最短路径优先OSPF

一、实验目的 1.验证OSPF协议的作用&#xff1b; 二、实验要求 1.使用Cisco Packet Tracer仿真平台&#xff1b; 2.观看B站湖科大教书匠仿真实验视频&#xff0c;完成对应实验。 三、实验内容 1.构建网络拓扑&#xff1b; 2.验证OSPF协议的作用。 四、实验步骤 1.构建网…

SpringCloud跨微服务的远程调用,如何发起网络请求,RestTemplate

在我们的业务流程之中不一定都会是自己模块查询自己模块的信息&#xff0c;有些时候就需要去结合其他模块的信息来进行一些查询完成相应的业务流程&#xff0c;但是在SpringCloud每个模块都相对独立&#xff0c;数据库也有数据隔离。所以当我们需要其他微服务模块的信息的时候&…

NuGet 中国国内镜像

有时候网络不好&#xff0c;需要使用国内的一些镜像源。 华为资源&#xff1a; https://repo.huaweicloud.com/repository/nuget/v3/index.json 腾讯资源&#xff1a; https://mirrors.cloud.tencent.com/nuget/ 微软中国镜像源&#xff1a;https://nuget.cdn.azure.cn/v3…

BIOTIN-Gly-Asp-Gly-Asp-Glu-Val-Asp-Gly-Cys 生物素标记多肽

BIOTIN-Gly-Asp-Gly-Asp-Glu-Val-Asp-Gly-Cys 生物素标记多肽 生物素标记多肽的特点 高亲和力结合&#xff1a;生物素与链霉亲和素&#xff08;streptavidin&#xff09;或抗生物素蛋白&#xff08;avidin&#xff09;之间的结合是生物学上已知的最强的非共价相互作用之一。这种…

大模型知识大全2-资源【大模型】

文章目录 可用的模型部分模型的训练规模和数据常用数据集外文中文书籍和论文百科数据代码混合数据集微调数据集对齐数据集 可用的模型 仅记录中文可用的表现较优的模型LLaMA、LLaMA2、LLaMA3&#xff08;规模在7,13,30,65左右&#xff0c;需要搭配ChineseLLaMA的LoRA权重使用&…

Mybatis操作和动态SQL

Mybatis操作 准备 准备数据库表 emp 创建一个新的 springboot 工程&#xff0c;选择引入对应的起步依赖&#xff08; mybatis 、 mysql 驱动、 lombok &#xff09; application.properties 中引入数据库连接信息 spring.datasource.driver-class-namecom.mysql.cj.jdbc.Driv…

关于Linux的操作作业!24道题

&#x1f3c6;本文收录于「Bug调优」专栏&#xff0c;主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案&#xff0c;希望能够助你一臂之力&#xff0c;帮你早日登顶实现财富自由&#x1f680;&#xff1b;同时&#xff0c;欢迎大家关注&&收藏&&…

文章SameStr(四):图4代码

“Publication Figure 4” 百度云盘链接: https://pan.baidu.com/s/15g7caZp354zIWktpnWzWhQ 提取码: 4sh7 Libraries Standard Import library(tidyverse) library(cowplot) library(scales) library(ggpubr)Special library(caret) library(plotROC) library(tidymodel…

To美术-渲染管线及优化方向(CPU方向)

一、CPU与GPU 1、CPU与GPU的区别 橙黄色&#xff1a;控制单元   橙红色&#xff1a;存储单元  绿色&#xff1a;计算单元 CPU:结构组成复杂、控制逻辑丰富&#xff0c;计算量小&#xff0c;适合复杂运算 GPU&#xff1a;结构组成简单&#xff0c;核心数量多&#xff0c;计…

jmeter-beanshell学习5-beanshell加减乘除运算

我用到的场景是计算金额&#xff0c;所以主要以金额为主&#xff0c;感觉这部分有点麻烦&#xff0c;直接写遇到的几个坑&#xff0c;就不演示解决的过程了。 1.最早写了个两数相减&#xff0c;但是小数精度容易出现问题。比如1-0.010.989999997这种情况&#xff0c;随便写的几…

Windows 电脑查看 WiFi 密码的方法都有哪些?

从设置面板中查看 当你使用的是笔记本电脑并且连接 WiFi 之后可以在设置面板中查看 WiFi 密码&#xff0c;首先打开设置界面&#xff0c;然后点击网络和 Internet&#xff0c;找到 WiFi 之后点击进入&#xff0c;然后点击管理已知网络。 然后点击已经连接好的无线网络。 进入之…

前端Din字体和造字工房力黑字体文件

Din 字体是一种经典的、简洁的无衬线字体&#xff0c;它源自1930年代的德国交通标志设计。 造字工房力黑字体适用于数字&#xff0c;驾驶舱标题等统计界面 DIN-Medium.otf 案例 造字工房力黑.TTF 案例

最新综述:多模态引导的基于文生图大模型的图像编辑算法

文章目录 综述亮点1. 图像编辑任务的范围2. 一般性编辑算法的统一框架3. 统一框架在多模态编辑任务中的应用4. 不同组合在文本引导编辑场景下的比较5. 未来研究方向 近期&#xff0c;复旦大学 FVL 实验室和南洋理工大学的研究人员对于多模态引导的基于文生图&#xff08;Text-t…

nvidia driver和cuda版本较低,ubuntu系统更新nvidia驱动的方法(对于小白最快最最保险的方法)

问题描述&#xff1a; 系统&#xff1a;ubuntu22.04LTS 这两天安装另一个低版本的pytorch环境&#xff0c;提示我的cuda版本很旧&#xff0c;然后运行程序时候甚至直接报错&#xff0c;如下所示&#xff1a; .local/lib/python3.10/site-packages/torch/cuda/__init__.py&quo…

DBA 数据库管理 部署Mysql 服务,基础查询

数据库&#xff1a;存储数据的仓库 数据库服务软件&#xff1a; 关系型数据库&#xff1a; 存在硬盘 &#xff0c;制作表格的 数据库的参数 [rootmysql50 ~]# cat /etc/my.cnf.d/mysql-server.cnf 主配置文件 [mysqld] datadir/var/lib/mysql 存放数据库目录…