JDBC 结构优化2

JDBC 结构优化2

文章目录

  • JDBC 结构优化2
    • 结构优化2 - ATM系统(存,取,转,查)
      • 1 Service
      • 2 事务
      • 3 ThreadLocal
      • 4 事务的封装

结构优化2 - ATM系统(存,取,转,查)

1 Service

什么是业务?

  • 代表用户完成的一个业务功能,可以由一个或多个DAO的调用组成。
  • 软件所提供的一个功能叫业务。

核心思想:

  • 用户角度: 功能即业务

    • 转账这一功能就是一个业务
  • 服务者角度: 业务包含多个步骤

    1. 从A账户扣钱
    2. 给B账户加钱
  • 程序设计角度: 每一个步骤都是一个方法

    • AccountDao

      • save() //存钱
      • take() //取钱
    • AccountService

      • transfer(){
        	dao.take(A);
        	dao.save(B);
        }
        

**目的:**DAO提高数据操作的重用性,Service提高业务功能的重用性。

2 事务

概念:

  • 事务是一个原子操作,由一个或多个SQL语句组成。
  • 在同一个事务当中,所有的SQL语句要么全部执行成功,要么全部失败。

事务的边界:

  • 在JDBC中,先获得Connection对象。
  • 开始事务:conn.setAutoCommit(false); #set autocommit=0;
  • 手动提交事务:conn.commit(); #commit
  • 手动回滚事务:conn.rollback(); #rollback;

在Service中,调用了多次DAO操作。每一个业务功能都要控制事务。

在Service层中控制事务:

在这里插入图片描述

问题: 当程序发生异常,事务回滚并没有成功。

原因:Service控制事务和DAO访问数据库时的连接对象,并非同一个

在这里插入图片描述

注意: 在一个业务流程中,要保证多个数据库访问操作的连接对象是同一个。

解决方法:

  • 可以将整个线程中(单线程)中,存储一个共享值。
  • 线程拥有一个类似Map的属性,键值对结构<ThreadLocal对象,值>。

3 ThreadLocal

ThreadLocal(线程本地变量):

  • 每一个线程通过ThreadLocal绑定一个连接对象。
  • 在整个流程中任一环节可以存值或取值。

ThreadLocal应用流程:

在这里插入图片描述

ThreadLocal应用:

在这里插入图片描述

  • 在释放资源的方法中,连接对象关闭后,提供threadLocal.remove()。

  • 将关闭后的连接从当前ThreadLocal中移除。

4 事务的封装

现有问题:

  • 事务的开启、提交、回滚的代码都写在了Service中。
  • Service的主要职责是模拟现实业务功能。要将事务的控制封装在工具类中。

工具类封装事务控制方法:

  • 开启事务:begin();

代码演示:

// 4 开启事务
public static void begin() throws SQLException{
    Connection connection = DBUtils.getConnection();
    if (connection != null) {
        connection.setAutoCommit(false);
    }
}
  • 提交事务:commit();

代码演示:

// 5 提交事务
public static void commit() throws SQLException{
    Connection connection = DBUtils.getConnection();
    if (connection != null) {
        connection.commit();
    }
}
  • 回滚事务:rollback();

代码演示:

// 6 回滚事务
public static void rollback() throws SQLException{
    Connection connection = DBUtils.getConnection();
    if (connection != null) {
        connection.rollback();
    }
}
  • 关闭连接close(); 解除绑定连接

代码演示:

// 3 释放资源
public static void closeAll(Connection connection, Statement statement, ResultSet resultSet) {

    try {
        if (resultSet != null) {
            resultSet.close();
        }
        if (statement != null) {
            statement.close();
        }
        if (connection != null) {
            //判断是否开启事务, 如果没有开启事务, 要关闭
            //若开启事务, 则在事务结束后手动关闭
            if (connection.getAutoCommit()) {
                connection.close();
                //从线程中移除
                threadLocal.remove();
            }
        }
    } catch (SQLException e) {
        System.out.println("资源释放失败: "+e.getMessage());
        throw new RuntimeException(e);
    }
}
// 7 关闭事务连接
public static void close() throws SQLException{
    Connection connection = DBUtils.getConnection();
    if (connection != null) {
        connection.close();
        threadLocal.remove();
    }
}

