MySQL | JDBC连接数据库详细教程【全程干货】

news2024/11/25 13:52:17

在这里插入图片描述

文章目录

  • 一、什么是JDBC?
  • 二、JDBC工作原理
  • 三、使用JDBC连接MySQL数据库【✔】
    • 1、安装对应数据驱动包
    • 2、将jar包导入项目中
    • 3、编写代码连接数据库【⭐】
      • 1️⃣ 创建数据源
      • 2️⃣ 和数据库建立网络连接
      • 3️⃣ 构造SQL语句
      • 4️⃣ 执行SQL语句
      • 5️⃣ 断开连接,释放资源
    • 4、测试观察
    • 5、代码优化
      • 1️⃣灵活变通SQL
      • 2️⃣防止SQL注入攻击
    • 6、SQL查找的写法
  • 四、整体代码展示
  • 五、总结与提炼

一、什么是JDBC?

JDBC,即Java Database Connectivity,java数据库连接。是一种用于执行SQL语句的Java API,它是Java中的数据库连接规范。这个API由 java.sql.*,javax.sql.* 包中的一些类和接口组成,它为Java开发人员操作数据库提供了一个标准的API,可以为多种关系数据库提供统一访问

二、JDBC工作原理

JDBC 为多种关系数据库提供了统一访问方式,作为特定厂商数据库访问API的一种高级抽象,它主要包含一些通用的接口类

下面是我画的一张有关JDBC访问数据库层次结构的图:

可以看到,我们程序员在开发JavaEE应用的时候,通过调用Sun公司提供给我们的JDBC相关API,就可以切访问数据库的JSBC驱动程序,继而通过驱动来加载数据库

在这里插入图片描述
JDBC优势:

  1. Java语言访问数据库操作完全面向抽象接口编程
  2. 开发数据库应用不用限定在特定数据库厂商的API
  3. 程序的可移植性大大增强

三、使用JDBC连接MySQL数据库【✔】

1、安装对应数据驱动包

那有同学就问了,数据驱动包是什么?

💬 对于数据驱动包而言,其实就是上面我所说到的各个【数据库厂商】所提供给我们的JDBC驱动程序,因为只有数据库厂商那边提供给了我们加载数据库的驱动,我们才能去获取到数据库中的数据表然后进行CRUD等操作

  • 这样说呢可能还不是很好理解,我们可以通过画图来帮助理解,知道了JDBC的工作原理后,我们还要去了解什么去驱动包

在这里插入图片描述

接下去就要去下载这个驱动包,去哪里下呢?当然是官网!

  • 但是呢,官网很麻烦,你要一个个地去找,不知道要下那个才好,下面是MySQL的官网的驱动下载处,很好找。但若是你用的是Oracle数据库的话,就会很麻烦了,因为它们这家公司很大,产品也很多,所以官网中会很很多东西,你很难去找到自己想要的驱动程序

在这里插入图片描述

难道这就没办法了吗?要一个个找吗😥当然有,那就是我们心爱的【Maven】

  • 对于Maven而言,学Java的同学一定不陌生,它是一个 “中央仓库”,Maven 作为 Java 项目管理工具,它不仅可以用作包管理,还有许多的插件,可以支持整个项目的开发、打包、测试及部署等一系列行为,简直是居家旅行、杀人灭口必备良药!
  • 咳咳,然后我们就可以去下载MySQL对应的驱动包了,按照下图指示来即可

在这里插入图片描述

  • 点进去之后就可以选择对应的驱动包版本了,但是这里在选择的时候要注意了,需要和你MySQL安装的版本一致才可,我的MySQL是5.7的,所以这里大版本不能变,要选择5.1开头的,后面的小版本没关系;当然如果你的MySQL是8.x版本的,你就要选择8开头的驱动包了

在这里插入图片描述

  • 点进去之后我们看到有个jar,点一下它,你就可以拥有一个MySQL相关的驱动包了,就和我们平常使用的.rar.zip压缩包没什么区别,就是Java的压缩包以.jar结尾

