首页 > 数据库 >openGauss账本数据库,你不知道的那些事儿

openGauss账本数据库,你不知道的那些事儿

时间:2024-03-14 11:44:33浏览次数:27  
标签:hash name 数据库 usertable hist omm rel openGauss 账本

openGauss 账本数据库,你不知道的那些事儿
摘要
本文将通过对比官方文档关于“设置账本数据库”中的几个章节,结合源码来说说文档中操作步骤背后的原理。

账本数据库概述
你知道的那些事儿
官方文档
账本数据库融合了区块链思想,将用户操作记录至两种历史表中:用户历史表和全局区块表。当用户创建防篡改用户表时,系统将自动为该表添加一个 hash 列来保存每行数据的 hash 摘要信息,同时在 blockchain 模式下会创建一张用户历史表来记录对应用户表中每条数据的变更行为;而用户对防篡改用户表的一次修改行为将记录至全局区块表中。由于历史表具有只可追加不可修改的特点,因此历史表记录串联起来便形成了用户对防篡改用户表的修改历史。

你不知道的那些事儿
操作步骤
1.创建防篡改模式。
openGauss=# CREATE SCHEMA ledgernsp WITH BLOCKCHAIN;
首先在这个 SQL 中我们可以看到WITH BLOCKCHAIN ,这里说明创建出来的 SCHEMA 与普通的 SCHEMA 不同,但就行不同在哪里我们后面会提到。

从语法解析看,增加了对 BLOCKCHAIN 的处理,标记了是否为账本模式。
CreateSchema ::= CREATE SCHEMA schema_name
[ AUTHORIZATION user_name ] [WITH BLOCKCHAIN] [ schema_element [ ... ] ];
CreateSchemaStmt 结构中增加了 bool 类型字段 hasBlockChain
typedef struct CreateSchemaStmt {
NodeTag type;
char schemaname; / the name of the schema to create */
char authid; / the owner of the created schema /
bool hasBlockChain; /
whether this schema has blockchain */
List schemaElts; / schema components (list of parsenodes) /
TempType temptype; /
if the schema is temp table's schema */
List uuids; / the list of uuid(only create sequence or table with serial type need) */
} CreateSchemaStmt;
你不知道的限制
账本数据库对于 ALTER SCHEMA 的几个限制

1、dbe_perf 和 snapshot 两个模式不能 ALTER 为 blockchain 模式。
if (withBlockchain && ((strncmp(nspName, "dbe_perf", STR_SCHEMA_NAME_LENGTH) == 0) ||
(strncmp(nspName, "snapshot", STR_SNAPSHOT_LENGTH) == 0))) {
ereport(ERROR, (errcode(ERRCODE_OPERATE_FAILED),
errmsg("The schema '%s' doesn't allow to alter to blockchain schema", nspName)));
}
2、系统模式不能 ALTER 为 blockchain 模式。
if (withBlockchain && !g_instance.attr.attr_common.allowSystemTableMods &&
!u_sess->attr.attr_common.IsInplaceUpgrade && IsReservedName(nspName))
ereport(ERROR,
(errcode(ERRCODE_RESERVED_NAME),
errmsg("The system schema "%s" doesn't allow to alter to blockchain schema", nspName)));
3、包含了表的 SCHEMA 不能 ALTER 为 blockchain 模式。
/*
* If the any table exists in the schema, do not change to ledger schema.
*/
StringInfo existTbl = TableExistInSchema(HeapTupleGetOid(tup), TABLE_TYPE_ANY);
if (existTbl->len != 0) {
if (withBlockchain) {
ereport(ERROR,
(errcode(ERRCODE_RESERVED_NAME),
errmsg("It is not supported to change "%s" to blockchain schema which includes tables.",
nspName)));
} else {
ereport(ERROR,
(errcode(ERRCODE_RESERVED_NAME),
errmsg("It is not supported to change "%s" to normal schema which includes tables.",
nspName)));
}

}

