【Android】Room—数据库的基本操作
引言
在Android开发中,数据持久化是一个不可或缺的部分。随着应用的复杂度增加,选择合适的数据存储方式变得尤为重要。Room数据库作为Android Jetpack架构组件之一,提供了一种抽象层,使得开发者能够以更简洁、更安全的方式操作SQLite数据库。本文将带你深入了解Room数据库的基本概念、使用方式以及最佳实践。
核心组件
Room是一个持久化库,它提供了一个抽象层,用于在SQLite数据库中存储和查询数据。它通过注解处理器和编译时检查,确保数据库操作的类型安全,减少了运行时错误的可能性。
Entity(实体):
- Entity 是一个注解,用于标记一个类作为数据库中的一个表。
- 每个 Entity 都映射到数据库中的一个表。
- 你可以使用注解来定义表的名称(
@PrimaryKey
用于定义主键,@Entity(tableName = "name")
),以及定义列(@Column
)。 - 一个 Entity 类通常包含数据字段和适当的 getter 和 setter 方法。
Dao(数据访问对象):
- Dao 是一个接口,它定义了对数据库的操作,如插入、查询、更新和删除。
- 每个 Dao 都与一个 Entity 相关联,并且定义了与该 Entity 相关的数据库操作。
- 你可以在 Dao 接口中使用注解来定义 SQL 语句,如
@Query
、@Insert
、@Update
、@Delete
等。 - Room 会根据 Dao 接口自动生成实现代码。
Database(数据库):
- Database 是一个抽象类,它定义了整个数据库的结构,包括所有的 Dao。
- 它使用
@Database
注解来标记,并且可以定义版本号和包含的实体。 - 你可以使用
Room.databaseBuilder()
方法来创建数据库实例。 - 通常,一个应用中只有一个数据库类,它包含了所有的 Dao。
示例
只说概念可能会觉得抽象还是没有理解,接下来就通过一个示例来看看吧!
在使用room之前一定不要忘记它是要添加依赖的
implementation("androidx.room:room-runtime:2.4.3")
annotationProcessor("androidx.room:room-compiler:2.4.3")
我们就设置一个学生示例吧,进行学生信息的增删改查操作,我们将学生的信息展示在页面上,设置增删改查操作的按钮,这部分的代码就不做说明了,你一定非常熟悉了。
- 定义Entity:我们要存放的是学生信息,先创建一个学生类:
@Entity(tableName = "student")
public class student {@PrimaryKey(autoGenerate = true)@ColumnInfo(name = "id", typeAffinity = ColumnInfo.INTEGER)int id;@ColumnInfo(name = "name", typeAffinity = ColumnInfo.TEXT)String name;@ColumnInfo(name = "age", typeAffinity = ColumnInfo.INTEGER)int age;public student(int id, String name, int age) {this.id = id;this.name = name;this.age = age;}@Ignore //告诉Room不要用这个构造方法,这是我们所用的方法public student(String name, int age) {this.name = name;this.age = age;}@Ignorepublic student(int id) {this.id = id;}public int getAge() {return age;}
}
里面的方法大家已经写了很多遍了,接下来就给大家解释里面的注解:
@Entity(tableName = "student")
:- 这个注解标记了这个类是一个数据库表的映射。
tableName
属性指定了数据库中表的名称,这里是"student"
。
- 这个注解标记了这个类是一个数据库表的映射。
@PrimaryKey(autoGenerate = true)
:- 这个注解标记了一个字段作为表的主键。
autoGenerate = true
表示主键值是自动生成的,通常是自增的。
- 这个注解标记了一个字段作为表的主键。
@ColumnInfo(name = "id", typeAffinity = ColumnInfo.INTEGER)
:- 这个注解提供了字段的额外信息。
name
属性指定了数据库中列的名称,这里是"id"
。typeAffinity
属性指定了列的数据类型,这里是ColumnInfo.INTEGER
,表示这个字段在数据库中是整型。
- 这个注解提供了字段的额外信息。
ColumnInfo.INTEGER
:表示列的数据类型为整型。ColumnInfo.TEXT
:表示列的数据类型为文本。ColumnInfo.REAL
:表示列的数据类型为浮点数。ColumnInfo.BLOB
:表示列的数据类型为二进制数据(例如图片或文件)。ColumnInfo.FLOAT
:表示列的数据类型为浮点数,与REAL
类似,但更明确表示为浮点数。ColumnInfo.LONG
:表示列的数据类型为长整型。ColumnInfo.SHORT
:表示列的数据类型为短整型。ColumnInfo.BOOLEAN
:表示列的数据类型为布尔值。
@Ignore
:- 这个注解用于告诉Room忽略接下来的构造函数,不将其作为数据库操作的一部分。这通常用于那些仅用于应用逻辑,而不是数据库操作的构造函数。
- 定义DAO:创建一个接口来定义数据访问对象(DAO),使用
@Dao
注解标记。在这个接口中,定义方法来执行数据库操作,如插入、查询、更新和删除。
@Dao
public interface StudentDao {@Insertvoid insertStudent(student... students);@Deletevoid deleteStudent(student... students);@Updatevoid updateStudent(student... students);@Query("SELECT * FROM student")List<student> getAllStudent();@Query("SELECT * FROM student WHERE id = :id")List<student> getStudentById(int id);
}
都是接口当中定义的方法名,这里的注解大家也都见过,在学习SQLite的时候对这四个方法用了很多次了,这里是直接使用注解,来标记这个方法。看最后一个注解就是为了告诉它根据表的id来寻找你所要查找的学生信息。
- 定义Database:创建一个抽象类来继承
RoomDatabase
,并使用@Database
注解标记。在这个类中,定义数据库的版本和包含的实体和DAO。
@Database(entities = {student.class}, version = 1, exportSchema = false)
public abstract class MyDataBase extends RoomDatabase {private static MyDataBase mInstance;private static final String DATABASE_NAME = "my_db.db";public static synchronized MyDataBase getInstance(Context context) {if (mInstance == null) {mInstance = Room.databaseBuilder(context.getApplicationContext(), MyDataBase.class, DATABASE_NAME).build();}return mInstance;}public abstract StudentDao getStudentDao(); //Room会帮我们自动实现
}
@Database(entities = {student.class}, version = 1, exportSchema = false)
:@Database
注解用于定义数据库的配置。它告诉Room这个数据库包含哪些实体(entities
),数据库的版本(version
),以及其他一些配置项。entities = {student.class}
:指定了这个数据库包含的实体类。在这个例子中,它包含student
实体。version = 1
:指定了数据库的版本号。当数据库结构发生变化时(例如添加、删除或修改实体),需要增加这个版本号。exportSchema = false
:指定是否允许Room导出数据库的schema文件。如果设置为true
,Room会在编译时生成一个包含数据库schema的文件,这有助于调试和测试。在这个例子中,它被设置为false
,意味着不导出schema文件。
public abstract class MyDataBase extends RoomDatabase
:- 定义了一个名为
MyDataBase
的抽象类,它继承自RoomDatabase
。这个类将作为数据库的顶层接口,用于创建和管理数据库实例。
- 定义了一个名为
private static MyDataBase mInstance;
:- 定义了一个静态的
MyDataBase
实例,用于实现数据库的单例模式。这样可以确保整个应用程序中只有一个数据库实例。
- 定义了一个静态的
private static final String DATABASE_NAME = "my_db.db";
:- 定义了一个常量,指定了数据库文件的名称。在这个例子中,数据库文件将被命名为
my_db.db
。
- 定义了一个常量,指定了数据库文件的名称。在这个例子中,数据库文件将被命名为
public static synchronized MyDataBase getInstance(Context context)
:- 定义了一个静态方法,用于获取数据库的单例实例。这个方法使用了
synchronized
关键字来确保线程安全,避免在多线程环境下创建多个数据库实例。 context.getApplicationContext()
:获取应用程序级别的上下文,用于创建数据库实例。
- 定义了一个静态方法,用于获取数据库的单例实例。这个方法使用了
if (mInstance == null) { mInstance = Room.databaseBuilder(...).build(); }
:- 如果
mInstance
为null
,则使用Room.databaseBuilder()
方法创建一个新的数据库实例。这个方法链式调用了多个配置方法,包括指定数据库类、数据库名称、以及数据库构建的其他配置。
- 如果
public abstract StudentDao getStudentDao();
:- 定义了一个抽象方法,用于获取
StudentDao
的实例。Room在编译时会为这个方法生成实现代码,这样你就可以在应用程序中通过调用getStudentDao()
来获取StudentDao
的实例,进而执行数据库操作。
- 定义了一个抽象方法,用于获取
- 接下来就可以为按钮注册点击事件,运行程序看数据的更改了
public class MainActivity extends AppCompatActivity {StudentDao studentDao;@Overrideprotected void onCreate(Bundle savedInstanceState) {//......获取按钮,页面的滚动控件buttonAdd.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {student s1 = new student("Jack", 20);student s2 = new student("Rose", 30);new InsertStudentTask(studentDao).execute(s1, s2);}});buttonResearch.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {new GetAllStudentTask(studentDao).execute();}});buttonUpdata.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {student s1 = new student(3,"Tason", 21);new UpdataStudentTask(studentDao).execute(s1);}});buttonDelete.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {student s1 = new student(2);new DeleteStudentTask(studentDao).execute(s1);}});}class DeleteStudentTask extends AsyncTask<student, Void, Void> {private StudentDao studentDao;public DeleteStudentTask(StudentDao studentDao) {this.studentDao = studentDao;}@Overrideprotected Void doInBackground(student... students) {studentDao.deleteStudent(students);return null;}}class UpdataStudentTask extends AsyncTask<student, Void, Void> {private StudentDao studentDao;public UpdataStudentTask(StudentDao studentDao) {this.studentDao = studentDao;}@Overrideprotected Void doInBackground(student... students) {studentDao.updateStudent(students);return null;}}class InsertStudentTask extends AsyncTask<student, Void, Void> {private StudentDao studentDao;public InsertStudentTask(StudentDao studentDao) {this.studentDao = studentDao;}@Overrideprotected Void doInBackground(student... students) {studentDao.insertStudent(students);return null;}}class GetAllStudentTask extends AsyncTask<Void, Void, List<student>> {private StudentDao studentDao;public GetAllStudentTask(StudentDao studentDao) {this.studentDao = studentDao;}@Overrideprotected List<student> doInBackground(Void... voids) {return studentDao.getAllStudent();}@Overrideprotected void onPostExecute(List<student> students) {super.onPostExecute(students);studentRecyclerViewAdapter.setstudent(students);studentRecyclerViewAdapter.notifyDataSetChanged();}}
}
注意我们不在主线程进行数据库的相关操作,
接下来就运行一下:
当我们进行增加操作,页面没有发生变化,我们点击查询操作,我们在上面的代码当中知道没当按下查询操作就会获取所有的学生信息,并将其展示在页面的滚动控件当中:
再看看删除和修改操作吧,上面的代码我们将2号学生删除,并修改3号学生的信息,操作后按下查询按钮:
优化
我们看到数据库的信息已经修改了。每次修改都要进行查询才能看到更新的信息就很麻烦,而且你也不知道到底有没有跟新就要进行查询操作,在之前我们学习了LiveData每当数据变化,就会自动告诉View,使其自动更新,那不就大大优化了程序吗
官方给出的架构指南,Model当中使用Room访问SQLite的内容。接下来就看看如何使用吧!
1、2、3步是一样的,这里就不多说了,只是对于数据库的操作有稍微的修改,第一次学习写的就不修改了,大家看看吧
@Dao
public interface StudentDao {@Insertvoid insertStudent(student... students);@Deletevoid deleteStudent(student... students);@Updatevoid updateStudent(student... students);@Query("DELETE FROM student")void deleteAllAtudent();@Query("SELECT * FROM student")LiveData<List<student>> getAllStudentsLive();
}
- 创建Repository:Repository作为数据层的抽象,封装了数据来源,它可以是一个类,包含了一系列方法来执行数据库操作。这些方法通常会调用DAO中定义的操作,并将结果包装成LiveData或Flow对象,以便ViewModel可以观察数据变化。
public class StudentRepository {private StudentDao studentDao;public StudentRepository(Context context) {MyDataBase dataBase = MyDataBase.getInstance(context);this.studentDao = dataBase.getStudentDao();}//对数据进行添加public void insertStudent(student... students) {new InsertStudentTask(studentDao).execute(students);}class InsertStudentTask extends AsyncTask<student, Void, Void> {private StudentDao studentDao;public InsertStudentTask(StudentDao studentDao) {this.studentDao = studentDao;}@Overrideprotected Void doInBackground(student... students) {studentDao.insertStudent(students);return null;}}//对数据进行修改public void updateStudent(student... students) {new UpdataStudentTask(studentDao).execute(students);}class UpdataStudentTask extends AsyncTask<student, Void, Void> {private StudentDao studentDao;public UpdataStudentTask(StudentDao studentDao) {this.studentDao = studentDao;}@Overrideprotected Void doInBackground(student... students) {studentDao.updateStudent(students);return null;}}//对数据进行删除public void deleteStudent(student... students) {new DeleteStudentTask(studentDao).execute(students);}class DeleteStudentTask extends AsyncTask<student, Void, Void> {private StudentDao studentDao;public DeleteStudentTask(StudentDao studentDao) {this.studentDao = studentDao;}@Overrideprotected Void doInBackground(student... students) {studentDao.deleteStudent(students);return null;}}//对数据进行查找public LiveData<List<student>> getAllStudentsLive() {return studentDao.getAllStudentsLive();}//将所有数据都删除public void deleteAllStudent() {new DeleteAllStudentTask(studentDao).execute();}class DeleteAllStudentTask extends AsyncTask<Void, Void, Void> {private StudentDao studentDao;public DeleteAllStudentTask(StudentDao studentDao) {this.studentDao = studentDao;}@Overrideprotected Void doInBackground(Void... voids) {studentDao.deleteAllAtudent();return null;}}
}
- 构造函数:
StudentRepository
的构造函数接收一个Context
对象,用它来获取数据库的实例。MyDataBase.getInstance(context)
是一个单例模式的数据库实例获取方法,确保整个应用中只有一个数据库实例。
- 异步任务(AsyncTask):
InsertStudentTask
、UpdataStudentTask
、DeleteStudentTask
和DeleteAllStudentTask
是内部类,它们继承自AsyncTask
。这些类用于在后台线程上执行数据库操作,以避免阻塞主线程。AsyncTask
已经被标记为过时,推荐使用java.util.concurrent
包中的类、Kotlin Coroutines
或者其他现代的并发解决方案。
- doInBackground方法:
- 在
AsyncTask
的doInBackground
方法中执行实际的数据库操作。这个方法在后台线程上运行,接收的参数是传递给execute
方法的参数。
- 在
- LiveData:
getAllStudentsLive
方法返回一个LiveData
对象,它包含了数据库中所有学生数据的列表。LiveData
是一个可观察的数据存储器,它具有生命周期感知能力,只会在观察者处于活跃状态时发送数据更新。
- 使用ViewModel:ViewModel负责管理UI相关的数据,它可以调用Repository中的方法来获取数据,并根据数据变化更新UI。ViewModel使用LiveData或Flow作为数据的载体,这样当数据发生变化时,UI可以自动更新。
public class StudentViewModel extends AndroidViewModel {//当要使用上下文的时候使用AndroidViewModelprivate StudentRepository repository;public StudentViewModel(@NonNull Application application) {super(application);this.repository = new StudentRepository(application);}public void insertStudent(student... students) {repository.insertStudent(students);}public void deleteStudent(student... students) {repository.deleteStudent(students);}public void deleteAllStudent() {repository.deleteAllStudent();}public void updateStudent(student... students) {repository.updateStudent(students);}public LiveData<List<student>> research() {return repository.getAllStudentsLive();}
}
- 观察数据变化:在UI层(如Activity或Fragment),观察ViewModel中的LiveData或Flow对象。当数据发生变化时,UI层会收到通知并更新界面。
studentViewModel = new ViewModelProvider(this, new ViewModelProvider.AndroidViewModelFactory(getApplication())).get(StudentViewModel.class);
studentViewModel.research().observe(this, new Observer<List<student>>() {@Overridepublic void onChanged(List<student> students) {studentRecyclerViewAdapter.setstudent(students);studentRecyclerViewAdapter.notifyDataSetChanged();}
}); //为其注册监听,到获取到的数据库内容进行变化,就更新UI
buttonAdd.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {student s1 = new student("Jack", 20);student s2 = new student("Rose", 30);studentViewModel.insertStudent(s1,s2);}
}); //只展示一个按钮的点击事件,其他的你也肯定会写了
接下来就运行程序看看吧!
我们进行了增加删除修改操作,页面直接自己更新了,带来了很大的便利
按下清空页面就没有了,数据库就被清空了!
本篇内容介绍了数据库的基本操作,感谢你的阅读!
文章到这里就结束了!
相关文章:

【Android】Room—数据库的基本操作
引言 在Android开发中,数据持久化是一个不可或缺的部分。随着应用的复杂度增加,选择合适的数据存储方式变得尤为重要。Room数据库作为Android Jetpack架构组件之一,提供了一种抽象层,使得开发者能够以更简洁、更安全的方式操作SQ…...

「数组」堆排序 / 大根堆优化(C++)
目录 概述 核心概念:堆 堆结构 数组存堆 思路 算法过程 up() down() Code 优化方案 大根堆优化 Code(pro) 复杂度 总结 概述 在「数组」快速排序 / 随机值优化|小区间插入优化(C)中,我们介绍了三种基本排序中的冒泡…...

Edegex Foundry docker和源码安装
edgex文档下载 https://github.com/edgexfoundry/edgex-docs/branches/all 在线文档查看 首先要安装python3环境 然后后安装 打开超级终端 #pip3 install mkdocs #mkdocs serve 在浏览器中输入 http://127.0.0.1:8000/edgex-docs/2.3/ 即可打开在线文档 edgex入门可以参考…...

阿里P8和P9级别有何要求
阿里巴巴的P8和P9级别,代表着公司的资深技术专家或管理者岗位,要求候选人具有丰富的职业经历、深厚的技术能力以及出色的领导力。以下是对P8和P9级别的要求、考察点以及准备建议的详细分析。 P8 级别要求 1. 职业经历: 8年以上的工作经验&a…...