在这里插入图片描述

  • 下载好了之后就是这个样子的

在这里插入图片描述

2、将jar包导入项目中

现在我们有了这个jar后,就要把它用起来

  • 那现在我们去新建一个项目,然后把这个包导进来,不过在此之前要新建一个名为lib的目录,它是library库的缩写,当然你也可以写其他的名称

在这里插入图片描述

  • 然后右键这个目录选择【添加为库…】

在这里插入图片描述

  • 此时这个MySQL的驱动库就导入到我们的项目中了

在这里插入图片描述

3、编写代码连接数据库【⭐】

加载完MySQL的驱动包之后,我们就可以使用一些API去操作数据库了,准备好了吗,发车了🚗

1️⃣ 创建数据源

👉首先第一点,要连接数据库的话,我们就要知道这个数据库在哪里

  • 这里我会介绍两种方式,一个是通过DataSource(数据源)对象获取,另一种则是通过DriverManager(驱动管理类)的静态方法获取,不过重点在于第一种

DataSource(数据源)对象获取

  • 下面是创建数据源的相关代码,我使用到了Java中的向上转型和向下转型,不是很清楚的同学可以先去了解一下
DataSource dataSource = new MysqlDataSource();
((MysqlDataSource)dataSource).setURL("jdbc:mysql://127.0.0.1:3306/test?characterEncoding=utf8&useSSL=false");
((MysqlDataSource)dataSource).setUser("root");
((MysqlDataSource)dataSource).setPassword("xxxxxxxxxxx");
  • 也就是这一句我使用到了向上转型,DataSource是Java中的一个接口,MysqlDataSource则是它的实现类,下面这种便是很典型的使用接口来实现向上转型
DataSource dataSource = new MysqlDataSource();
  • 然后下面的setURL()setUser()setPassWord()都是【MysqlDataSource】这个接口实现类中的一些方法,但是由于向上转型后父类无法去调用子类独有的方法,所以我们还要去进行一个向下转型
  • 通过去进行一个强制类型转换就可以做到,此时我们便能调用到实现类中的方法了
((MysqlDataSource)dataSource).setURL(...)
((MysqlDataSource)dataSource).setUser(...)
((MysqlDataSource)dataSource).setPassWord(...)

💬 那有同学问:为什么要这么麻烦呢?转上又转下?直接使用MysqlDataSource不就好了

  • 道理确实是这样,如果我们直接将代码写成下面这样子,那么就可以直接去调用这个类中的方法了,无需再去做一个向下转型但是这却隐藏着巨大的隐患…
MysqlDataSource dataSource1 = new MysqlDataSource();

可能没有真正做过开发的同学还体会不到这一点,因为我们在设计一些接口、类的时候始终要遵循【高内聚 · 低耦合】的思想,即类的内部各个方法之间要紧密关联,但类与类之间不要产生太大的依赖

  • 光这么说可能说不太清楚,有兴趣的同学可以先去了解一下高内聚 · 低耦合

  • 因为在上面我们使用的是【MysqlDataSource 】,所以在程序的其他方法中,我们可能也需要使用到这个类去做一个接受,就是因为这种高内聚的原因,导致了【MysqlDataSource 】被大量地进行扩散,此危害性就在于后续我们的项目要更换数据库的就需要改动很多的代码,加大了工作量!!!

void func1(MysqlDataSource dataSource1){}
void func2(MysqlDataSource dataSource2){}

然后呢我们再来说说为什么要去调用这些API

  • 首先的话是serURL(),对于【URL】来说,学习过HTTP协议的同学应该是非常熟悉,对于浏览器发送的HTTP请求中第一行就是URL,它叫做 统一资源定位符,也就是我们日常生活中所说的【网址】,描述了互联网上唯一一个资源的位置