ATM系统

代码演示:

Dao层

Dao接口:

public interface AccountDao {
    //查询
    Account select(String cardNo, String password);
    Account select(String cardNo);
    //更新
    //更新存钱
    void updateSave(String cardNo, BigDecimal money);
    //更新取钱
    void updateTake(String cardNo, BigDecimal money);
    //更新密码
    void updatePassword(String cardNo, String password);
}

DaoImpl:

public class AccountDaoImpl implements AccountDao {

    //登录
    @Override
    public Account select(String cardNo, String password) {
        ResultSet resultSet = null;
        PreparedStatement preparedStatement = null;
        Connection connection = null;
        try {
            // 1 获取连接
            connection = DBUtils.getConnection();
            // 2 创建预编译命令
            String sql = "SELECT * FROM companydb.account where cardNo = ? And password = ?";
            preparedStatement = connection.prepareStatement(sql);
            // 给'?'占位符赋值
            preparedStatement.setObject(1, cardNo);
            preparedStatement.setObject(2, password);
            // 3 执行命令
            resultSet = preparedStatement.executeQuery();
            Account account = null;
            // 4 处理
            if (resultSet.next()) {
                int id = resultSet.getInt("id");
                String name = resultSet.getString("name");
                BigDecimal money = resultSet.getBigDecimal("money");
                account = new Account(id, name, money, cardNo, password);
            }
            return account;
        } catch (SQLException e) {
            throw new RuntimeException(e);
        } finally {
            // 5 释放资源
            DBUtils.closeAll(connection, preparedStatement, resultSet);
        }
    }

    //通过卡号查询
    @Override
    public Account select(String cardNo) {
        ResultSet resultSet = null;
        PreparedStatement preparedStatement = null;
        Connection connection = null;
        try {
            // 1 获取连接
            connection = DBUtils.getConnection();
            // 2 创建预编译命令
            String sql = "SELECT * FROM companydb.account where cardNo = ?";
            preparedStatement = connection.prepareStatement(sql);
            // 参数赋值
            preparedStatement.setObject(1, cardNo);
            // 3 执行命令
            resultSet = preparedStatement.executeQuery();
            Account account = null;
            // 4 处理(获取结果集中的数据)
            if (resultSet.next()) {
                int id = resultSet.getInt("id");
                String name = resultSet.getString("name");
                BigDecimal money = resultSet.getBigDecimal("money");
                String password = resultSet.getString("password");
                account = new Account(id, name, money, cardNo, password);
            }
            return account;
        } catch (SQLException e) {
            throw new RuntimeException(e);
        } finally {
            //关闭资源
            DBUtils.closeAll(connection, preparedStatement, resultSet);
        }
    }

    //存钱
    @Override
    public void updateSave(String cardNo, BigDecimal money) {
        Connection connection = null;
        PreparedStatement preparedStatement = null;
        try {
            //1 获取连接
            connection = DBUtils.getConnection();
            //2 创建预编译SQL命令
            String sql = "UPDATE companydb.account SET account.money = account.money + ? WHERE account.cardNo = ?";
            preparedStatement = connection.prepareStatement(sql);
            //2.1 参数赋值
            preparedStatement.setObject(1, money);
            preparedStatement.setObject(2, cardNo);
            //3 执行命令
            preparedStatement.executeUpdate();
        } catch (SQLException e) {
            throw new RuntimeException(e);
        } finally {
            //4 释放资源
            DBUtils.closeAll(connection, preparedStatement, null);
        }
    }