【目标检测数据集】锯子数据集1107张VOC+YOLO格式
数据集格式:Pascal VOC格式YOLO格式(不包含分割路径的txt文件,仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数):1107 标注数量(xml文件个数):1107 标注数量(txt文件个数):1107 标注…...

移动产业处理器接口(MIPI)协议是什么?
未来汽车的宏伟愿景备受瞩目,特别是驱动这些汽车的技术更是成为焦点。如今,传感器对于汽车视觉和安全技术的下一阶段至关重要,因为驾驶员和乘客都依赖于它们。这些传感器能够支持众多应用,这些应用往往基于人工智能(AI…...

OpenAI o1:隐含在训练与推理间的动态泛化与流形分布
随着OpenAI o1发布,进一步激发了产业与学术各界对AGI的期待以及new scaling law下的探索热情,也看到来自社区和专业机构对o1的阐释,但总感觉还差点什么,因此决定以自己的角度分篇幅梳理下,并分享给大伙: O…...

沉浸式体验和评测Meta最新超级大语言模型405B
2024年7月23日, 亚马逊云科技的AI模型托管平台Amazon Bedrock正式上线了Meta推出的超级参数量大语言模型 - Llama 3.1模型,小李哥也迫不及待去体验和试用了该模型,那这么多参数量的AI模型究竟强在哪里呢?Llama 3.1模型是Meta&…...

Python 课程10-单元测试
前言 在现代软件开发中,单元测试 已成为一种必不可少的实践。通过测试,我们可以确保每个功能模块在开发和修改过程中按预期工作,从而减少软件缺陷,提高代码质量。而测试驱动开发(TDD) 则进一步将测试作为开…...

【嵌入式硬件开发基础】Arduino板常用外设及应用:MPU6050空间运动传感器(简介,类库函数,卡尔曼滤波),继电器(原理介绍,含应用实例/代码)
当一个人不能拥有的时候,他唯一能做的便是不要忘记。 🎯作者主页: 追光者♂🔥 🌸个人简介: 📝[1] CSDN 博客专家📝 🏆[2] 人工智能领域优质创作者🏆 🌟[3] 2022年度博客之星人工智能领域TOP4🌟 🌿[4] 2023年城市之星领跑者TOP1(哈尔滨…...

Pandas Series对象创建,属性,索引及运算详解
目录 Series对象创建 实例化参数 index参数 选用array-like创建Series对象 list ndarray 显示索引与隐式索引 选用dict创建Series对象 不指定索引 指定索引 选用标量创建Series对象 使用标量创建的广播机制 Series属性 name size shape index values Series索…...
优化算法(一)—遗传算法(Genetic Algorithm)附MATLAB程序
遗传算法(Genetic Algorithm, GA)是一种启发式搜索算法,用于寻找复杂优化问题的近似解。它模拟了自然选择和遗传学中的进化过程,主要用于解决那些传统算法难以处理的问题。 遗传算法的基本步骤: 初始化种群࿰…...

高等数学 2.3 高阶导数
一般地,函数 y f ( x ) y f(x) yf(x) 的导数 y ′ f ′ ( x ) y\ f\ (x) y ′f ′(x) 仍然是 x x x 的函数。我们把 y ′ f ′ ( x ) y\ f\ (x) y ′f ′(x) 的导数叫做函数 y f ( x ) y f(x) yf(x) 的二阶导数,记作 y ′ ′ y\ y ′…...

app抓包 chrome://inspect/#devices
一、前言: 1.首先不支持flutter框架,可支持ionic、taro 2.初次需要翻墙 3.app为debug包,非release 二、具体步骤 1.谷歌浏览器地址:chrome://inspect/#devices qq浏览器地址:qqbrowser://inspect/#devi…...

SAP自动化-ME12批量更新某行价格
Python源码 #-Begin-----------------------------------------------------------------#-Includes-------------------------------------------------------------- import sys, win32com.client import os#-Sub Main----------------------------------------------------…...

数据库系统 第58节 概述源码示例
深入探讨数据库技术,我们将通过具体的源代码示例来进一步解释数据库分区、复制、集群和镜像等高级特性。 数据库分区的源代码示例 哈希分区 在PostgreSQL中,可以使用哈希分区来创建一个分区表: CREATE TABLE measurements (city_id …...

软件设计师——程序设计语言
目录 低级语言和高级语言 编译程序和解释程序 正规式,词法分析的一个工具 有限自动机 编辑 上下文无关法 编辑 中后缀表示法 杂题 编辑 低级语言和高级语言 编译程序和解释程序 计算机只能理解由0、1序列构成的机器语言,因此高级程序设计…...

【在Linux世界中追寻伟大的One Piece】五种IO模型和阻塞IO
目录 1 -> 五种IO模型 1.1 -> 阻塞IO(Blocking IO) 1.2 -> 非阻塞IO(Non-blocking IO) 1.3 -> 信号驱动IO(Signal-Driven IO) 1.4 -> IO多路转接(IO Multiplexing) 1.5 -> 异步IO(Asynchronous IO) 2 -> 高级IO概念 2.1 -> 同步通信VS异步通信…...

nginx实现权重机制(nginx基础配置二)
在上一篇文章中我们已经完成了对轮询机制的测试,详情请看轮询机制。 接下来我们进行权重机制的测试 一、conf配置 upstream backServer{ server 127.0.0.1:8080 weight2; server 127.0.0.1:8081 weight1; } server { listen 80; server_name upstream.boyatop.cn…...

华为的仓颉和ArkTS这两门语言有什么区别
先贴下官网: ArkTs官网 仓颉官网 ArkTS的官网介绍说,ArkTS是TypeScript的进一步强化版本,简单来说就是包含了TS的风格,但是做了一些改进。 了解TypeScript的朋友都应该知道,其实TypeScript就是JavaScript的改进版本&…...

(SERIES10)DM逻辑备份还原
1 概念 逻辑备份还原是对数据库逻辑组件(如表、视图和存储过程等数据库对象)的备份还原。逻辑导出(dexp)和逻辑导入(dimp)是 DM 数据库的两个命令行工具,分别用来实现对 DM 数据库的逻辑备份和逻…...

Java零基础-StringBuilder类详解
哈喽,各位小伙伴们,你们好呀,我是喵手。运营社区:C站/掘金/腾讯云/阿里云/华为云/51CTO;欢迎大家常来逛逛 今天我要给大家分享一些自己日常学习到的一些知识点,并以文字的形式跟大家一起交流,互…...

免费爬虫软件“HyperlinkCollector超链采集器v0.1”
HyperlinkCollector超链采集器单机版v0.1 软件采用python的pyside2和selenium开发,暂时只支持window环境,抓取方式支持普通程序抓取和selenium模拟浏览器抓取。软件遵守robots协议。 首先下载后解压缩,然后运行app目录下的HyperlinkCollector.exe 运行…...

OPENAIGC开发者大赛企业组AI黑马奖 | AIGC数智传媒解决方案
在第二届拯救者杯OPENAIGC开发者大赛中,涌现出一批技术突出、创意卓越的作品。为了让这些优秀项目被更多人看到,我们特意开设了优秀作品报道专栏,旨在展示其独特之处和开发者的精彩故事。 无论您是技术专家还是爱好者,希望能带给您…...

k8s(kubernetes)的PV / PVC / StorageClass(理论+实践)
NFS总是不支持PVC扩容 先来个一句话总结:PV、PVC是K8S用来做存储管理的资源对象,它们让存储资源的使用变得可控,从而保障系统的稳定性、可靠性。StorageClass则是为了减少人工的工作量而去自动化创建PV的组件。所有Pod使用存储只有一个原则&…...

前端Excel热成像数据展示及插值算法
🎬 江城开朗的豌豆:个人主页 🔥 个人专栏:《 VUE 》 《 javaScript 》 📝 个人网站 :《 江城开朗的豌豆🫛 》 ⛺️生活的理想,就是为了理想的生活! 目录 📘 前言 📘一、热成像数…...

VBA_NZ系列工具NZ01: VBA二维码应用技术
我的教程一共九套及VBA汉英手册一部,分为初级、中级、高级三大部分。是对VBA的系统讲解,从简单的入门,到数据库,到字典,到高级的网抓及类的应用。大家在学习的过程中可能会存在困惑,这么多知识点该如何组织…...

小明震惊OpenAI 的新模型 01
在硅谷的中心,繁忙的咖啡馆和创业中心周围,年轻的软件工程师小明坐在他的办公桌前,面露困惑。科技界一直在盛传一项新的AI突破,但他持怀疑态度,不敢抱太大希望。他认为AI泡沫即将破灭,炒作列车即将出轨&…...

Clickhouse使用笔记
clickhouse官方文档:https://clickhouse.com/docs/zh/sql-reference/data-types/decimal 一,建表 create table acitivity_user_record ( id String DEFAULT generateUUIDv4(), -- 主键自增 activityId String, userId String, userName Nullable(Strin…...

基于高通主板的ARM架构服务器
一、ARM架构服务器的崛起 (一)市场需求推动 消费市场寒冬,全球消费电子需求下行,服务器成半导体核心动力之一。Arm 加速布局服务器领域,如 9 月推出 Neoverse V2。长久以来,x86 架构主导服务器市场&#…...