Java操作数据库 —— JDBC ① 基础篇
我走我的路,有人拦也走,没人陪也走
—— 24.6.7
JDBC
JDBC就是使用Java语言操作关系型数据库的一套API
一、JDBC简介
JDBC 概念
JDBC 就是使用Java语言操作关系型数据库的一套API
全称:(Java DataBase Connectivity)意为Java 数据库连接
JDBC 本质:
① 官方(sun公司)定义的一套操作所有关系型数据库的规则,即接口
② 各个数据库厂商去实现这套接口,提供数据库驱动jar包
③ 我们可以使用这套接口(JDBC)编程,真正执行的代码是驱动jar包中的实现类

JDBC 好处:
① 各数据库厂商使用相同的接口,Java代码不需要针对不同数据库分别开发
② 可随时替换底层数据库,访问数据库的Java代码基本不变
二、JDBC快速入门
步骤




package JavaJDBCBase;import java.sql.*;public class Demo1JDBCQuick {public static void main(String[] args) throws ClassNotFoundException, SQLException {// 1. 注册驱动Class.forName("com.mysql.cj.jdbc.Driver");// 2.获取数据库连接String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";// 获取链接对象Connection conn = DriverManager.getConnection(url,username,password);// 3.获取执行sql的对象,statement(把sql语句发送给MySQL)Statement stmt = conn.createStatement();// 4.编写sql语句并执行,以及接收返回的结果集String sql = "select emp_id,emp_name,emp_salary,emp_age from t_tmp";ResultSet result = stmt.executeQuery(sql);// 5.处理结果,遍历result结果集 next方法,判断有没有下一行while (result.next()) {int empId = result.getInt("emp_id");String empName = result.getString("emp_name");double empSalary = result.getDouble("emp_salary");int empAge = result.getInt("emp_age");System.out.println(empId+"\t"+empName+"\t"+empSalary+"\t"+empAge);}// 6.释放资源 先开启后关闭原则result.close();stmt.close();conn.close();}
}

三、JDBC 核心API 详解
1.注册驱动
Class.forName("com.mysql.cj.jdbc.Driver");
在 Java 中,当使用 JDBC(Java Database Connectivity)连接数据库时,需要加载数据库特定的驱动程序,以便与数据库进行通信。加载驱动程序的目的是为了注册驱动程序使得 JDBC API能够识别并与特定的数据库进行交互。
// 1. 注册驱动
// Class.forName("com.mysql.cj.jdbc.Driver");DriverManager.registerDriver(new Driver());
从JDK6开始,不再需要显式地调用 class.forName()来加载JDBC 驱动程序,只要在类路径中集成了对应的jar文件,会自动在初始化时注册驱动程序。
2.Connection
Connection接口是JDBC API的重要接口,用于建立与数据库的通信通道。换而言之,Connection对象不为空则代表一次数据库连接。
在建立连接时,需要指定数据库URL、用户名、密码参数。
URL:jdbc:mysql://localhost:3306/atguigu"
jdbc:mysql://IP地址:端口号/数据库名称 ? 参数键值对1 & 参数键值对2
Connection 接口还负责管理事务,Connection 接口提供了 commit 和 rollback 方法,用于提交事务和回滚事务。
可以创建 statement 对象,用于执行 SQL语句并与数据库进行交互。
在使用JDBC技术时,必须要先获取Connection对象,在使用完毕后,要释放资源,避免资源占用浪费及泄漏。
3.Statement(了解)
Statement 接口用于执行 SQL语句并与数据库进行交互。它是 JDBC API 中的一个重要接口。通过Statement 对象,可以向数据库发送 SQL语句并获取执行结果。
结果可以是一个或多个结果。
增删改:受影响行数单个结果。
查询:单行单列、多行多列、单行多列等结果。
但是 Statement 接口在执行SQL语句时,会产生SQL注入攻击问题:
当使用 statement 执行动态构建的 SQL查询时,往往需要将查询条件与SQL语句拼接在一起,直接将参数和SQL语句一并生成,让SQL的查询条件始终为true得到结果。
4.PreparedStatement
Preparedstatement是 Statement 接口的子接口,用于执行 预编译的 SQL查询,作用如下。
预编译SQL语句:在创建Preparedstatement时,就会预编译SQL语句,也就是SQL语句已经固定
防止SQL注入: Preparedstatement 支持参数化查询,将数据作为参数传递到SQL语句中,采用?占位符的方式,将传入的参数用一对单引号包裹起来",无论传递什么都作为值。有效防止传入关键字或值导致SQL注入问题。
性能提升:Preparedstatement是预编译SQL语句,同一SQL语句多次执行的情况下,可以复用,不必每次重新编译和解析。
import java.sql.*;
import java.util.Scanner;public class Demo2PreparedStatement {public static void main(String[] args) throws Exception {// 1.注册驱动// 2.获取链接对象String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";Connection conn = DriverManager.getConnection(url,username,password);// 3.获取执行sql语句对象PreparedStatement preparedStatement = conn.prepareStatement("select emp_id,emp_name,emp_salary,emp_age from t_tmp where emp_id=?");System.out.println("请输入员工编号:");Scanner sc = new Scanner(System.in);String name = sc.nextLine();// 4.为?占位符赋值,并执行sql语句,并执行,接受返回的结果preparedStatement.setString(1,name);ResultSet resultSet = preparedStatement.executeQuery();// 5.处理结果,遍历esultSetwhile(resultSet.next()){int emp_id = resultSet.getInt("emp_id");String emp_name = resultSet.getString("emp_name");int emp_age = resultSet.getInt("emp_age");double emp_salary = resultSet.getDouble("emp_salary");System.out.println(emp_id+"\t"+emp_name+"\t"+emp_age+"\t"+emp_salary);}// 6.释放资源resultSet.close();preparedStatement.close();conn.close();}
}

package JavaJDBCBase;import java.sql.*;
import java.util.Scanner;public class Demo2PreparedStatement {public static void main(String[] args) throws Exception {// 1.注册驱动// 2.获取链接对象String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";Connection conn = DriverManager.getConnection(url,username,password);// 3.获取执行sql语句对象PreparedStatement preparedStatement = conn.prepareStatement("select emp_id,emp_name,emp_salary,emp_age from t_tmp where emp_name=? || emp_salary=?");System.out.println("请输入员工姓名:");Scanner sc = new Scanner(System.in);String name = sc.nextLine();System.out.println("请输入员工工资");double salary = sc.nextDouble();// 4.为?占位符赋值,并执行sql语句,并执行,接受返回的结果 int类型:参数的下标,从0开始,要替换的值为多少preparedStatement.setString(1,name);preparedStatement.setDouble(2,salary);ResultSet resultSet = preparedStatement.executeQuery();// 5.处理结果,遍历resultSetwhile(resultSet.next()){int emp_id = resultSet.getInt("emp_id");String emp_name = resultSet.getString("emp_name");int emp_age = resultSet.getInt("emp_age");double emp_salary = resultSet.getDouble("emp_salary");System.out.println(emp_id+"\t"+emp_name+"\t"+emp_age+"\t"+emp_salary);String goal = resultSet.getString("emp_id");System.out.println(emp_name+"id为:"+goal);}// 6.释放资源resultSet.close();preparedStatement.close();conn.close();}
}

5.ResultSet
ResultSet 是JDBC API中的一个接口,用于表示从数据库中 执行査询语句所返回的结果集。它提供了一种于遍历和访问查询结果的方式。
遍历结果:Resultset可以使用 next()方法将游标移动到结果集的下一行,逐行遍历数据库查询的结果,返回值为boolean类型,true代表有下一行结果,false则代表没有。
获取单列结果:可以通过get类型()的方法获取单列的数据,该方法为重载方法,支持索引和列名进行获取。
String goal = resultSet.getString("emp_name");System.out.println(goal);
package JavaJDBCBase;import java.sql.*;
import java.util.Scanner;public class Demo2PreparedStatement {public static void main(String[] args) throws Exception {// 1.注册驱动// 2.获取链接对象String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";Connection conn = DriverManager.getConnection(url,username,password);// 3.获取执行sql语句对象PreparedStatement preparedStatement = conn.prepareStatement("select emp_id,emp_name,emp_salary,emp_age from t_tmp where emp_id=?");System.out.println("请输入员工编号:");Scanner sc = new Scanner(System.in);String name = sc.nextLine();// 4.为?占位符赋值,并执行sql语句,并执行,接受返回的结果preparedStatement.setString(1,name);ResultSet resultSet = preparedStatement.executeQuery();// 5.处理结果,遍历esultSetwhile(resultSet.next()){int emp_id = resultSet.getInt("emp_id");String emp_name = resultSet.getString("emp_name");int emp_age = resultSet.getInt("emp_age");double emp_salary = resultSet.getDouble("emp_salary");System.out.println(emp_id+"\t"+emp_name+"\t"+emp_age+"\t"+emp_salary);String goal = resultSet.getString("emp_name");System.out.println(goal);}// 6.释放资源resultSet.close();preparedStatement.close();conn.close();}
}

四、基于Preparedstatement实现CRUD
1.查询单行单列
// 单行单列查询@Testpublic void testQuerySingleRowAndCol() throws SQLException {// 1.注册驱动// 2.获取连接String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";Connection connection = DriverManager.getConnection(url, username, password);// 3.预编译SQL语句得到PreparedStatement对象PreparedStatement preparedStatement = connection.prepareStatement("select count(*) as count from t_tmp");// 4.执行语句 获取结果ResultSet resultSet = preparedStatement.executeQuery();// 5.处理结果,进行遍历,如果明确只有一个结果,那么resultSet至少要做一次next的判断,才能拿到我们要的列的结果while (resultSet.next()) {// 用下标获取int anInt = resultSet.getInt(1);System.out.println(anInt);}// 6.释放资源resultSet.close();preparedStatement.close();connection.close();}
2.查询单行多列
// 单行多列查询@Testpublic void testQuerySingleRowAndCol2() throws SQLException {// 1.注册驱动// 2.获取链接Connection connection = DriverManager.getConnection("jdbc:mysql://localhost:3306/JDBC", "root", "954926928lcl");// 3.预编译SQL语句获得PreparedStatement对象PreparedStatement preparedStatement = connection.prepareStatement("select emp_id,emp_name,emp_salary,emp_age from t_tmp where emp_id = ?");// 4.为占位符赋值,然后执行,并接受结果preparedStatement.setInt(1,5);ResultSet resultSet = preparedStatement.executeQuery();// 5.处理结果while (resultSet.next()) {int empId = resultSet.getInt("emp_id");String empName = resultSet.getString("emp_name");double empSalary = resultSet.getDouble("emp_salary");int empAge = resultSet.getInt("emp_age");System.out.println(empId+"\t"+empName+"\t"+empSalary+"\t"+empAge);}// 6.资源释放resultSet.close();preparedStatement.close();connection.close();}
3.查询多行多列
// 多行多列查询@Testpublic void testQueryMoreRow() throws SQLException {String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";Connection conn = DriverManager.getConnection(url,username,password);PreparedStatement preparedStatement = conn.prepareStatement("select emp_id,emp_name,emp_salary,emp_age from t_tmp where emp_age > ?");preparedStatement.setInt(1,25);ResultSet resultSet = preparedStatement.executeQuery();while (resultSet.next()) {int empId = resultSet.getInt("emp_id");String empName = resultSet.getString("emp_name");double empSalary = resultSet.getDouble("emp_salary");int empAge = resultSet.getInt("emp_age");System.out.println(empId+"\t"+empName+"\t"+empSalary+"\t"+empAge);}resultSet.close();preparedStatement.close();conn.close();}
4.新增
// 新增@Testpublic void testInsert() throws SQLException {String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";// 获取链接对象Connection conn = DriverManager.getConnection(url,username,password);PreparedStatement preparedStatement = conn.prepareStatement("insert into t_tmp(emp_name,emp_salary,emp_age) values(?,?,?)");preparedStatement.setString(1,"rose");preparedStatement.setDouble(2,345.67);preparedStatement.setInt(3,36);int result = preparedStatement.executeUpdate();// 根据受影响行数做判断,得到成功或失败if (result>0){System.out.println("成功!");}else{System.out.println("失败!");}preparedStatement.close();conn.close();}
5.修改
// 修改@Testpublic void testUpdate() throws SQLException {String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";// 获取链接对象Connection conn = DriverManager.getConnection(url,username,password);PreparedStatement preparedStatement = conn.prepareStatement("update t_tmp set emp_salary = ? where emp_id=?");preparedStatement.setDouble(1,888.88);preparedStatement.setInt(2,6);int result = preparedStatement.executeUpdate();if (result>0){System.out.println("成功");}else{System.out.println("失败");}preparedStatement.close();conn.close();}
6.删除
// 删除@Testpublic void testDelete() throws SQLException {String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";// 获取链接对象Connection conn = DriverManager.getConnection(url,username,password);// 手写SQL语句PreparedStatement preparedStatement = conn.prepareStatement("delete from t_tmp where emp_id=?");preparedStatement.setInt(1,5);int result = preparedStatement.executeUpdate();if (result>0){System.out.println("成功");}else{System.out.println("失败");}preparedStatement.close();conn.close();}
7.总结
package JavaJDBCBase;// 在类中写入@Test,自动导入org.junit.Test,便不用重复写main函数
import org.junit.Test;import java.sql.*;public class Demo3JDBCOperation {// 单行单列查询@Testpublic void testQuerySingleRowAndCol() throws SQLException {// 1.注册驱动// 2.获取连接String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";Connection connection = DriverManager.getConnection(url, username, password);// 3.预编译SQL语句得到PreparedStatement对象PreparedStatement preparedStatement = connection.prepareStatement("select count(*) as count from t_tmp");// 4.执行语句 获取结果ResultSet resultSet = preparedStatement.executeQuery();// 5.处理结果,进行遍历,如果明确只有一个结果,那么resultSet至少要做一次next的判断,才能拿到我们要的列的结果while (resultSet.next()) {// 用下标获取int anInt = resultSet.getInt(1);System.out.println(anInt);}// 6.释放资源resultSet.close();preparedStatement.close();connection.close();}// 单行多列查询@Testpublic void testQuerySingleRowAndCol2() throws SQLException {// 1.注册驱动// 2.获取链接Connection connection = DriverManager.getConnection("jdbc:mysql://localhost:3306/JDBC", "root", "954926928lcl");// 3.预编译SQL语句获得PreparedStatement对象PreparedStatement preparedStatement = connection.prepareStatement("select emp_id,emp_name,emp_salary,emp_age from t_tmp where emp_id = ?");// 4.为占位符赋值,然后执行,并接受结果preparedStatement.setInt(1,5);ResultSet resultSet = preparedStatement.executeQuery();// 5.处理结果while (resultSet.next()) {int empId = resultSet.getInt("emp_id");String empName = resultSet.getString("emp_name");double empSalary = resultSet.getDouble("emp_salary");int empAge = resultSet.getInt("emp_age");System.out.println(empId+"\t"+empName+"\t"+empSalary+"\t"+empAge);}// 6.资源释放resultSet.close();preparedStatement.close();connection.close();}// 多行多列查询@Testpublic void testQueryMoreRow() throws SQLException {String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";Connection conn = DriverManager.getConnection(url,username,password);PreparedStatement preparedStatement = conn.prepareStatement("select emp_id,emp_name,emp_salary,emp_age from t_tmp where emp_age > ?");preparedStatement.setInt(1,25);ResultSet resultSet = preparedStatement.executeQuery();while (resultSet.next()) {int empId = resultSet.getInt("emp_id");String empName = resultSet.getString("emp_name");double empSalary = resultSet.getDouble("emp_salary");int empAge = resultSet.getInt("emp_age");System.out.println(empId+"\t"+empName+"\t"+empSalary+"\t"+empAge);}resultSet.close();preparedStatement.close();conn.close();}// 新增@Testpublic void testInsert() throws SQLException {String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";// 获取链接对象Connection conn = DriverManager.getConnection(url,username,password);PreparedStatement preparedStatement = conn.prepareStatement("insert into t_tmp(emp_name,emp_salary,emp_age) values(?,?,?)");preparedStatement.setString(1,"rose");preparedStatement.setDouble(2,345.67);preparedStatement.setInt(3,36);int result = preparedStatement.executeUpdate();// 根据受影响行数做判断,得到成功或失败if (result>0){System.out.println("成功!");}else{System.out.println("失败!");}preparedStatement.close();conn.close();}// 修改@Testpublic void testUpdate() throws SQLException {String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";// 获取链接对象Connection conn = DriverManager.getConnection(url,username,password);PreparedStatement preparedStatement = conn.prepareStatement("update t_tmp set emp_salary = ? where emp_id=?");preparedStatement.setDouble(1,888.88);preparedStatement.setInt(2,6);int result = preparedStatement.executeUpdate();if (result>0){System.out.println("成功");}else{System.out.println("失败");}preparedStatement.close();conn.close();}// 删除@Testpublic void testDelete() throws SQLException {String url = "jdbc:mysql://localhost:3306/JDBC";// 用户名String username = "root";// 密码String password = "954926928lcl";// 获取链接对象Connection conn = DriverManager.getConnection(url,username,password);// 手写SQL语句PreparedStatement preparedStatement = conn.prepareStatement("delete from t_tmp where emp_id=?");preparedStatement.setInt(1,5);int result = preparedStatement.executeUpdate();if (result>0){System.out.println("成功");}else{System.out.println("失败");}preparedStatement.close();conn.close();}
}

五、常见问题
1.资源的管理

2.SQL语句问题

3.SQL语句未设置参数问题

4.用户名或密码错误问题

5.通信异常

相关文章:
Java操作数据库 —— JDBC ① 基础篇
我走我的路,有人拦也走,没人陪也走 —— 24.6.7 JDBC JDBC就是使用Java语言操作关系型数据库的一套API 一、JDBC简介 JDBC 概念 JDBC 就是使用Java语言操作关系型数据库的一套API 全称:(Java DataBase Connectivity)意为Java 数据库连接 JDBC 本质: ①…...
webpack和vite区别
一、Webpack 1. 概述 Webpack 是一个模块打包工具,它会递归地构建依赖关系图,并将所有模块打包成一个或多个bundle(包)。 2. 特点 配置灵活:Webpack提供了高度可定制的配置文件,可以根据项目需求进行各…...
FL Studio21永久免费破解中文版下载,让我这个音乐制作爱好者如获至宝!
FL Studio21永久免费破解中文版下载,让我这个音乐制作爱好者如获至宝!🎶 这款软件功能强大,操作简单易上手。我可以轻松地创作出各种风格的音乐作品。无论是流行、摇滚还是电子音乐,都能轻松驾驭。🎧 使用F…...
vue3 监听器,组合式API的watch用法
watch函数 在组合式 API 中,我们可以使用 watch 函数在每次响应式状态发生变化时触发回调函数 watch(ref,callback(newValue,oldValue),option:{}) ref:被监听的响应式量,可以是一个 ref (包括计算属性)、一个响应式…...
苹果WWDC开幕发布AI大模型,股价却跌近2%
KlipC报道:北京时间6月11日凌晨,苹果一年一度的“全球开发者大会”(WWDC)开幕。会上,先后介绍了iOS 18、iPadOS 18、watchOS 11等系统的更新,同时还展示了多个AI功能。宣布与OpenAI构建合作伙伴关系。然而&…...
C++ 11 【可变参数模板】【lambda】
💓博主CSDN主页:麻辣韭菜💓 ⏩专栏分类:C修炼之路⏪ 🚚代码仓库:C高阶🚚 🌹关注我🫵带你学习更多C知识 🔝🔝 目录 前言 一、新的类功能 1.1默认成员函数—…...
c 宏应用举例
1.概要 #include <iostream> //变量可以直接使用 #define fun() a 100; //用变量计算可以 #define fun2(a) a*2; //用变量替换可以 #define fun3(a) d[a] a; //##链接的作用,一般用于链接变量名 #define fun4(type,name) type name##_s 4; //#的作用是转换…...
微信公众号(公众平台) 和 微信开放平台的scope的差异
微信公众号(公众平台) 和 微信开放平台 是两码事。 公众号(公众平台)获取的scope只包括两种:snsapi_base 和snsapi_userinfo,前者是静默获取,用户无感知;后者是需要用户确认同意的。…...
基于pytorch实现的DenseUnet医学图像分割(腹部多脏器)
1、前言 本章将介绍将densenet的主干网络引入unet中 官方实现的代码:kits19-challenge/network at master nitsaick/kits19-challenge (github.com) 本章实现的项目目录如下: 主要代码有train、evaluate、predict脚本 2、代码介绍 数据预处理脚本 数据…...
富格林:正规策划实现安全做单
富格林悉知,在投资理财的过程中,最重要的是控制风险实现安全做单避免损失。但是市场客观因素带来的风险并不能完全避免,因此投资者需要采取一些正规技能来减低风险投资风险实现安全做单。接下来就由富格林给大家分享一些实现安全做单的正规方…...
02. 异常捕捉和处理
检索特定内容的邮件,当检索失败,就会在终端输出“获取不了值” try: #代码块A except: #代码B 试一下运行代码A,当代码A报错时,执行代码B 这是main_exe.py文件中的内容 略过 #今天 for job_name,end_time in zip(bji.job_inf…...
Oracle和mysql中插入时间字段
例如有id 和 times两个字段 Oracle insert into xxx values|(1,sysdate) mysql insert into xxx values(1,now()) 在 MySQL 中,SYSDATE() 函数也是可用的,它与 NOW() 类似,但略有不同: NOW…...
注册小程序
每个小程序都需要在 app.js 中调用 App 方法注册小程序实例,绑定生命周期回调函数、错误监听和页面不存在监听函数等。 详细的参数含义和使用请参考 App 参考文档 。 整个小程序只有一个 App 实例,是全部页面共享的。开发者可以通过 getApp 方法获取到全…...
【YOLOv8改进[CONV]】使用MSBlock二次创新C2f模块实现轻量化 + 含全部代码和详细修改方式 + 手撕结构图 + 轻量化 + 涨点
本文将使用MSBlock二次创新C2f模块实现轻量化,助力YOLOv8目标检测效果的实践,文中含全部代码、详细修改方式以及手撕结构图。助您轻松理解改进的方法,实现有效涨点。 改进前和改进后的参数对比: 目录 一 MSBlock 二 使用MSBlock二次创新C2f模块实现轻量化 1 整体修改 …...
three.js使用环境贴图或者加载hdr图
1、three.js使用环境贴图 1.1、效果视频 环境贴图 1.2、使用步骤(个人认为) (1)导入引入相关方法 (2)创建场景 (3)创建相机 (4)添加物体材质 (5…...
GPT-4o多模态大模型的架构设计
GPT-4o:大模型风向,OpenAI大更新 OpenAI震撼发布两大更新!桌面版APP与全新UI的ChatGPT上线,简化用户操作,体验更自然。同时,全能模型GPT-4o惊艳亮相,跨模态即时响应,性能卓越且性价比…...
Facebook:社交世界的引领者
导语 在当今数字化时代,Facebook已经成为了人们社交生活的重要一环。然而,除了成为社交媒体的象征外,它还在不断探索并领导着社交世界的新方向。 1. 社交平台的发展者 Facebook不仅仅是一个社交平台,更是社交方式的引领者。从其…...
qt 加载字体 c++
目录 qt 加载字体 c label设置大小和字体: 资源配置路径失败 解决方法:exe相对目录: pro配置: resource.qrc qt 加载字体 c #include <QApplication> #include <QLabel> #include <QFontDatabase> #incl…...
Linux ldd和ldconfig
ldconfig ldconfig 查看默认库路径和ld.so.conf包含的库路径,来建立运行时动态装载的库查找路径。 ldconfig命令的用途,主要是在默认搜寻目录(/lib和/usr/lib)以及动态库配置文件/etc/ld.so.conf内所列的目录下,搜索出可共享的动态链接库(格式如前介绍,lib*.so*),…...
Python 学习flask创建项目
1、使用pycharm创建flask项目 2、运行访问地址 3、可以看到访问地址内容 4、可以增加路由,尝试访问获取参数...
从零实现富文本编辑器#5-编辑器选区模型的状态结构表达
先前我们总结了浏览器选区模型的交互策略,并且实现了基本的选区操作,还调研了自绘选区的实现。那么相对的,我们还需要设计编辑器的选区表达,也可以称为模型选区。编辑器中应用变更时的操作范围,就是以模型选区为基准来…...
基于当前项目通过npm包形式暴露公共组件
1.package.sjon文件配置 其中xh-flowable就是暴露出去的npm包名 2.创建tpyes文件夹,并新增内容 3.创建package文件夹...
React19源码系列之 事件插件系统
事件类别 事件类型 定义 文档 Event Event 接口表示在 EventTarget 上出现的事件。 Event - Web API | MDN UIEvent UIEvent 接口表示简单的用户界面事件。 UIEvent - Web API | MDN KeyboardEvent KeyboardEvent 对象描述了用户与键盘的交互。 KeyboardEvent - Web…...
2025 后端自学UNIAPP【项目实战:旅游项目】6、我的收藏页面
代码框架视图 1、先添加一个获取收藏景点的列表请求 【在文件my_api.js文件中添加】 // 引入公共的请求封装 import http from ./my_http.js// 登录接口(适配服务端返回 Token) export const login async (code, avatar) > {const res await http…...
现代密码学 | 椭圆曲线密码学—附py代码
Elliptic Curve Cryptography 椭圆曲线密码学(ECC)是一种基于有限域上椭圆曲线数学特性的公钥加密技术。其核心原理涉及椭圆曲线的代数性质、离散对数问题以及有限域上的运算。 椭圆曲线密码学是多种数字签名算法的基础,例如椭圆曲线数字签…...
uniapp微信小程序视频实时流+pc端预览方案
方案类型技术实现是否免费优点缺点适用场景延迟范围开发复杂度WebSocket图片帧定时拍照Base64传输✅ 完全免费无需服务器 纯前端实现高延迟高流量 帧率极低个人demo测试 超低频监控500ms-2s⭐⭐RTMP推流TRTC/即构SDK推流❌ 付费方案 (部分有免费额度&#x…...
Device Mapper 机制
Device Mapper 机制详解 Device Mapper(简称 DM)是 Linux 内核中的一套通用块设备映射框架,为 LVM、加密磁盘、RAID 等提供底层支持。本文将详细介绍 Device Mapper 的原理、实现、内核配置、常用工具、操作测试流程,并配以详细的…...
安宝特方案丨船舶智造的“AR+AI+作业标准化管理解决方案”(装配)
船舶制造装配管理现状:装配工作依赖人工经验,装配工人凭借长期实践积累的操作技巧完成零部件组装。企业通常制定了装配作业指导书,但在实际执行中,工人对指导书的理解和遵循程度参差不齐。 船舶装配过程中的挑战与需求 挑战 (1…...
七、数据库的完整性
七、数据库的完整性 主要内容 7.1 数据库的完整性概述 7.2 实体完整性 7.3 参照完整性 7.4 用户定义的完整性 7.5 触发器 7.6 SQL Server中数据库完整性的实现 7.7 小结 7.1 数据库的完整性概述 数据库完整性的含义 正确性 指数据的合法性 有效性 指数据是否属于所定…...
【JVM面试篇】高频八股汇总——类加载和类加载器
目录 1. 讲一下类加载过程? 2. Java创建对象的过程? 3. 对象的生命周期? 4. 类加载器有哪些? 5. 双亲委派模型的作用(好处)? 6. 讲一下类的加载和双亲委派原则? 7. 双亲委派模…...