我们可以来解读一下这个所传入的这个URL
在这里插入图片描述

  • 那对于下面的UserPassWord就很清楚了,就是用户名和密码,MySQL用户名固定就是【root】,密码的话就是你安装MySQL的时候手动设置的密码。
  • 虽然很多数据库是根据用户名和密码来进行认证的,但是有些数据库,向SQLite就不是用的这个进行认证,所以这就是为什么这些方法均在【MysqlDataSource 】里的原因了,每个数据库都有它们独特的验证方式,所以我们在使用不同数据库时加载不同的驱动包即可

2️⃣ 和数据库建立网络连接

在上面呢,我们只是进行了数据源的创建操作,只是描述了服务器在哪,并没有真正得进行访问,那现在我们所要做的操作就是【连接】,通过网络去进行一个通信

  • 然后我们就可以通过【DataSource】接口中一个方法叫做getConnection()这个方法去建立一个网络连接,它会返回一个实例为【Connection】的对象,不过从下图可以看到很多种类的Connection,那我们选哪个呢?
  • 这里要注意,选择这个java.sql的,这才是我们想要的Connection

在这里插入图片描述

Connection connection = dataSource.getConnection();
  • 要注意这里了连接(Connection),而不是链接(Link),对于连接而言并不复杂,因为只有这么一句代码,因为在网络中客户端和服务器之间要去建立一个连接,就只是找个地方记录下而已,并不需要非常繁琐的步骤
  • 而且还要注意的是这个getConnection()还需要去处理异常,一般来说像我们在进行文件读取、数据库操作都是需要去捕获这个异常的,这里我直接【throws】了,你也可以【try…catch】,如果异常还不是很了解的同学再去学习一下

在这里插入图片描述

打个比方说,就像民政局登记结婚一样,只是找个地方登记一下我们结婚了,也无需像婚礼那样大动干戈

在这里插入图片描述

DriverManager(驱动管理类)的静态方法获取

讲了这么多,别忘了我们只讲了一种数据库连接方式,别忘了还有第二种,那就是使用DriverManager

  • 但是我并不打算讲解这种方式,因为它需要借助我们在JavaSE中所学习的【反射】,但是对于反射来说其实是不属于常规的编程手段,不在特殊的情况下还是不推荐使用的。你可以在看了下面这两种连接方式的区别后再来回顾
// 加载JDBC驱动程序
Class.forName("com.mysql.jdbc.Driver");
// 创建数据库连接
Connection connection = DriverManager.getConnection(url);

区别:

  1. DriverManager类来获取的Connection连接,是无法重复利用的,每次使用完以后释放资源时,通过connection.close()都是关闭物理连接
  2. DataSource提供连接池的支持。连接池在初始化时将内置了【数据库连接池】,这些连接是可以复用的,每次使用完数据库连接,释放资源调用connection.close()都是将Conncetion连接对象回收

💬 可能有同学对这个【数据库连接池】不太理解,我们这里来讲一讲(doge)

  • 在Java中呢我们学习过很多的,比方说:String字符串常量池、进程池、线程池、内存池,它们都是池
  • 那谈到池这个东西我们就想到了【池塘】,那假设呢你现在是个妹子,长得好看,又有才华,所以追求者就与很多,但是呢同一时间只能谈一个对象,但是你又不想拒绝其他的追求者,因为如果当前这个对象谈到后面谈不来的话,就需要再换一个,此时的话就会有很多的成本,因为需要重新培养感情

在这里插入图片描述

  • 那于是你就想了一个很不错的办法来提高效率,那就是【养鱼🐟】
  • 怎么养鱼呢?也就是在和A谈对象的同时和B搞暧昧,同时培养感情基础,一旦你和A分手了,就可以立马和B在一起,此时大大提高了效率,那么这个B就被亲切地称之为【备胎】

在这里插入图片描述

  • 但是你的备胎可能还不只B这一个,可能还有CDEFGHI…,此时它们就构成了一个“备胎池”,若是B不合适了还可以换其他人上位,而且呢,这个时候A也还在池子里,若是你和其他人都不和的时候,又想到A了,此时你就可以再和A在一起

当然,上面只是举一个小小的案例,现实生活中我们不提倡这么去做~ 但是在计算机里这种做法还是很常见的,可以了解一下