    //取钱
    @Override
    public void updateTake(String cardNo, BigDecimal money) {
        Connection connection = null;
        PreparedStatement preparedStatement = null;
        try {
            //1 获取连接
            connection = DBUtils.getConnection();
            //2 创建预编译SQL命令
            String sql = "UPDATE companydb.account SET account.money = account.money - ? WHERE account.cardNo = ?";
            preparedStatement = connection.prepareStatement(sql);
            //2.1 参数赋值
            preparedStatement.setObject(1, money);
            preparedStatement.setObject(2, cardNo);
            //3 执行命令/处理
            preparedStatement.executeUpdate();
        } catch (SQLException e) {
            throw new RuntimeException(e);
        } finally {
            //释放资源
            DBUtils.closeAll(connection, preparedStatement, null);
        }
    }

    //修改密码
    @Override
    public void updatePassword(String cardNo, String password) {
        Connection connection = null;
        PreparedStatement preparedStatement = null;
        try {
            //1 获取连接
            connection = DBUtils.getConnection();
            String sql = "UPDATE companydb.account SET account.password = ? WHERE account.cardNo = ?";
            //2 创建预编译SQL命令
            preparedStatement = connection.prepareStatement(sql);
            //2.1 参数赋值
            preparedStatement.setObject(1, password);
            preparedStatement.setObject(2, cardNo);
            //3 执行命令/处理
            preparedStatement.executeUpdate();
        } catch (SQLException e) {
            throw new RuntimeException(e);
        } finally {
            //4 释放资源
            DBUtils.closeAll(connection, preparedStatement, null);
        }
    }
}

entity

Account:

public class Account {
    private Integer id;
    private String name;
    private BigDecimal money;
    private String cardNo;
    private String password;

    public Account() {
    }

    public Account(Integer id, String name, BigDecimal money, String cardNo, String password) {
        this.id = id;
        this.name = name;
        this.money = money;
        this.cardNo = cardNo;
        this.password = password;
    }

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public BigDecimal getMoney() {
        return money;
    }

    public void setMoney(BigDecimal money) {
        this.money = money;
    }

    public String getCardNo() {
        return cardNo;
    }

    public void setCardNo(String cardNo) {
        this.cardNo = cardNo;
    }

    public String getPassword() {
        return password;
    }

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

service层

service接口:

public interface AccountService {
    //登录
    Account login(String cardNo, String password);
    //存钱
    void save(String cardNo, BigDecimal money);
    //取钱
    void take(String cardNo, BigDecimal money);
    //转账
    void trans(String from, String to, BigDecimal money);
    //查询余额
    Account queryMoney(String cardNo, String password);
    //修改密码
    void modifyPassword(String cardNo, String oldPwd, String newPwd);
}

serviceImpl:

public class AccountServiceImpl implements AccountService {
    //声明数据访问对象, 以调用Dao层操作
    private static final AccountDaoImpl accountDao = new AccountDaoImpl();
    //登录
    public Account login(String cardNo, String password) {
        //通过卡号和密码查询数据, 都正确时放回数据, 登录成功
        Account account = accountDao.select(cardNo, password);
        if (account == null) {
            throw new RuntimeException("账户或密码错误");
        }
        return account;
    }

    //存钱
    public void save(String cardNo, BigDecimal money) {
        //添加限制
        //因money是BigDecimal类型, 使用compareTo方法进行比较
        //通过相减的方式进行比较判断
        if (money.compareTo(new BigDecimal(0))<=0) {
            throw new RuntimeException("存款金额必须大于0");
        }
        if (money.compareTo(new BigDecimal(50000))>0) {
            throw new RuntimeException("存款金额必须小于50000");
        }
        accountDao.updateSave(cardNo, money);
    }

    //取款
    public void take(String cardNo, BigDecimal money) {
        if (money.compareTo(new BigDecimal(0))<=0) {
            throw new RuntimeException("取款金额必须大于0");
        }
        if (money.compareTo(new BigDecimal(20000))>0) {
            throw new RuntimeException("取款金额必须小于20000");
        }
        //通过卡号查询当前账户的余额
        Account account = accountDao.select(cardNo);
        if (money.compareTo(account.getMoney())>0) {
            throw new RuntimeException("账户余额不足...");
        }
        accountDao.updateTake(cardNo, money);
    }