查看模式
2.在防篡改模式下创建防篡改用户表。
openGauss=# CREATE TABLE ledgernsp.usertable(id int, name text);
你不知道的限制
创建账本表的同时会自动创建一个“历史表”和“历史表的索引”。
在建表时CreateCommand会调用AlterCreateChainTables,如果是账本表再去调用create_hist_relation来创建历史表

CreateCommand -> AlterCreateChainTables -> create_hist_relation

/*

  • AlterCreateChainTables
  • If it is a ledger usertable, that should invoking this function.
  • then create a history table.
    */
    void AlterCreateChainTables(Oid relOid, Datum reloptions, CreateStmt *mainTblStmt)
    {
    Relation rel = NULL;
rel = heap_open(relOid, AccessExclusiveLock);

/* Ledger user table only support for the regular relation. */
if (!rel->rd_isblockchain) {
    heap_close(rel, NoLock);
    return;
}

create_hist_relation(rel, reloptions, mainTblStmt);
heap_close(rel, NoLock);

}
历史表命名规则,参见函数 get_hist_name
bool get_hist_name(Oid relid, const char *rel_name, char *hist_name, Oid nsp_oid, const char nsp_name)
{
errno_t rc;
if (!OidIsValid(relid) || rel_name == NULL) {
return false;
}
nsp_oid = OidIsValid(nsp_oid) ? nsp_oid : get_rel_namespace(relid);
nsp_name = (nsp_name == NULL) ? get_namespace_name(nsp_oid) : nsp_name;
int part_hist_name_len = strlen(rel_name) + strlen(nsp_name) + 1;
if (part_hist_name_len + strlen("hist") >= NAMEDATALEN) {
rc = snprintf_s(hist_name, NAMEDATALEN, NAMEDATALEN - 1, "%d
%d_hist", nsp_oid, relid);
securec_check_ss(rc, "", "");
} else {
rc = snprintf_s(hist_name, NAMEDATALEN, NAMEDATALEN - 1, "%s_%s_hist", nsp_name, rel_name);
securec_check_ss(rc, "", "");
}
return true;
}
表名最大长度 #define NAMEDATALEN 64
如果没有超过长度限制:schema_table_hist
如果超过长度限制:schema(oid)_talbe(oid)_hist,因为 oid 是 unsigned int 类型最大值为 4294967295 为 10 位,所以这种命名规则的最大长度为 10+1+10+1+4+\0=27,因此永远不会超过最大长度 64。
omm=# create schema aaaaaaaaaaaaaaaaaaaaaaaaaaaaaa with blockchain;
CREATE SCHEMA
omm=# create table aaaaaaaaaaaaaaaaaaaaaaaaaaaaaa.bbbbbbbbbbbbbbbbbbbbbbbbbbbbbb(id int);
CREATE TABLE
历史表索引命名规则,参见函数 get_hist_name
/
now create index for this new history table */
char hist_index_name[NAMEDATALEN];
rc = snprintf_s(hist_index_name, NAMEDATALEN, NAMEDATALEN - 1, "gs_hist_%u_index", relid);
命名规则:gshist\$(账本表 oid)_index。
3、修改防篡改用户表数据
对防篡改用户表执行 INSERT/UPDATE/DELETE。

openGauss=# INSERT INTO ledgernsp.usertable VALUES(1, 'alex'), (2, 'bob'), (3, 'peter');
INSERT 0 3
openGauss=# SELECT *, hash FROM ledgernsp.usertable ORDER BY id;
id | name | hash
----+-------+------------------
1 | alex | 1f2e543c580cb8c5
2 | bob | 8fcd74a8a6a4b484
3 | peter | f51b4b1b12d0354b
(3 rows)

openGauss=# UPDATE ledgernsp.usertable SET name = 'bob2' WHERE id = 2;
UPDATE 1
openGauss=# SELECT *, hash FROM ledgernsp.usertable ORDER BY id;
id | name | hash
----+-------+------------------
1 | alex | 1f2e543c580cb8c5
2 | bob2 | 437761affbb7c605
3 | peter | f51b4b1b12d0354b
(3 rows)

openGauss=# DELETE FROM ledgernsp.usertable WHERE id = 3;
DELETE 1
openGauss=# SELECT *, hash FROM ledgernsp.usertable ORDER BY id;
id | name | hash
----+------+------------------
1 | alex | 1f2e543c580cb8c5
2 | bob2 | 437761affbb7c605
(2 rows)
查看账本历史操作记录
你知道的那些事儿
官方文档
前提条件

系统中需要有审计管理员或者具有审计管理员权限的角色。
数据库正常运行,并且对防篡改数据库执行了一系列增、删、改等操作,保证在查询时段内有账本操作记录结果产生。
你不知道的那些事儿
基本操作
1、查询全局区块表记录。
omm=# SELECT * FROM gs_global_chain;
blocknum | dbname | username | starttime | relid | relnsp | relname | relhash | globalhash |
txcommand
----------+--------+----------+-------------------------------+-------+-----------+-----------+------------------+----------------------------------+----------------

    1 | omm    | omm      | 2022-09-17 13:59:37.84824+00  | 16404 | ledgernsp | usertable | a41714001181a294 | 83927d11ba1fd678e8f4b0723a9cd5f2 | INSERT INTO led

gernsp.usertable VALUES(1, 'alex'), (2, 'bob'), (3, 'peter');
2 | omm | omm | 2022-09-17 13:59:51.723068+00 | 16404 | ledgernsp | usertable | b3a9ed0755131181 | b5ee73b6c20c817230182f6373c78e20 | UPDATE ledgerns
p.usertable SET name = 'bob2' WHERE id = 2;
3 | omm | omm | 2022-09-17 13:59:58.159596+00 | 16404 | ledgernsp | usertable | 0ae4b4e4ed2fcab5 | 0cc9938cf7f1ed7f7f1a03c29954380a | DELETE FROM led
gernsp.usertable WHERE id = 3;
(3 rows)
注册钩子,在对账本做修改操作的时候注册的钩子函数ledger_ExecutorEnd被回调。
/*

  • ledger_hook_init -- install of gchain block record hook.
    */
    void ledger_hook_init(void)
    {
    t_thrd.security_ledger_cxt.prev_ExecutorEnd = (void *)ExecutorEnd_hook;
    ExecutorEnd_hook = ledger_ExecutorEnd;
    }
    生成 globalhash 规则
    全局区块表记录主要是生成 globalhash.

调用过程:

ledger_ExecutorEnd —> ledger_gchain_append —> set_gchain_comb_string

—> get_next_g_blocknum

—> gen_global_hash

set_gchain_comb_string,是一组字符串拼接成的:rel_name + nsp_name + query_string + rel_hash
get_next_g_blocknum,用全局变量 g_blocknum 保存
gen_global_hash,是的 set_gchain_comb_string 拼出来的串+上一条的 hash 值拼串然后再去 hash——区块链的基本原理
bool gen_global_hash(hash32_t *hash_buffer, const char *info_string, bool exist, const hash32_t *prev_hash)
{
errno_t rc = EOK;
int comb_strlen;
char comb_string = NULL;
/

* Previous block not exists means current insertion block is genesis,
* then we use global systable as origin combine string for globalhash
* generation. If previous block exists, we will use previous global
* hash as combine string to calculate globalhash.
/
if (!exist) {
/
generate genesis block globalhash */
comb_strlen = strlen(GCHAIN_NAME) + strlen(info_string) + 1;
comb_string = (char )palloc0(comb_strlen);
rc = snprintf_s(comb_string, comb_strlen, comb_strlen - 1, "%s%s", GCHAIN_NAME, info_string);
securec_check_ss(rc, "", "");
} else {
/
use previous globalhash and current block info to calculate globalhash. */
char *pre_hash_str = DatumGetCString(DirectFunctionCall1(hash32out, HASH32GetDatum(prev_hash)));
comb_strlen = strlen(pre_hash_str) + strlen(info_string) + 1;
comb_string = (char *)palloc0(comb_strlen);
rc = snprintf_s(comb_string, comb_strlen, comb_strlen - 1, "%s%s", info_string, pre_hash_str);
securec_check_ss(rc, "", "");
pfree_ext(pre_hash_str);
}

if (!pg_md5_binary(comb_string, comb_strlen - 1, hash_buffer->data)) {
    pfree(comb_string);
    ereport(ERROR, (errcode(ERRCODE_OUT_OF_MEMORY), errmsg("Failed to generate globalhash, out of memory")));
    return false;
}
pfree(comb_string);
return true;

}
在src/gausskernel/runtime/executor/nodeModifyTable.cpp中更新_hist 表的 hash 值。
通过 set_user_tuple_hash 得到账本表 hash 列的值。
/*

  • set_user_tuple_hash -- calculate and fill the hash attribute of user table's tuple.

  • tup: row data of user table

  • rel: user table

  • hash_exists: whether tuple comes with tuplehash.

  • Note: if hash_exists is true, we should recompute

  • tuple hash and compare with tuplehash of itself.
    */
    HeapTuple set_user_tuple_hash(HeapTuple tup, Relation rel, bool hash_exists)
    {
    uint64 row_hash = gen_user_tuple_hash(rel, tup);
    int hash_attrno = user_hash_attrno(rel->rd_att);
    if (hash_exists) {
    bool is_null;
    Datum hash = heap_getattr(tup, hash_attrno + 1, rel->rd_att, &is_null);
    if (is_null || row_hash != DatumGetUInt64(hash)) {
    ereport(ERROR, (errcode(ERRCODE_OPERATE_INVALID_PARAM), errmsg("Invalid tuple hash.")));
    }
    return tup;
    }
    Datum values = NULL;
    bool nulls = NULL;
    bool replaces = NULL;
    /
    Build modified tuple /
    int2 nattrs = RelationGetNumberOfAttributes(rel);
    values = (Datum
    )palloc0(nattrs * sizeof(Datum));
    nulls = (bool
    )palloc0(nattrs * sizeof(bool));
    replaces = (bool
    )palloc0(nattrs * sizeof(bool));
    values[hash_attrno] = UInt64GetDatum(row_hash);
    replaces[hash_attrno] = true;
    HeapTuple newtup = heap_modify_tuple(tup, RelationGetDescr(rel), values, nulls, replaces);

    pfree_ext(values);
    pfree_ext(nulls);
    pfree_ext(replaces);
    return newtup;
    }
    校验账本数据一致性
    你知道的那些事儿
    官方文档
    数据库正常运行,并且对防篡改数据库执行了一系列增、删、改等操作,保证在查询时段内有账本操作记录结果产生。

你不知道的那些事儿
基本操作
1、校验防篡改用户表 ledgernsp.usertable 与其对应的历史表是否一致。
omm=# SELECT pg_catalog.ledger_hist_check('ledgernsp', 'usertable');
ledger_hist_check

t
(1 row)
校验用户权限 Only super user or audit admin have access right to blockchain nsp
/* Only super user or audit admin have access right to blockchain nsp */
if (nsp_oid == PG_BLOCKCHAIN_NAMESPACE) {
return gs_blockchain_aclmask(roleid, mask);
}
校验历史表 hash 值
is_hist_hash_identity —> get_usertable_hash_sum

—> get_histtable_hash_sum

/*

  • is_hist_hash_identity -- check whether user table hash and history table hash are equal

  • relid: user table oid

  • res_hash: hash sum of history table
    */
    bool is_hist_hash_identity(Oid relid, uint64 *res_hash)
    {
    uint64 user_hash_sum;
    uint64 hist_hash_sum;
    char hist_name[NAMEDATALEN];
    char *rel_name = get_rel_name(relid);
    if (!get_hist_name(relid, rel_name, hist_name)) {
    ereport(ERROR, (errcode(ERRCODE_UNDEFINED_OBJECT), errmsg("get hist table name failed.")));
    }
    Oid histoid = get_relname_relid(hist_name, PG_BLOCKCHAIN_NAMESPACE);
    if (!OidIsValid(histoid)) {
    ereport(ERROR, (errcode(ERRCODE_UNDEFINED_OBJECT), errmsg("could not find hist table of "%s".", rel_name)));
    }

    user_hash_sum = get_usertable_hash_sum(relid);
    hist_hash_sum = get_histtable_hash_sum(histoid);

    *res_hash = hist_hash_sum;
    return user_hash_sum == hist_hash_sum;
    }
    2、查询防篡改用户表 ledgernsp.usertable 与其对应的历史表以及全局区块表中关于该表的记录是否一致。
    omm=# SELECT pg_catalog.ledger_gchain_check('ledgernsp', 'usertable');
    ledger_gchain_check


t
(1 row)
校验是否为账本表ledger_usertable_check
校验用户权限has_ledger_consistent_privilege
校验历史表 hash 值is_hist_hash_identity
计算/校验全局表 hash get_gchain_relhash_sum
/*

  • get_gchain_relhash_sum -- calculate relhash from gs_global_chain

  • relid: user table oid
    */
    static uint64 get_gchain_relhash_sum(Oid relid)
    {
    uint64 relhash = 0;
    HeapTuple tuple = NULL;

    /* scan the gs_global_chain catalog by relid */
    Relation gchain_rel = heap_open(GsGlobalChainRelationId, AccessShareLock);
    Form_gs_global_chain rdata = NULL;
    TableScanDesc scan = heap_beginscan(gchain_rel, SnapshotNow, 0, NULL);
    while ((tuple = heap_getnext(scan, ForwardScanDirection)) != NULL) {
    rdata = (Form_gs_global_chain)GETSTRUCT(tuple);
    if (rdata == NULL || rdata->relid != relid) {
    continue;
    }
    relhash += rdata->relhash;
    }
    heap_endscan(scan);
    heap_close(gchain_rel, AccessShareLock);
    return relhash;
    }
    归档账本数据库
    你知道的那些事儿
    官方文档
    前提条件:

系统中需要有审计管理员或者具有审计管理员权限的角色。
数据库正常运行,并且对防篡改数据库执行了一系列增、删、改等操作,保证在查询时段内有账本操作记录结果产生。
数据库已经正确配置审计文件的存储路径 audit_directory。
你不知道的那些事儿
基本操作
1、对指定用户历史表进行归档操作。

omm=# SELECT pg_catalog.ledger_hist_archive('ledgernsp', 'usertable');
ledger_hist_archive

t
(1 row)

omm=# SELECT * FROM blockchain.ledgernsp_usertable_hist;
rec_num | hash_ins | hash_del | pre_hash
---------+------------------+------------------+----------------------------------
4 | e78e75b00d396899 | 84e8bfc3b974e9cf | 6475a497b7a272a92bab012d7f3d615b
(1 row)
主要步骤如下:

Copy user history table.
Do unify and truncate.
sum all hash_ins and hash_del for unification.
Do real truncate.heap_truncate_one_rel
Do insertion for unified row.simple_heap_insert
Flush history hash table cache.
2、执行全局区块表导出操作
omm=# SELECT * FROM gs_global_chain;
blocknum | dbname | username | starttime | relid | relnsp | relname | relhash | globalhash |
txcommand
----------+--------+----------+-------------------------------+-------+-----------+-----------+------------------+----------------------------------+----------------

    1 | omm    | omm      | 2022-09-17 13:59:37.84824+00  | 16404 | ledgernsp | usertable | a41714001181a294 | 83927d11ba1fd678e8f4b0723a9cd5f2 | INSERT INTO led

gernsp.usertable VALUES(1, 'alex'), (2, 'bob'), (3, 'peter');
2 | omm | omm | 2022-09-17 13:59:51.723068+00 | 16404 | ledgernsp | usertable | b3a9ed0755131181 | b5ee73b6c20c817230182f6373c78e20 | UPDATE ledgerns
p.usertable SET name = 'bob2' WHERE id = 2;
3 | omm | omm | 2022-09-17 13:59:58.159596+00 | 16404 | ledgernsp | usertable | 0ae4b4e4ed2fcab5 | 0cc9938cf7f1ed7f7f1a03c29954380a | DELETE FROM led
gernsp.usertable WHERE id = 3;
(3 rows)

omm=# SELECT pg_catalog.ledger_gchain_archive();
ledger_gchain_archive

t
(1 row)

omm=# SELECT * FROM gs_global_chain;
blocknum | dbname | username | starttime | relid | relnsp | relname | relhash | globalhash | txcommand
----------+--------+----------+------------------------------+-------+-----------+-----------+------------------+----------------------------------+-----------
2 | omm | omm | 2022-09-17 13:59:37.84824+00 | 16404 | ledgernsp | usertable | 62a5b5ec53c47eca | 7252d09679b0b3836a2e63da17284ad5 | Archived.
(1 row)
gs_global_chain 主要处理流程:

Init and prepare bak dictionary.
Using CopyStmt to copy global chain.
Do unify and truncate.
Using hash table to do unify, each hash_entry refers to one relid informations.
Split gs_global_chain by relid, and accumulate rel_hash to a new record for each rel.
Do rel truncate.
Insert newest record to gchain order by relid.
Flush global_hash cache.
修复账本数据库
你知道的那些事儿
前提条件:

系统中需要有审计管理员或者具有审计管理员权限的角色。
数据库正常运行,并且对防篡改数据库执行了一系列增、删、改等操作,保证在查询时段内有账本操作记录结果产生。
你不知道的那些事儿
基本操作
1、执行历史表修复操作
omm=# select * from blockchain.ledgernsp_usertable_hist;
rec_num | hash_ins | hash_del | pre_hash
---------+------------------+------------------+----------------------------------
4 | e78e75b00d396899 | 84e8bfc3b974e9cf | 6475a497b7a272a92bab012d7f3d615b
(1 row)

omm=# SELECT pg_catalog.ledger_hist_repair('ledgernsp', 'usertable');
ledger_hist_repair

0000000000000000
(1 row)
2、执行全局区块表修复操作
omm=# select * from gs_global_chain ;
blocknum | dbname | username | starttime | relid | relnsp | relname | relhash | globalhash | txcommand
----------+--------+----------+------------------------------+-------+-----------+-----------+------------------+----------------------------------+-----------
2 | omm | omm | 2022-09-17 13:59:37.84824+00 | 16404 | ledgernsp | usertable | 62a5b5ec53c47eca | 7252d09679b0b3836a2e63da17284ad5 | Archived.
(1 row)

omm=# SELECT pg_catalog.ledger_gchain_repair('ledgernsp', 'usertable');
ledger_gchain_repair

62a5b5ec53c47eca
(1 row)
首先判断用户权限,之后通过 get_gchain_relhash_sum 函数计算 relhash 字段

/*

  • get_gchain_relhash_sum -- calculate relhash from gs_global_chain

  • relid: user table oid
    */
    static uint64 get_gchain_relhash_sum(Oid relid)
    {
    uint64 relhash = 0;
    HeapTuple tuple = NULL;

    /* scan the gs_global_chain catalog by relid */
    Relation gchain_rel = heap_open(GsGlobalChainRelationId, AccessShareLock);
    Form_gs_global_chain rdata = NULL;
    TableScanDesc scan = heap_beginscan(gchain_rel, SnapshotNow, 0, NULL);
    while ((tuple = heap_getnext(scan, ForwardScanDirection)) != NULL) {
    rdata = (Form_gs_global_chain)GETSTRUCT(tuple);
    if (rdata == NULL || rdata->relid != relid) {
    continue;
    }
    relhash += rdata->relhash;
    }
    heap_endscan(scan);
    heap_close(gchain_rel, AccessShareLock);
    return relhash;
    }
    主要是计算并修复gs_global_chain中的relhash字段。

总结
账本数据库其实并不像我们想象的那么复制,实际上就是利用了区块链的最基本的原理,即当前记录的特征值 + 上一条记录特征值的 hash 值,再进行 hash。下一条与上一条记录具有数据关联性,形成“链”的结构,如果篡改了其中的数据,则会导致“链”断开,导致不能与后面数据记录形成 hash 关联。_hist 表记录了用户表每一步数据变化的过程,gs_global_chain 表记录了所有防篡改模式下对用户表的操作记录。用户表结合_hist 和 global 表就能完整记录和校验。

标签:hash,name,数据库,usertable,hist,omm,rel,openGauss,账本
From: https://www.cnblogs.com/renxyz/p/18072517

相关文章

  • 数据库设计规范-Oracle
    参照以下原则进行数据库设计:1.方便业务功能实现、业务功能扩展。2.方便设计开发、增强系统的稳定性和可维护性。3.保证数据完整性和准确性。4.提高数据存储效率,在满足业务需求的前提下,使时间开销和空间开销达到优化平衡。项目信息化建设全文档:软件开发全套资料_数字中台......
  • Database Connection Pool 数据库连接池-01-概览及简单手写实现
    拓展阅读第一节从零开始手写mybatis(一)MVP版本。第二节从零开始手写mybatis(二)mybatisinterceptor插件机制详解第三节从零开始手写mybatis(三)jdbcpool从零实现数据库连接池第四节从零开始手写mybatis(四)-mybatis事务管理机制详解连接池的作用资源重用由于数据库......
  • 2024-03-11-Nodejs(3-数据库与身份验证)
    3.数据库与身份验证3.1数据库基本概念数据库是用来组织、存储和管理数据的仓库;传统数据库中,数据结构分为数据库(database)、数据表(table)、数据行(tow)、字段(field)四大部分。3.2配置mysql模块安装mysql模块npminstallmysql建立连接constmysql=require('mysql')......
  • oracle数据库 ,当某个字段为空,用 != null 查询不出来
     oracel中某个字段有varchar类型,有三个属性值:null,service,center,当向查询不要center的值搂数据时,SELECTCOUNT(1)FROMT_MOCK_TASKWHERE RUN_ON!='CENTER'查出来的不包括null的,只能查出来等于service的;null的用 run_on='',也查不出来,需要用isnull才能查......
  • MongoDB数据库之主从复制配置实战【转】
    一、MongoDB介绍 1.1MongoDB简介MongoDB是一个开源的文档数据库,使用JSON格式存储和操作数据,具有高度灵活性和可扩展性。MongoDB的数据模型是面向文档的,这意味着它可以存储各种类型的数据,如数组、嵌套文档和二进制数据。MongoDB是一种NoSQL数据库,不需要使用传统的表格结构。M......
  • Oracle EMCC数据库集中管理平台安装配置与使用
    一、风哥OracleEMCC数据库集中管理平台培训本课程由风哥发布的OracleEMCC课程,学完本课程可以掌握OracleEMCC基础知识,OracleEMCC安装要求,ORACLEEMCCforLinux环境准备,ORACLEEMCCOMR资料数据库安装,OracleEMCC管理平台安装,OracleEMCC启动停止与基本命令,OracleEMCC管理主......
  • InnoDB是事务型数据库的首选引擎
    支持事务安全表(ACID),支持行锁定和外键;MySQL事务的ACID特性是确保数据准确性和可靠性的基本原则,包括**原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)和持久性(Durability)**。具体如下:1.**原子性(Atomicity)**:原子性指的是事务作为最小的执行单位,其包含的操作要么全部......
  • 数据库---存储过程
    一、存储过程的概念1、存储过程(StoredProcedure)是一组为了完成特定功能的SQL语句集。经编译后存储在数据库中,是"数据库函数"。2、存储过程是数据库中的一个重要对象,用户通过指定存储过程的名字并给出参数(如果该存储过程带有参数)来执行它。3、存储过程是由流控制和SQL语句书写......
  • 如何在数据库中存储小数:FLOAT、DECIMAL还是BIGINT?
    前言这里还是用前面的例子:在线机票订票系统的数据表设计。此时已经完成了大部分字段的设计,可能如下:CREATETABLEflights(flight_idINTAUTO_INCREMENTPRIMARYKEY,flight_numberVARCHAR(10),departure_airport_codeVARCHAR(3),arrival......
  • 关系型数据库原理
    一、关系型数据库1.关系型数据库(RDBMS)建立在关系模型基础上,由多张相互连接的二维表组成的数据库。也就是将数据以表的形式存储并且这些表存在相互联系。表是由行和列组成。2.关系型数据库的设计范式设计关系数据库时,遵从不同的规范要求,设计出合理的关系型数据库,这些不......