3️⃣ 构造SQL语句

接下去呢,我们就要去构造SQL语句了

  • 首先定义一个字符串表示一条插入语句
String sql = "insert into student values (1, 'zhansan')";
  • 但是呢有这条语句还不够,在JDBC中,我们还需要搭配一个对象去描述这里的sql情况
PreparedStatement statement = connection.prepareStatement(sql);

💬 这里主要讲一下这个PreparedStatement ,可能有很多同学看懵了。首先对于【startment】对象来说,是将SQL语句发送到数据库中。JDBC API中主要提供了三种Statement对象

  1. Statement
  2. PreparedStatement
  3. CallableStartment

不过我们在工作中用得最多的还是PreparedStatement,字面意思的话就是【预处理语句

  • 对于PreparedStatement来说,它其实在底层帮我们做了许多事情,不过这里不做详解,有兴趣的同学可以去了解一下
    Java数据库JDBC——prepareStatement的用法和解释

4️⃣ 执行SQL语句

在构造完SQL语句后,我们就可以去通过这个statement对象去执行SQL语句了

  • 只需要调用这个类中的executeUpdate()即可,如果你要执行的SQL语句是【增】、【删】、【改】操作的话,那么你的SQL执行语句就写下面这一条就可以了
statement.executeUpdate();
  • 但若是你执行的是【查】操作的话,结果就不是这样了,而是要写成下面这句
statement.executeQuery();

5️⃣ 断开连接,释放资源

当然在执行完了之后还要再断开连接,释放掉必要的资源

  • 可以看到我们上面所创建出来的两个对象connectionstatement,但是你自己观察就可以发现我在关闭资源的时候是反着来的,这就是因为资源释放的顺序和创建的顺序是相反的
statement.close();
connection.close();
  • 如果读者学习过C++的话就会很好理解了,因为C++类和对象中存在【析构函数】,对象销毁的时候是存在顺序的;但是因为Java中有GC(垃圾自动回收机制),所以我们一般不需要自己去释放申请的空间,编译器会去做这件事的

这里我还是来讲一下这个构造析构的顺序,帮助读者来进行理解

  • 不知道你去过东北没有,反正那边是冬天是挺冷的❄,零下三四十度,不过北方的房子还是很有意思的,一般我们家里都是只有一个院子的,但是在北方一般都有两个院子,而且有两个门,你跨入第一个门相当于进入外院,再跨一个门才算是进入了内院(主院)
  • 那此时也就是意味着你要进入内院的话就需要先通过【大门】,然后再通过【二门】;但若是你要出来的话肯定就得先通过【二门】,才能到达【大门】(假设你不会飞🐦)
  • 那通过上面这样已说明,相信你对构造、析构的这么一个过程很熟悉了,后构造的对象就需要先析构才可以

在这里插入图片描述

4、测试观察

写好了JDBC连接MySQL数据库的代码逻辑,接下去我们就来测试一下是否真的可以去操作数据库

  • 首先可以看到的是当前这个数据表studennt中是不存在任何记录的

在这里插入图片描述

  • 然后我们到IDEA中去执行一下这段代码的话就可以看到ret的结果为1,表名SQL语句执行成功

在这里插入图片描述

  • 然后我们再到数据库中去查看的时候就可以发现有一条记录被插入进来了,这也就实现了用JDBC去操作数据表的功能

在这里插入图片描述
💬 如果你是第一次看到的话,一定会觉得很神奇,就像我在第一次看到后端代码可以与前端进行交互的时候,我也很吃惊;现在又看到可以通过后端的代码来操作数据库,又让我觉得耳目一新

5、代码优化

上面只是做了基本的测试,代表我们的代码可以去连接到数据库,但是这份代码还不够【完善】,我们再来完善一下~

1️⃣灵活变通SQL

  • 这是我们写的SQL语句,但你是否觉得这样去写太死了,通常我们在操作数据库的时候都会写很多SQL语句,是非常灵活的,但是就下面这样的话确实是不太好