    //转账
    public void trans(String from, String to, BigDecimal money) {
        if (money.compareTo(new BigDecimal(0))<=0) {
            throw new RuntimeException("转账金额必须大于0");
        }
        if (money.compareTo(new BigDecimal(20000))>0) {
            throw new RuntimeException("转账金额必须小于20000");
        }
        //通过卡号查询转账账户的余额
        Account account = accountDao.select(from);
        if (money.compareTo(account.getMoney())>0) {
            throw new RuntimeException("账户余额不足...");
        }
        //判断转账双方是否是同一人
        if (from.equals(to)) {
            throw new RuntimeException("不能给自己转账...");
        }
        //通过卡号查询判断接收方是否存在
        if (accountDao.select(to) == null) {
            throw new RuntimeException("对方账户不存在...");
        }
        //转账
        try {
            //开启事务, 调用线程中的连接
            DBUtils.begin();
            //使用同一连接
            accountDao.updateTake(from, money);
            accountDao.updateSave(to, money);
            DBUtils.commit();//手动提交
        } catch (Exception e) {
            try {
                DBUtils.rollback();//异常回滚
            } catch (SQLException ex) {
                e.printStackTrace();
            }
            throw new RuntimeException(e);
        } finally {
            //手动关闭
            try {
                DBUtils.close();//关闭事务连接
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
    }

    //查询余额
    public Account queryMoney(String cardNo, String password) {
        AccountDao accountDao = new AccountDaoImpl();
        Account account = accountDao.select(cardNo, password);
        if (account==null) {
            throw new RuntimeException("密码有误");
        }
        return account;
    }

    //修改密码
    public void modifyPassword(String cardNo, String oldPwd, String newPwd) {
        AccountDao accountDao = new AccountDaoImpl();
        Account account = accountDao.select(cardNo,oldPwd);
        if (account == null) {
            throw new RuntimeException("原始密码有误...");
        }
        if (oldPwd.equals(newPwd)) {
            throw new RuntimeException("新密码不能与原始密码相同");
        }
        accountDao.updatePassword(cardNo,newPwd);
    }
}

test:

ATMSystem:

public class ATMSystem {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        System.out.println("请输入卡号");
        String cardNo = scanner.next();
        System.out.println("请输入密码");
        String password = scanner.next();
        //创建业务对象
        AccountService accountService = new AccountServiceImpl();
        try {
            Account account = accountService.login(cardNo, password);
            boolean flag = true;
            System.out.println("登录成功, 欢迎: "+account.getName());
            do {
                System.out.println("1 存钱, 2 取钱, 3 转账, 4 查询余额, 5 修改密码, 0 退出");
                System.out.println("请选择:");
                int choose = scanner.nextInt();
                switch (choose) {
                    case 1:
                        try {
                            System.out.println("请输入存钱金额:");
                            BigDecimal money = scanner.nextBigDecimal();
                            accountService.save(account.getCardNo(), money);
                            System.out.println("存钱成功...");
                        } catch (Exception e) {
                            System.out.println(e.getMessage());
                        }
                        break;
                    case 2:
                        try {
                            System.out.println("请输入取款金额:");
                            BigDecimal money = scanner.nextBigDecimal();
                            accountService.take(account.getCardNo(), money);
                            System.out.println("取款成功...");
                        } catch (Exception e) {
                            System.out.println(e.getMessage());
                        }
                        break;
                    case 3:
                        System.out.println("请输入对方的卡号:");
                        String toCard = scanner.next();
                        System.out.println("请输入转账金额:");
                        BigDecimal money = scanner.nextBigDecimal();
                        try {
                            accountService.trans(account.getCardNo(),toCard,money);
                            System.out.println("转账成功...");
                        } catch (Exception e) {
                            System.out.println(e.getMessage());
                        }
                        break;
                    case 4:
                        System.out.println("请输入密码:");
                        String pwd = scanner.next();
                        try {
                            System.out.println(
                                    "当前账户余额为: "
                                    + accountService.queryMoney(account.getCardNo(), pwd).getMoney()
                            );
                        } catch (Exception e) {
                            System.out.println(e.getMessage());
                        }
                        break;
                    case 5:
                        System.out.println("请输入原始密码");
                        String oldPwd = scanner.next();
                        System.out.println("请输入新密码");
                        String newPwd = scanner.next();
                        System.out.println("请再次输入新密码");
                        String againPwd = scanner.next();
                        if (newPwd.equals(againPwd)) {
                            try {
                                accountService.modifyPassword(account.getCardNo(), oldPwd, newPwd);
                                System.out.println("密码修改成功...");
                            } catch (Exception e) {
                                System.out.println(e.getMessage());
                            }
                        } else {
                            System.out.println("两次密码不一致...");
                        }
                        break;
                    case 0:
                        flag = false;
                        break;
                    default:
                        System.out.println("输入有误...");
                }
            } while (flag);
            System.out.println("欢迎下次光临...");
        } catch (Exception e) {
            System.out.println(e.getMessage());
        }
    }
}

util工具类

DbUtils:

public class DBUtils {
    private static  String url;
    private static  String user;
    private static  String pwd;
    private static final ThreadLocal<Connection> threadLocal = new ThreadLocal<>();
    // 1 注册驱动
    static {
        try {
            // 读取属性配置文件
            Properties properties = new Properties();
            FileInputStream fis = new FileInputStream("Properties/db.properties");
            properties.load(fis);
            fis.close();
            // 变量赋值
            String driver = properties.getProperty("driver");
            url = properties.getProperty("url");
            user = properties.getProperty("user");
            pwd = properties.getProperty("pwd");
            // 获取驱动对象
            Class.forName("com.mysql.jdbc.Driver");
        } catch (Exception e) {
            System.out.println("注册驱动失败: "+e.getMessage());
        }
    }

