1.BoltDB源码解析(六)Get操作
2.使用Sequelize快速构建PostgreSQL数据的任务任务CRUD操作详解
3.如何看待acwing闫学灿(y总)多次表示学习源码没用?
4.client-go 源码分析(4) - ClientSet客户端 和 DynamicClient客户端
5.Mysql走天下教你简单实现CRUD操作mysql中curd
BoltDB源码解析(六)Get操作
在我们深入了解BoltDB的DB文件结构后,接下来我们将分析其CRUD操作中的源码源码Get方法。首先来看Bucket的任务任务Get API,这个操作相对简单,源码源码无论是任务任务读事务还是写事务,都可以通过它获取Bucket中指定key的源码源码诱导源码吧value。以下是任务任务关键代码:
代码的核心是Cursor对象,其seek方法在B-tree上定位key,源码源码返回存储在B-tree页面中的任务任务key和value指针。
特别需要注意的源码源码是,如果查到的任务任务value是另一个Bucket,函数会返回nil,源码源码因为Get方法主要针对普通key,任务任务而非Bucket。源码源码如果需要操作Bucket,任务任务应使用Bucket方法,如rootBucket.Bucket("user"),就像在MySQL中操作表一样。
Get方法和Bucket查找过程相似,都通过Cursor.seek定位,但Bucket方法多了openBucket步骤。相似的原因在于BoltDB将Bucket视为value类型存储在B-tree中,这样可以共用一个数据结构来存储Bucket和普通value,Cursor.seek则负责在不分类型的B-tree中查找。
seek方法会在key不存在时返回大于该key的源码底层原理下一个key,这有利于通用性,包括insert、update和delete操作。Cursor的search方法递归查找,根据isLeaf属性决定是在node还是page上进行。
BoltDB的写事务会先copy页面到node进行修改,因此读写操作在node和page的处理有所区别。Cursor的search方法根据当前事务类型,选择在node(写事务)或page(读事务)上搜索。
searchNode和searchPage分别针对node和page执行递归搜索,使用一个stack记录递归路径,确保Cursor能够支持遍历B-tree的操作。BoltDB的高效体现在读操作中,全程基于mmap的page指针操作,实现了真正的零拷贝。
使用Sequelize快速构建PostgreSQL数据的CRUD操作详解
之前写过一个专栏《布道API》来介绍API的REST风格及推荐实践,今天开始来构建一个管理系统的API服务,首先需要处理的就是数据存储,本文将结合实际开发总结在NodeJS下使用Sequelize快速构建PostgreSQL数据的CRUD操作。项目源代码:github.com/QuintionTang/pretender-service
SequelizeSequelize是一个基于promise的Node.jsORM工具,它具有强大的事务支持、关联关系、预读和延迟加载、读取复制等功能,箱体支撑源码支持的数据库包括:PostgreSQL、MySQL、MariaDB、SQLite和MSSQL。
Sequelize类是引用sequlize模块后获取一个顶级对象,通过它来创建sequlize实例,也可以通过该对象来获取模内其它对象的引用,如:Utils工具类、Transaction事务类等。创建实例后,可以通过实例来创建或定义Model(模型)、执行查询、同步数据库结构等操作。
官方网站:docs.sequelizejs.com/
添加和配置在安装模块之前,首先安装开发工具Sequelize-CLI
sudonpminstall-gsequelize-cli接下来在项目目录下安装数据存储相关的模块。
npminstallsequelize--savenpminstallpgpg-hstore--save现在在项目根目录下创建文件.sequelizerc,代码如下:
constpath=require('path');module.exports={ "config":path.resolve('./config','db.json'),"models-path":path.resolve('./models'),'seeders-path':path.resolve('./seeders'),'migrations-path':path.resolve('./migrations')};该文件将告诉Sequelize初始化,以生成config、models到特定目录。接下来,输入命令初始化Sequelize。
sequelizeinit该命令将创建config/db.json,models/index.js,migrations和seeders目录和文件。成都源码网站命令执行完毕之后打开并编辑config/db.json来配置数据库连接信息。
{ "development":{ "username":"dbusername","password":"dbpassword","database":"crayon-admin","host":".0.0.1","dialect":"postgres","options":{ "operatorsAliases":false},"logging":false},"test":{ "username":"dbusername","password":"dbpassword","database":"crayon-admin","host":".0.0.1","dialect":"postgres"},"production":{ "username":"dbusername","password":"dbpassword","database":"crayon-admin","host":".0.0.1","dialect":"postgres"}}目录说明:
migrations:所有迁移文件,通过sequelizedb:migrate创建相应数据表
seeders:种子文件,即初始化需要插入到数据库中的数据,运行sequelizedb:seed:all
创建Models和Migrations使用CLI工具Sequelize-CLI创建administrators表
sequelizemodel:create--nameadministrators--attributesid:integer,add_time:integer,last_login:integer,username:string,email:string,login_ip:string执行后会生成两个文件
/src/migrations/-create-administrators.js:创建数据表脚本,用于数据库初始化。
"usestrict";module.exports={ up:async(queryInterface,Sequelize)=>{ awaitqueryInterface.createTable("administrators",{ id:{ allowNull:false,autoIncrement:true,primaryKey:true,type:Sequelize.INTEGER,},add_time:{ type:Sequelize.INTEGER,},last_login:{ type:Sequelize.INTEGER,},username:{ type:Sequelize.STRING,},password:{ type:Sequelize.STRING,},email:{ type:Sequelize.STRING,},login_ip:{ type:Sequelize.STRING,},});},down:async(queryInterface,Sequelize)=>{ awaitqueryInterface.dropTable("administrators");},};/src/models/administrators.js:生成的model文件
"usestrict";const{ Model}=require("sequelize");module.exports=(sequelize,DataTypes)=>{ classadministratorsextendsModel{ }administrators.init({ id:{ type:DataTypes.INTEGER,autoIncrement:true,primaryKey:true,},add_time:DataTypes.INTEGER,last_login:DataTypes.INTEGER,username:DataTypes.STRING,password:DataTypes.STRING,email:DataTypes.STRING,login_ip:DataTypes.STRING,},{ sequelize,indexes:[{ unique:true,fields:["id"],},],freezeTableName:true,timestamps:false,//是否自动添加时间戳createAt,updateAtmodelName:"administrators",});returnadministrators;};现在执行命令:
sequelizedb:migrate执行成功后将在连接的数据库中创建数据表:administrators。
创建seedseed用于初始化插入数据,如管理员,在系统运行前需要创建一个默认账号,这些默认账号信息就写在seed文件中。
创建seed命令如下:
sequelizeseed:create--nameadministrator执行成功后将会在seeders文件夹中创建文件,修改代码如下:
npminstallsequelize--savenpminstallpgpg-hstore--save0现在将seed中的数据插入到数据库中,执行一下命令:
npminstallsequelize--savenpminstallpgpg-hstore--save1创建Services创建文件夹services,文件夹中代码封装与model交互的方法,包括所有CRUD(创建,读取,更新和删除)操作,创建administrators.js,实现的逻辑为获取账号信息、更新账号信息,代码如下:
npminstallsequelize--savenpminstallpgpg-hstore--save2创建Controllers上面创建的services文件用于控制器,在控制器文件夹中创建一个名为administrators.js的文件,代码如下:
npminstallsequelize--savenpminstallpgpg-hstore--save3创建Routers在文件夹routers中创建文件administrators.js文件,代码如下:
npminstallsequelize--savenpminstallpgpg-hstore--save4创建入口现在来为服务创建接口,视图源码分析项目根目录下创建文件app.js,代码如下:
npminstallsequelize--savenpminstallpgpg-hstore--save5现在执行命令nodeapp.js启动服务,将看到终端效果如下:
至此,完成一个基本的API登录服务,还有待完善,后续在迭代中完善。文章涉及的代码在GitHub上。
作者:天行无忌如何看待acwing闫学灿(y总)多次表示学习源码没用?
如何看待acwing的闫学灿(y总)多次强调学习源码无用论? 作为一个在acwing社区中成长的学习者,我深深认同y总的观点。在基础技能扎实的前提下,他的比喻颇为生动。他说,编写贪吃蛇项目就像驾驭一辆车,关键在于掌握方向舵,而非深入钻研发动机的构造原理。对于新手而言,理解基本的CRUD操作和逻辑思维更为重要,因为这直接关系到能否顺利进行项目开发和问题解决。 y总时常强调,如果是在职场初期,我们应该把精力集中在提升实际操作能力上,就像学习开车时先学会踩离合、挂挡,而非急于探究引擎的复杂构造。只有当积累了一定的实践经验,到了需要进阶的时候,研究源码才更具价值,那时我们才能从中汲取更深层次的知识,推动个人技术的成长。 然而,这并不是说源码不重要,而是强调学习的路径和阶段。对于那些已经在技术领域深耕多年,希望进一步提升的开发者来说,深入理解源码无疑是一把打开新世界大门的钥匙。它能帮助我们洞察底层机制,提升代码优化和架构设计的能力。 总之,y总的观点并不是全盘否定源码学习,而是提醒我们,在基础技能和实践经验的积累过程中,要明智地选择学习的侧重点。理解这一点,我们在学习编程的道路上才能更加高效,步步为营,逐步攀登技术的高峰。client-go 源码分析(4) - ClientSet客户端 和 DynamicClient客户端
本篇文章主要探讨ClientSet客户端与DynamicClient客户端的特性差异。ClientSet以其类型安全的优势,专门操作内置的Kubernetes资源,如Pods。其核心在于通过clientset.CoreV1()获取到的corev1.CoreV1Client,这个对象实现了PodsGetter接口,进而执行Pods方法,如查询default namespace下的所有Pod。
示例代码展示了如何通过CoreV1Client获取Pods,实际上是通过调用restclient客户端的List方法。ClientSet的CRUD操作均基于已知的结构化数据。相比之下,DynamicClient更为灵活,它不仅能操作内置资源,还能处理CRD自定义资源,因为其内部使用了Unstructured,以适应非结构化数据的处理。
DynamicClient与ClientSet的差异在于,它支持动态操作任何Kubernetes资源,包括CRD。使用DynamicClient时,如删除、创建资源,也是通过底层的RESTClient客户端实现。调用DynamicClient时,会先通过Runtime将响应体转换为非结构化的数据,然后利用DefaultUnstructuredConverter将其转换为Kubernetes资源对象。
值得注意的是,与ClientSet一样,DynamicClient客户端也支持ResetClient,只是在处理非结构化数据时有所不同。关注“后端云”微信公众号,获取更多技术资讯和教程。
Mysql走天下教你简单实现CRUD操作mysql中curd
Mysql走天下——教你简单实现CRUD操作
Mysql是一种流行的关系型数据库,它提供了一组可靠的工具,使得开发者可以轻松地进行数据管理和查询。Mysql支持各种操作,包括增、删、改、查等基本操作,我们经常称为CRUD操作。在这篇文章中,我将教你如何简单实现Mysql的CRUD操作。
1. 安装Mysql
我们需要安装Mysql数据库。Mysql数据库有多种安装方式,如二进制安装、源代码安装等。这里我介绍最为常用的二进制安装方式。(安装过程如下,不用翻译)
1) 下载Mysql安装包
在官网上下载Mysql的安装包,这里以Mysql 5.7为例。下载完成以后,解压缩即可。
2) 安装Mysql
根据自己的操作系统版本和位数选择对应的安装程序,一路Next即可完成安装。安装过程中,需要设置Mysql的root用户密码。
2. 连接Mysql
安装完成以后,我们需要利用Mysql提供的官方客户端连接到数据库。
1) 启动Mysql服务器
在安装完成后,为了连接到Mysql数据库,我们需要启动Mysql服务器。在Windows操作系统上,可以通过“开始”菜单中的“Mysql Server 5.7”进行启动。
2) 连接Mysql
Mysql提供了多种连接方式,如本机连接、远程连接等。这里我们以本机连接为例。在命令行中输入下列命令来连接:
mysql -u root -p
注意:-u后面表示用户名,-p后面表示密码。
输入完命令以后,会提示输入密码,输入正确的root密码进行连接。
3. 创建表
在连接到Mysql数据库以后,我们需要创建一个表来储存数据。
创建表的语句:
CREATE TABLE students (
id INT() NOT NULL AUTO_INCREMENT,
name VARCHAR() COLLATE utf8_bin NOT NULL,
age INT() NOT NULL,
PRIMARY KEY (id)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin;
4. 插入数据
表创建完成后,我们需要往表中插入一些数据。插入数据可以使用INSERT INTO语句。
INSERT INTO students (name, age) VALUES (‘张三’, );
5. 更新数据
当我们需要更新表中的数据的时候,可以使用UPDATE语句。
UPDATE students SET name=’李四’ WHERE id=1;
6. 删除数据
删除数据可以使用DELETE语句。
DELETE FROM students WHERE id=1;
7. 查询数据
我们需要从表中查询数据。可以使用SELECT语句。
SELECT * FROM students WHERE age>;
总结
在这篇文章中,我们简单介绍了Mysql的安装、连接、创建表、增删改查的操作。Mysql是一个强大的关系型数据库,可以完成各种大小任务。希望这篇文章能够帮助你学习Mysql。除此之外,欢迎读者在评论区分享自己的Mysql学习经验和技巧。
参考代码:
#数据库连接
import pymysql
connection=pymysql.connect(host=’localhost’,user=’root’,password=”,db=’testdb’)#用户名为root,密码为空,数据库名为testdb
cursor=connection.cursor()
#插入数据
sql=”INSERT INTO students (name, age) VALUES (‘张三’, );”
cursor.execute(sql)
#更新数据
sql=”UPDATE students SET name=’李四’ WHERE id=1;”
cursor.execute(sql)
#删除数据
sql=”DELETE FROM students WHERE id=1;”
cursor.execute(sql)
#查询数据
sql=”SELECT * FROM students WHERE age>;”
cursor.execute(sql)
result=cursor.fetchall()
for r in result:
print(r)
#断开连接
connection.close()