String sql = "insert into student values (1, 'zhansan')";
  • 所以我们可以将代码改为下面这样,记录中的【学号】和【姓名】字段我们可以通过自己输入来进行控制
// 3.输入学号和姓名
System.out.println("请输入学号:");
Scanner sc1 = new Scanner(System.in);
int id = sc1.nextInt();
System.out.println("请输入姓名:");
Scanner sc2 = new Scanner(System.in);
String name = sc2.next();

// 4.构造一个 SQL 语句,来完成插入操作
//String sql = "insert into student values (1, 'zhansan')";
String sql = "insert into student values (" + id + ", '" + name + "')";
  • 此时我们通过控制台去进行输入就后构造出SQL语句然后执行就可以发现也是可以插入一条记录的,

在这里插入图片描述

2️⃣防止SQL注入攻击

但是对于上面这样的SQL语句构造你可以看出来是比较丑的,不方便阅读与理解,不仅如此,它还容易引起【SQL注入攻击】

  • 可能有的同学不太清楚这个【SQL注入攻击】,它是黑客入侵服务器的一种形式,黑客可以通过代码直接拼接构造sql语句导致sql语句结构可被恶意篡改,有兴趣的同学可以去了解一下 链接

对于上面这种危害,我们进行预防最靠谱的方案,就是使用PreparedStatement中占位符替换的方式,来实现SQL的构造~

  • 我们可以把SQL语句构造成下面这样,两个【?】就相当于是占位符
String sql = "insert into student values (?, ?)";       //通过占位符进行替换
  • 还记得我们上面使用到的那个statement对象吗,它可以用来描述SQL的情况,我们可以通过里面的【setInt()】和【setString()】方法来设置记录中的两个字段,从而将这个SQL语句构造完整
// jdbc中还需要搭配一个特定的对象,来搭配描述这里的sql情况
PreparedStatement statement = connection.prepareStatement(sql);
statement.setInt(1, id);
statement.setString(2, name);
  • 然后再去运行的话就可以看到也是可以把记录插入到数据表中的

在这里插入图片描述
💬 对于上面这种形式,就是较为安全的SQL语句写法,代码会在文末给出,可供读者参考

6、SQL查找的写法

上面的话就只是SQL的插入,学习过CRUD的同学们一定还会其他操作,对于【删除】、【修改】的话和插入是一模一样的,只需要修改一下SQL语句的构造即可。

👉但是对于【查找】操作就不一样了,和上面的这些操作相比,多了一个遍历结果集

  • 首先的话我们要去修改一下SQL语句,改为查询的语句
String sql = "select * from student";
  • 然后在执行SQL语句的时候就不能写成executeUpdate()了,而是要写成executeQuery(),而且返回的结果也不一样,若是【增】、【删】、【改】的话返回的就只是一个【int】类型的值,表示SQL语句是否执行成功
  • 但是对于【查】的话就不一样了,因为一个数据表中的记录不止一条,所以它查找就会返回一个结果集,我们要使用[ResultSet]来进行接受
ResultSet ret = statement.executeQuery();
  • 那既然这是一个集合的话,我们要看到里面的内容就要去进行一个遍历的操作,此时,和【迭代器】的遍历很相似,通过ret.next()去进行判断是否还有下一条记录需要遍历即可,内部去获取到记录中的相关字段,然后显示打印出来即可
while(ret.next())
{
    int id = ret.getInt("id");
    String name = ret.getString("name");
    System.out.println("id = " + id + "; name = " + name);
}
  • 执行程序就可以看到,遍历出了这张表中的所有内容

在这里插入图片描述

四、整体代码展示

上面就是通过JDBC连接数据库的这么一套流程,你学会了吗😎

JDBC往数据表插入记录