    // 2 获取连接
    public static Connection getConnection() {
        try {
            //从线程中获取连接
            Connection connection = threadLocal.get();
            //判断线程中是否有连接, 没有则从数据库中获取
            if (connection == null) {
                connection = DriverManager.getConnection(url,user,pwd);
                //将连接放入线程中
                threadLocal.set(connection);
            }
            //若线程中有连接, 则直接返回
            return connection;
        } catch (SQLException e) {
            throw new RuntimeException(e);
        }
    }

    // 3 释放资源
    public static void closeAll(Connection connection, Statement statement, ResultSet resultSet) {

        try {
            if (resultSet != null) {
                resultSet.close();
            }
            if (statement != null) {
                statement.close();
            }
            if (connection != null) {
                //判断是否开启事务, 如果没有开启事务, 要关闭
                //若开启事务, 则在事务结束后手动关闭
                if (connection.getAutoCommit()) {
                    connection.close();
                    //从线程中移除
                    threadLocal.remove();
                }
            }
        } catch (SQLException e) {
            System.out.println("资源释放失败: "+e.getMessage());
            throw new RuntimeException(e);
        }
    }

    // 4 开启事务
    public static void begin() throws SQLException{
        Connection connection = DBUtils.getConnection();
        if (connection != null) {
            connection.setAutoCommit(false);
        }
    }

    // 5 提交事务
    public static void commit() throws SQLException{
        Connection connection = DBUtils.getConnection();
        if (connection != null) {
            connection.commit();
        }
    }

    // 6 回滚事务
    public static void rollback() throws SQLException{
        Connection connection = DBUtils.getConnection();
        if (connection != null) {
            connection.rollback();
        }
    }