public class JDBCInsertDemo {
    public static void main(String[] args) throws SQLException {
        /**
         * 使用 JDBC 往数据库中插入一个记录
         * 需要提前准备好数据库(java106) 和 数据表(student)
         */

        // 1.创建数据源,描述了数据库服务器在哪
        DataSource dataSource = new MysqlDataSource();
        ((MysqlDataSource)dataSource).setURL("jdbc:mysql://127.0.0.1:3306/test?characterEncoding=utf8&useSSL=false");
        ((MysqlDataSource)dataSource).setUser("root");
        ((MysqlDataSource)dataSource).setPassword("panjialie1218");

        // 2.和数据库建立网络连接 (写JDBC实际上是实现一个MySQL客户端,通过网络和服务器进行通讯)
        Connection connection = dataSource.getConnection();

        // 3.输入学号和姓名
        System.out.println("请输入学号:");
        Scanner sc1 = new Scanner(System.in);
        int id = sc1.nextInt();
        System.out.println("请输入姓名:");
        Scanner sc2 = new Scanner(System.in);
        String name = sc2.next();

        // 4.构造一个 SQL 语句,来完成插入操作
        //String sql = "insert into student values (1, 'zhansan')";     -- 太死板
        //String sql = "insert into student values (" + id + ", '" + name + "')";   -- 会SQL注入攻击
        String sql = "insert into student values (?, ?)";       //通过占位符进行替换
        // jdbc中还需要搭配一个特定的对象,来搭配描述这里的sql情况
        PreparedStatement statement = connection.prepareStatement(sql);
        statement.setInt(1, id);
        statement.setString(2, name);

        // 5.执行 SQL 语句 (控制客户端给服务器发送请求)
        int ret = statement.executeUpdate();
        System.out.println("ret = " + ret);

        // 6.断开和数据库的连接,并且释放掉必要的资源
        statement.close();
        connection.close();
    }
}

JDBC查询数据表记录

public class JDBCSelectDemo {
    public static void main(String[] args) throws SQLException {
        /**
         * 使用 JDBC 往数据库中插入一个记录
         * 需要提前准备好数据库(java106) 和 数据表(student)
         */

        // 1.创建数据源,描述了数据库服务器在哪
        DataSource dataSource = new MysqlDataSource();
        ((MysqlDataSource)dataSource).setURL("jdbc:mysql://127.0.0.1:3306/test?characterEncoding=utf8&useSSL=false");
        ((MysqlDataSource)dataSource).setUser("root");
        ((MysqlDataSource)dataSource).setPassword("panjialie1218");

        // 2.和数据库建立网络连接 (写JDBC实际上是实现一个MySQL客户端,通过网络和服务器进行通讯)
        Connection connection = dataSource.getConnection();

        // 3.构造一个 SQL 语句,来完成插入操作
        String sql = "select * from student";
        // jdbc中还需要搭配一个特定的对象,来搭配描述这里的sql情况
        PreparedStatement statement = connection.prepareStatement(sql);

        // 5.执行 SQL 语句 (控制客户端给服务器发送请求)
        ResultSet ret = statement.executeQuery();
        while(ret.next())
        {
            int id = ret.getInt("id");
            String name = ret.getString("name");
            System.out.println("id = " + id + "; name = " + name);
        }

        // 6.断开和数据库的连接,并且释放掉必要的资源
        statement.close();
        connection.close();
    }
}

五、总结与提炼

最后来总结一下本文所学习的内容📖

  • 在本文中,我们首先了解了JDBC的概念以及其相关工作原理,然后的话就通过调用这些JDBC相关的API去加载MySQL数据库的驱动,一步步地去创建数据源、建立连接、构造与执行SQL语句,最后释放连接,完成了通过后端代码去操作数据库,成功往数据表中插入了一条记录
  • 接下去呢我们又对代码去进行了一个优化,既提高了代码的灵活性,还可以防止SQL注入攻击,加固了代码的安全性。然后又完成了查询数据表的相关记录的操作,需要涉及对集合的遍历

💬 可能本文的读者不会很多,但如果你读到这里的话那我要和你说一个事实:因为现在几乎都不用JDBC了,市面上有一个很成熟的框架叫做【MyBatis】,它是⼀款优秀的持久层框架,它⽀持⾃定义 SQL、存储过程以及⾼级映射,但是其底层的实现还是基于JDBC的。框架都是层出不穷的,唯有内部核心是永恒不变的,学到底层才是关键🙂

以上就是本文要介绍的所有内容,感谢您的阅读🌹

在这里插入图片描述

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

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

相关文章

【AI提示】ChatGPT提示工程课程(吴恩达OpenAI)推理文本(中文chatgpt版)

设置 Setup产品评论文本情感(正面/负面)Sentiment (positive/negative)识别情绪类型从客户评论中提取产品和公司名称一次完成多项任务Inferring topics 推断主题为某些主题制作新闻提醒 Inferring 推理 在本课中,您将从产品评论和新闻文章中推…

计算机专业主要学习什么

2020计算机专业主要学习什么 1计算机专业都学习哪些方面的知识 1、可视化编程 掌握编程方法和可视化技术,熟悉一个可视化平台及其软件开发技术。能够获取Delphi编程人员系列,Java Basic或VB开发专家认证。 就业方向:企业,政府&…

Elasticsearch:验证 Elasticsearch Docker 镜像并安装 Elasticsearch

Elasticsearch 可以作为 Docker 镜像使用。 www.docker.elastic.co 上提供了所有已发布的 Docker 图像和标签的列表。 源文件在 Github 中。此软件包包含免费和订阅功能。 开始 30 天试用以试用所有功能。 从 Elasticsearch 8.0 开始,默认启用安全性。 启用安全性后…

为什么很多企业把35岁视为分水岭

(点击即可收听) 为什么很多企业把35岁视为分水岭 有时候,别人的故事,若干年后,就是自己的故事,只要身在互联网这个行业里,可以说,每个人都避免不了35岁危机 不要五十步笑百步 前阵子,朋友圈一位行业知名大佬,35岁,每天兢兢业业,任劳任怨,本以为安稳渡过3个月试用期,正快要转正时…

AutoEncoder GAN

AE Auto-Encoder (AE) 是20世纪80年代晚期提出的,它是一种无监督学习算法,使用了反向传播算法,让目标值等于输入值。 是神经网络的一种,经过训练后能尝试将输入复制到输出。三层网络结构:输入- 隐层- 输出自编码网络…

一道经典的网红面试题:从URL输入到页面展现到底发生了什么?

目录 🏮 前言 一、URL 到底是啥 二、域名解析(DNS) 三、TCP 三次握手 四、发送 HTTP 请求 五、服务器处理请求并返回 HTTP 报文 六、浏览器解析渲染页面 1.根据 HTML 解析 DOM 树 2.根据 CSS 解析生成 CSS 规则树 3.结合 DOM 树和…

MySQL高级篇复盘笔记(一)【存储引擎、索引、SQL优化、视图、触发器、MySQL管理】

❤ 作者主页:欢迎来到我的技术博客😎 ❀ 个人介绍:大家好,本人热衷于Java后端开发,欢迎来交流学习哦!( ̄▽ ̄)~* 🍊 如果文章对您有帮助,记得关注、点赞、收藏、…

解决“Adobe Premiere安装后无法启动“问题

昨天在Win10 专业版上安装Adobe Premiere2019(后面简称 Adobe Pr2019)时,发现Pr2019安装之后无法启动,有时,桌面上有时会产生空白的Premiere图标。换了好几个Adobe Premirere安装包,都是如此。正在苦思冥想中,突然灵机…

YOLOv5独家原创改进,ShuffleNetV2网络结构,改进ShuffleNetV2准确率低问题

目录 一、介绍1、轻量化目标检测的瓶颈2、YOLOv5和ShuffleNetV2的概述 二、ShuffleNetV2的架构1、ShuffleNetV2的基本单元——ShuffleNet Unit2、ShuffleNetV2的网络结构 三、ShuffleNetV2的特点1、高效的通道重排操作2、逐通道矩阵乘法3、轻量级和高精度的平衡 四、YOLOv51、Y…

【2023 AI 写作工具大盘点】国内外 45 款免费 AI 写作神器汇总,轻松成为创作高手!