    // 7 关闭事务连接
    public static void close() throws SQLException{
        Connection connection = DBUtils.getConnection();
        if (connection != null) {
            connection.close();
            threadLocal.remove();
        }
    }
}

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

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

相关文章

【spring】服务注册和发现是什么意思?Spring Cloud 如何实现?

&#x1f34e;个人博客&#xff1a;个人主页 &#x1f3c6;个人专栏&#xff1a;Spring ⛳️ 功不唐捐&#xff0c;玉汝于成 目录 正文 我的其他博客 正文 当我们开始一个项目时&#xff0c;我们通常在属性文件中进行所有的配置。随着越来越多的服务开发和部署&#xff0c…

免费的ChatGPT网站(7个)

还在为找免费的chatGPT网站或者应用而烦恼吗&#xff1f;博主归纳总结了7个国内非常好用&#xff0c;而且免费的chatGPT网站&#xff0c;AI语言大模型&#xff0c;我们都来接触一下吧。 免费&#xff01;免费&#xff01;免费&#xff01;...&#xff0c;建议收藏保存。 1&…

熟悉MATLAB 环境

一、问题描述 熟悉MATLAB 环境。 二、实验目的 了解Matlab 的主要功能&#xff0c;熟悉Matlab 命令窗口及文件管理&#xff0c;Matlab 帮助系统。掌握命令行的输入及编辑&#xff0c;用户目录及搜索路径的配置。了解Matlab 数据的特点&#xff0c;熟悉Matlab 变量的命名规则&a…

安卓网格布局GridLayout

<?xml version"1.0" encoding"utf-8"?> <GridLayout xmlns:android"http://schemas.android.com/apk/res/android"xmlns:tools"http://schemas.android.com/tools"android:layout_width"match_parent"android:la…

故障诊断 | 一文解决,GRU门控循环单元故障诊断(Matlab)

文章目录 效果一览文章概述专栏介绍模型描述源码设计参考资料效果一览 文章概述 故障诊断 | 一文解决,GRU门控循环单元故障诊断(Matlab) 专栏介绍 订阅【故障诊断】专栏,不定期更新机器学习和深度学习在故障诊断中的应用;订阅

前端开发基于Qunee绘制网络拓扑图总结-02

1、渲染连线颜色 *关键函数一定要调用&#xff1a;graph.invalidate()* graph.forEach(function(element) {if (element instanceof Q.Edge) {let arr [#549BF1, #AA8A6E, #8F54F1,#5A70BC,#BCBF5C, #BC5A76, #67B4D4,#B4C9EF, #676AD4, #A86EAA,#5CBF7F, #EFB4B4];let inde…

TypeScript Array(九) 数组

1.TypeScript Array 数组 1.1. 描述 数组对象是使用单独的变量名来存储一系列的值。  &emsp比如&#xff0c;你现在有一组数据&#xff0c;存单独变量如下&#xff1a; var data1"Android"; var data2"Java"; var data3"Harmony";那如果有…

【unity小技巧】FPS简单的射击换挡瞄准动画控制

文章目录 射击动画控制换弹动画瞄准动画完结 射击动画控制 换弹动画 调用 瞄准动画 问题&#xff1a;瞄准时&#xff0c;但是动画会卡住&#xff0c;不会播放瞄准的待机动画 修改 调用 动画如果太快可以去修改播放速度 播放速度变慢了&#xff0c;可能导致切换待机动画也…

防火墙详解

一、基本定义 所谓“防火墙”是指一种将内部网和公众访问网&#xff08;如Internet&#xff09;分开的方法&#xff0c;它实际上是一种建立在现代通信网络技术和信息安全技术基础上的应用性安全技术&#xff0c;隔离技术。越来越多地应用于专用网络与公用网络的互联环境之中&a…

理想架构的高回退Doherty功率放大器理论与ADS仿真-Multistage

理想架构的高回退Doherty功率放大器理论与仿真-Multistage 参考&#xff1a; 三路Doherty设计 01 射频基础知识–基础概念 Switchmode RF and Microwave Power Amplifiers、 理想架构的Doherty功率放大器&#xff08;等分经典款&#xff09;的理论与ADS电流源仿真参考&#x…

springCloud gateway 防止XSS漏洞

springCloud gateway 防止XSS漏洞 一.XSS(跨站脚本)漏洞详解1.XSS的原理和分类2.XSS漏洞的危害3.XSS的防御 二.Java开发中防范XSS跨站脚本攻击的思路三.相关代码&#xff08;适用于spring cloud gateway&#xff09;1.CacheBodyGlobalFilter.java2.XssRequestGlobalFilter.java…

二进制部署promethues

1、定义&#xff1a;promethues是一个开源的系统监控以及报警系统&#xff0c;整合zabbix的功能&#xff08;监控系统、网络、设备&#xff09;&#xff0c;promethues可以兼容网络、设备、容器监控、告警系统。因为其与k8s是一个项目基金开发出来的产品&#xff0c;天生匹配k8…

OpenCV 8 - 模糊处理(均值滤波,高斯滤波,中值滤波,双边滤波)

模糊处理原理: Blur是图像处理中最简单和常用的操作之一,使用该操作的原因为了给图像预处理时候减低噪声使用,Blur操作其背后是数学的卷积计算, 通常这些卷积算子计算都是线性操作,所以又出线性虑波。 假设有6x6的图像像素点矩阵。卷积过程:6x6上面是个3x3的窗口,从左向右,…

【Linux】基本指令(上)

&#x1f984;个人主页:修修修也 &#x1f38f;所属专栏:Linux ⚙️操作环境:Xshell (操作系统:CentOS 7.9 64位) 目录 Xshell快捷键 Linux基本指令 ls指令 pwd指令 cd指令 touch指令 mkdir指令 rmdir指令/rm指令 结语 Xshell快捷键 AltEnter 全屏/取消全屏 Tab 进…

如何对Ajax请求进行封装操作,解决跨域问题的方法,如何使用core解决跨域

目录 1.Ajax原理 2.为什么要封装 3.如何进行封装 4.如何请求 5.如何解决Ajax跨域问题 6.使用CORS解决Ajax跨域问题 1.服务端 2.客户端 1.Ajax原理 Ajax&#xff08;Asynchronous JavaScript and XML&#xff09;是一种通过在后台与服务器进行少量数据交换&…

通用CI/CD软件平台TeamCity全新发布v2023.11——增强Git托管平台的集成

TeamCity是一个通用的 CI/CD 软件平台&#xff0c;可以实现灵活的工作流、协作和开发做法。我们的解决方案将帮助在您的 DevOps 流程中成功实现持续集成、持续交付和持续部署。 TeamCity 2023.11正式版下载 TeamCity 2023.11 带来了矩阵构建和构建缓存等多项备受期待的功能&a…

立体边界,让arcgis出图更酷炫一些

就是这样子的那个图—— 本期我们还是用长沙市为例&#xff0c; 来手把手的演示制作立体边界&#xff0c; 就是这个样子的边界—— 第一步—准备底图 其实你准备什么底图都可以哈&#xff0c;例如调用天地图、下载个影像图&#xff0c;或者用其他什么的底图&#xff0c;都是…

【图文详解】阿里云服务器放行高防IP加入安全组

打开阿里云的云服务器配置面板&#xff0c;在要操作实例的操作列找到更多 > 网络和安全组 > 安全组配置。 对已有安全组配置规则&#xff0c;或者直接添加安全组规则。 根据需要放通高防IP在内的IP段相应协议类型的端口访问。

linux交叉编译方法——虚拟机编译,在树莓派平台上运行

一、 交叉编译是什么 交叉编译 是在一个平台上生成另一个平台上的可执行代码。 我们再windows上面编写C51代码&#xff0c;并编译成可执行代码&#xff0c;如xx.hex, 是在c51上面运行&#xff0c;不是在windows上面运行 我们在ubuntu上面编写树…

设计模式——状态模式(State Pattern)

概述 状态模式(State Pattern)&#xff1a;允许一个对象在其内部状态改变时改变它的行为&#xff0c;对象看起来似乎修改了它的类。其别名为状态对象(Objects for States)&#xff0c;状态模式是一种对象行为型模式。状态模式用于解决系统中复杂对象的状态转换以及不同状态下行…