0. 未来百科 未来百科(https://nav.6aiq.com),是一个知名的AI产品导航网站 —— 为发现全球优质AI工具而生 。目前已 聚集全球3000优质AI工具产品 ,旨在帮助用户发现全球最好的AI工具,同时为研发AI垂直应用的创业公司提供展示窗口&#xff0c…

蓝海项目拼多多海外版Temu让外国人也体验砍一刀,普通人的机会

拼多多海外版Temu是一款专为全球华人打造的购物平台,它的出现为海外华人提供了更加便捷、实惠的购物体验。Temu的诞生,不仅是拼多多在海外市场的一次尝试,更是对全球华人消费需求的一次深入了解和满足。 一、Temu的背景 拼多多是中国最大的社交电商平台之一,它的出现改…

SaaS CRM系统的优势,与本地部署相比哪个更方便?

CRM系统主要有两种部署方式,分别是云部署和本地部署。那么,本地部署CRM软件真的比SaaS CRM好吗?本文将为您分析两种部署方式的区别,来为您答疑解惑。 云部署CRM的含义: 云部署CRM系统是指将CRM系统部署在云端&#x…

vue字符串拼接的多种方法

在 vue项目中,我们可以使用多个不同的方法来拼接字符串。今天我们就来介绍一下 vue中各种方法的用法。 第一种方法:使用 lodash进行字符串拼接,这是最简单的一个方法,它最大的缺点就是它比较慢,需要时间去执行拼接&…

解密高并发场景下的集合类问题,让程序更高效稳定!

大家好,我是小米,一个热爱技术分享的小伙伴。在日常开发中,我们经常会使用集合类来处理数据,但在高并发场景下,集合类可能会遇到一些线程安全的问题。今天,我们就来探讨一下集合类在高并发中是如何解决问题…

(转载)基于遗传算法的多目标优化算法(matlab实现)

1 理论基础 1.1 多目标优化及Pareto最优解 多目标优化问题可以描述如下: 其中,f(x)为待优化的目标函数;x为待优化的变量;Ib和ub分别为变量x的下限和上限约束;Aeq*xbeq为变量x的线性等式约束;A*x≤b为变…

Web基础 ( 六 ) AJAX

4.6.AJAX 4.6.1.什么是ajax Ajax(Asynchronous JavaScript and XML , Asynchronous 异步的)指的是一种使用 JavaScript、XML 和 HTTP 请求进行前端数据异步交互的技术。Ajax 不需要刷新整个页面就可以更新其中的一部分,使得网页的反应更快、…

RabbitMQ学习-延迟队列

延迟队列 背:也就是给队列设置个过期时间,然后到时间消息变成死信,消费死信队列中的消息就行,再没什么玩意,演示队列优化就是不给队列这只TTL,再生产者代码中消息里面设置消息TTL,因为 RabbitM…

ElasticSearch——Docker安装ElasticSearch和Kibana

Docker安装ElasticSearch 说明:由于是用docker安装,所以要确保已安装docker并docker环境可用。 docker安装步骤:https://wanli.blog.csdn.net/article/details/121445768 1、Docker安装ElasticSearch 获取指定版本的ES镜像 拉取镜像&#…

Ubuntu安装RabbitMQ server - 在ubuntu+cpolar+rabbitMQ环境下,实现mq服务端远程访问

文章目录 前言1.安装erlang 语言2.安装rabbitMQ3. 内网穿透3.1 安装cpolar内网穿透(支持一键自动安装脚本)3.2 创建HTTP隧道 4. 公网远程连接5.固定公网TCP地址5.1 保留一个固定的公网TCP端口地址5.2 配置固定公网TCP端口地址 转载自cpolar内网穿透的文章:无公网IP&…

nodejs+vue社区重点人员户籍信息查询系统

为了迎合时代需求,优化管理效率,各种各样的管理系统应运而生,各行各业相继进入信息管理时代,重点人员信息查询就是信息时代变革中的产物之一。 任何系统都要遵循系统设计的基本流程,本系统也不例外,同样需要…