MySql编程技巧-利用BIT_COUNT/BIT_OR统计登陆天数文章摘要mysql怎么编程:
1、BIT_COUNT/BIT_ORmysql怎么编程的用法
2、使用位组函数来计算每个月中用户登陆的天数 。
3、举一反三:统计每个年度用户登陆天数 。
示例表中含有代表用户访问网页的年-月-日值 。
将返回:
该查询计算mysql怎么编程了在表中按年/月组合的不同天数mysql怎么编程,BIT_OR(或)可以自动去除重复的询问 。
vs中怎样操作mysql数据库方法/步骤
1
新建一个工程名叫mysql,编程环境选择c#,然后选择windows窗体应用程序,新建一个窗体用于显示查询到sql数据库mysql怎么编程的数据集
2
从工具箱向form1窗体上拖一个按钮和datagridview控件mysql怎么编程 , 按钮是触发连接数据库获取数据集,按钮的名称为显示,datagridview控件是用于显示数据集内容
3
单击解决方案资源管理器中的引用文件夹然后右键选择添加引用,选择浏览后打开mysql.data.dll,这是c#连接mysql数据库的动态库,里面封装mysql怎么编程了很多常用的操作数据库的方法
4
在解决方案资源管理器中的form1.cs的代码中加入using MySql.Data.MySqlClient;这就是代码中的实际引用mysql.data.dll中的内容,有了这个c#就能很方便地操作sql数据库
5
在按钮的单击事件中添加如下代码
string str = "Server=127.0.0.1;User ID=root;Password=123456;Database=test;CharSet=gbk;";
MySqlConnection con = new MySqlConnection(str);//实例化链接
con.Open();//开启连接
string strcmd = "select * from user";
MySqlCommand cmd = new MySqlCommand(strcmd, con);
MySqlDataAdapter ada = new MySqlDataAdapter(cmd);
DataSet ds = new DataSet();
ada.Fill(ds);//查询结果填充数据集
dataGridView1.DataSource = ds.Tables[0];
con.Close();//关闭连接
6
使用navicat软件在数据库test中新建表user,然后新建两个字段username和password(图中的栏位),navicat软件是mysql的图形化界面工具,负责新建表以及备份等数据库操作,直观地通过界面来操作
7
数据库建好后就可以执行工程了,单击显示按钮执行结果如下,出现username和password说明数据库连接成功,由于没有添加数据所以下面为空
mysql的C API编程MYSQL_OPT_READ_TIMEOUT 是 MySQL c api 客户端中用来设置读取超时时间mysql怎么编程的参数 。在 MySQL 的官方文档中mysql怎么编程,该参数的描述是这样的:
MYSQL_OPT_READ_TIMEOUT (argument type: unsigned int *)The timeout in seconds for each attempt to read from the server. There are retries if necessary, so the total effective timeout value is three times the option value. You can set the value so that a lost connection can be detected earlier than the TCP/IPClose_Wait_Timeout value of 10 minutes.
也就是说在需要的时候mysql怎么编程 , 实际的超时时间会是设定值的 3 倍 。但是实际测试后发现实际的超时时间和设置的超时时间一致 。
而具体什么时候发生三倍超时,在文档中没有找到 。所以对 MySQL 5.7.20 的源码进行了一些分析 。
使用 GDB 调试代码找了实际与 mysql server 通信的代码,如下:
其中 vio_read() 函数中,使用 recv 和 poll 来读取报文和做读取超时 。net_should_retry() 函数只有在发生 EINTR 时才会返回 true 。从这段代码来看是符合测试结果的,并没有对读取进行三次重试 。只有在读取操作被系统中断打断时才会重试,但是这个重试并没有次数限制 。
从上面代码的分析可以看出 , 代码的逻辑和文档的描述不符 。于是在一顿搜索后,找到了一个 MySQL 的 BUG(Bug #31163) 。该 BUG 报告了在 MySQL 5.0 中 , MySQL c api 读取的实际超时时间是设置的三倍 , 与现有文档描述相符 。于是对 MySQL 5.0.96 的代码又进行分析 。
同样使用 GDB 找到了通信部分的代码 。这次找到了重试三次的代码,如下:
这个版本的 MySQL api 的读写超时是直接使用的 setsockopt 设置的 。第一次循环,在 A 点发生了第一次超时(虽然注释写的非阻塞,但是客户端的连接始终是阻塞模式的) 。然后在 B 点将该 socket 设置为阻塞模式 , C 点这里重置 retry 次数 。由于设置了 alarm 第二次以后的循环会直接进入 D 点的这个分支 , 并且判断循环次数 。作为客户端时net-retry_count 始终是 1,所以重试了两次,共计进行了 3 次 vioread 后从 E 点退出函数 。
由上面的分析可知,MySQL 文档对于该参数的描述已经过时,现在的 MYSQL_OPT_READ_TIMEOUT 并不会出现三倍超时的问题 。而 Bug #31163 中的处理结果也是将文档中该参数的描述更新为实际读取超时时间是设定时间的三倍 。也许是 MySQL 的维护者们在后续版本更新时忘记更新文档吧 。
mysql 核心内容-上1、SQL语句执行流程
MySQL大体上可分为Server层和存储引擎层两部分 。
Server层:
连接器:TCP握手后服务器来验证登陆用户身份 , A用户创建连接后,管理员对A用户权限修改mysql怎么编程了也不会影响到已经创建的链接权限 , 必须重新登陆 。
查询缓存:查询后的结果存储位置,MySQL8.0版本以后已经取消,因为查询缓存失效太频繁 , 得不偿失 。
分析器:根据语法规则,判断mysql怎么编程你输入的这个SQL语句是否满足MySQL语法 。
优化器:多种执行策略可实现目标,系统自动选择最优进行执行 。
执行器:判断是否有权限,将最终任务提交到存储引擎 。
存储引擎层
负责数据的存储和提取 。其架构模式是插件式的,支持InnoDB、MyISAM、Memory等多个存储引擎 。现在最常用的存储引擎是InnoDB , 它从MySQL 5.5.5版本开始成为了默认存储引擎(经常用的也是这个) 。
SQL执行顺序
2、BinLog、RedoLog、UndoLog
BinLog
BinLog是记录所有数据库表结构变更(例如create、alter table)以及表数据修改(insert、update、delete)的二进制日志 , 主从数据库同步用到的都是BinLog文件 。BinLog日志文件有三种模式 。
STATEMENT 模式
内容:binlog 记录可能引起数据变更的 sql 语句
优势:该模式下,因为没有记录实际的数据,所以日志量很少 IO 都消耗很低,性能是最优的
劣势:但有些操作并不是确定的 , 比如 uuid() 函数会随机产生唯一标识 , 当依赖 binlog 回放时,该操作生成的数据与原数据必然是不同的,此时可能造成无法预料的后果 。
ROW 模式
内容:在该模式下,binlog 会记录每次操作的源数据与修改后的目标数据,StreamSets就要求该模式 。
优势:可以绝对精准的还原,从而保证了数据的安全与可靠,并且复制和数据恢复过程可以是并发进行的
劣势:缺点在于 binlog 体积会非常大,同时,对于修改记录多、字段长度大的操作来说,记录时性能消耗会很严重 。阅读的时候也需要特殊指令来进行读取数据 。
MIXED 模式
内容:是对上述STATEMENT 跟 ROW 两种模式的混合使用 。
细节:对于绝大部分操作,都是使用 STATEMENT 来进行 binlog 没有记录,只有以下操作使用 ROW 来实现:表的存储引擎为 NDB,使用了uuid() 等不确定函数,使用了 insert delay 语句,使用了临时表
主从同步流程:
1、主节点必须启用二进制日志,记录任何修改了数据库数据的事件 。
2、从节点开启一个线程(I/O Thread)把自己扮演成 mysql 的客户端,通过 mysql 协议 , 请求主节点的二进制日志文件中的事件。
3、主节点启动一个线程(dump Thread),检查自己二进制日志中的事件,跟对方请求的位置对比,如果不带请求位置参数,则主节点就会从第一个日志文件中的第一个事件一个一个发送给从节点 。
4、从节点接收到主节点发送过来的数据把它放置到中继日志(Relay log)文件中 。并记录该次请求到主节点的具体哪一个二进制日志文件内部的哪一个位置(主节点中的二进制文件会有多个) 。
5、从节点启动另外一个线程(sql Thread ),把 Relay log 中的事件读取出来,并在本地再执行一次 。
mysql默认的复制方式是异步的,并且复制的时候是有并行复制能力的 。主库把日志发送给从库后不管了 , 这样会产生一个问题就是假设主库挂了,从库处理失败了,这时候从库升为主库后,日志就丢失了 。由此产生两个概念 。
全同步复制
主库写入binlog后强制同步日志到从库 , 所有的从库都执行完成后才返回给客户端,但是很显然这个方式的话性能会受到严重影响 。
半同步复制
半同步复制的逻辑是这样,从库写入日志成功后返回ACK确认给主库,主库收到至少一个从库的确认就认为写操作完成 。
还可以延伸到由于主从配置不一样、主库大事务、从库压力过大、网络震荡等造成主备延迟,如何避免这个问题?主备切换的时候用可靠性优先原则还是可用性优先原则?如何判断主库Crash了?互为主备的情况下如何避免主备循环复制?被删库跑路了如何正确恢复?( o )… 感觉越来越扯到DBA的活儿上去了 。
RedoLog
可以先通过下面demo理解:
饭点记账可以把账单写在账本上也可以写在粉板上 。有人赊账或者还账的话,一般有两种做法:
1、直接把账本翻出来,把这次赊的账加上去或者扣除掉 。
2、先在粉板上记下这次的账,等打烊以后再把账本翻出来核算 。
生意忙时选后者 , 因为前者太麻烦了 。得在密密麻麻的记录中找到这个人的赊账总额信息,找到之后再拿出算盘计算,最后再将结果写回到账本上 。
同样在MySQL中如果每一次的更新操作都需要写进磁盘,然后磁盘也要找到对应的那条记录,然后再更新 , 整个过程IO成本、查找成本都很高 。而粉板和账本配合的整个过程就是MySQL用到的是Write-Ahead Logging 技术 , 它的关键点就是先写日志,再写磁盘 。此时账本 = BinLog,粉板 = RedoLog 。
1、 记录更新时 , InnoDB引擎就会先把记录写到RedoLog(粉板)里面 , 并更新内存 。同时,InnoDB引擎会在空闲时将这个操作记录更新到磁盘里面 。
2、 如果更新太多RedoLog处理不了的时候,需先将RedoLog部分数据写到磁盘,然后擦除RedoLog部分数据 。RedoLog类似转盘 。
RedoLog有write pos 跟checkpoint
write pos :是当前记录的位置,一边写一边后移,写到第3号文件末尾后就回到0号文件开头 。
check point:是当前要擦除的位置,也是往后推移并且循环的,擦除记录前要把记录更新到数据文件 。
write pos和check point之间的是粉板上还空着的部分,可以用来记录新的操作 。如果write pos追上checkpoint , 表示粉板满了,这时候不能再执行新的更新,得停下来先擦掉一些记录 , 把checkpoint推进一下 。
有了redo log,InnoDB就可以保证即使数据库发生异常重启,之前提交的记录都不会丢失,这个能力称为crash-safe 。redolog两阶段提交:为了让binlog跟redolog两份日志之间的逻辑一致 。提交流程大致如下:
1 prepare阶段 -- 2 写binlog -- 3 commit
当在2之前崩溃时 , 重启恢复后发现没有commit,回滚 。备份恢复:没有binlog。一致
当在3之前崩溃时,重启恢复发现虽没有commit,但满足prepare和binlog完整,所以重启后会自动commit 。备份:有binlog. 一致
binlog跟redolog区别:
redo log是InnoDB引擎特有的mysql怎么编程;binlog是MySQL的Server层实现的 , 所有引擎都可以使用 。
redo log是物理日志,记录的是在某个数据页上做了什么修改mysql怎么编程;binlog是逻辑日志,记录的是这个语句的原始逻辑 , 比如给ID=2这一行的c字段加1 。
redo log是循环写的,空间固定会用完;binlog是可以追加写入的 。追加写是指binlog文件写到一定大小后会切换到下一个,并不会覆盖以前的日志 。
UndoLog
UndoLog 一般是逻辑日志 , 主要分为两种:
insert undo log
代表事务在insert新记录时产生的undo log, 只在事务回滚时需要,并且在事务提交后可以被立即丢弃
update undo log
事务在进行update或delete时产生的undo log; 不仅在事务回滚时需要,在快照读时也需要;所以不能随便删除,只有在快速读或事务回滚不涉及该日志时,对应的日志才会被purge线程统一清除
3、MySQL中的索引
索引的常见模型有哈希表、有序数组和搜索树 。
哈希表:一种以KV存储数据的结构,只适合等值查询,不适合范围查询 。
有序数组:只适用于静态存储引擎,涉及到插入的时候比较麻烦 。可以参考Java中的ArrayList 。
搜索树:按照数据结构中的二叉树来存储数据 , 不过此时是N叉树(B 树) 。广泛应用在存储引擎层中 。
B 树比B树优势在于:
B树非叶子节点存储的只是索引 , 可以存储的更多 。B 树比B树更加矮胖,IO次数更少 。
B树叶子节点前后管理,更加方便范围查询 。同时结果都在叶子节点,查询效率稳定 。
B 树中更有利于对数据扫描 , 可以避免B树的回溯扫描 。
索引的优点:
1、唯一索引可以保证每一行数据的唯一性
2、提高查询速度
3、加速表与表的连接
4、显著的减少查询中分组和排序的时间
5、通过使用索引 , 可以在查询的过程中 , 使用优化隐藏器,提高系统的性能 。
索引的缺点:
1、创建跟维护都需要耗时
2、创建索引时 , 需要对表加锁,在锁表的同时,可能会影响到其他的数据操作
3、 索引需要磁盘的空间进行存储,磁盘占用也很快 。
4、当对表中的数据进行CRUD的时,也会触发索引的维护,而维护索引需要时间,可能会降低数据操作性能
索引设计的原则不应该:
1、索引不是越多越好 。索引太多 , 维护索引需要时间跟空间 。
2、 频繁更新的数据,不宜建索引 。
3、数据量小的表没必要建立索引 。
应该:
1、重复率小的列建议生成索引 。因为重复数据少,索引树查询更有效率 , 等价基数越大越好 。
2、数据具有唯一性,建议生成唯一性索引 。在数据库的层面,保证数据正确性
3、频繁group by、order by的列建议生成索引 。可以大幅提高分组和排序效率
4、经常用于查询条件的字段建议生成索引 。通过索引查询,速度更快
索引失效的场景
1、模糊搜索:左模糊或全模糊都会导致索引失效,比如'%a'和'%a%' 。但是右模糊是可以利用索引的,比如'a%'。
2、隐式类型转换:比如select * from t where name = xxx , name是字符串类型,但是没有加引号,所以是由MySQL隐式转换的,所以会让索引失效 3、当语句中带有or的时候:比如select * from t where name=‘sw’ or age=14
4、不符合联合索引的最左前缀匹配:(A,B,C)的联合索引,你只where了C或B或只有B,C
关于索引的知识点:
主键索引:主键索引的叶子节点存的是整行数据信息 。在InnoDB里,主键索引也被称为聚簇索引(clustered index) 。主键自增是无法保证完全自增的哦,遇到唯一键冲突、事务回滚等都可能导致不连续 。
唯一索引:以唯一列生成的索引 , 该列不允许有重复值,但允许有空值(NULL)
普通索引跟唯一索引查询性能:InnoDB的数据是按数据页为单位来读写的,默认每页16KB,因此这两种索引查询数据性能差别微乎其微 。
change buffer:普通索引用在更新过程的加速 , 更新的字段如果在缓存中,如果是普通索引则直接更新即可 。如果是唯一索引需要将所有数据读入内存来确保不违背唯一性,所以尽量用普通索引 。
非主键索引:非主键索引的叶子节点内容是主键的值 。在InnoDB里 , 非主键索引也被称为二级索引(secondary index)
回表:先通过数据库索引扫描出数据所在的行,再通过行主键id取出索引中未提供的数据,即基于非主键索引的查询需要多扫描一棵索引树 。
覆盖索引:如果一个索引包含(或者说覆盖)所有需要查询的字段的值 , mysql怎么编程我们就称之为覆盖索引 。
联合索引:相对单列索引,组合索引是用多个列组合构建的索引,一次性最多联合16个 。
最左前缀原则:对多个字段同时建立的组合索引(有顺序,ABC,ACB是完全不同的两种联合索引) 以联合索引(a,b,c)为例,建立这样的索引相当于建立了索引a、ab、abc三个索引 。另外组合索引实际还是一个索引,并非真的创建了多个索引,只是产生的效果等价于产生多个索引 。
索引下推:MySQL 5.6引入了索引下推优化,可以在索引遍历过程中,对索引中包含的字段先做判断,过滤掉不符合条件的记录,减少回表字数 。
索引维护:B 树为了维护索引有序性涉及到页分裂跟页合并 。增删数据时需考虑页空间利用率 。
自增主键:一般会建立与业务无关的自增主键 , 不会触发叶子节点分裂 。
延迟关联:通过使用覆盖索引查询返回需要的主键 , 再根据主键关联原表获得需要的数据 。
InnoDB存储: * .frm文件是一份定义文件 , 也就是定义数据库表是一张怎么样的表 。*.ibd文件则是该表的索引 , 数据存储文件,既该表的所有索引树,所有行记录数据都存储在该文件中 。
MyISAM存储:* .frm文件是一份定义文件,也就是定义数据库表是一张怎么样的表 。* .MYD文件是MyISAM存储引擎表的所有行数据的文件 。* .MYI文件存放的是MyISAM存储引擎表的索引相关数据的文件 。MyISAM引擎下,表数据和表索引数据是分开存储的 。
MyISAM查询:在MyISAM下,主键索引和辅助键索引都属于非聚簇索引 。查询不管是走主键索引,还是非主键索引,在叶子结点得到的都是目的数据的地址,还需要通过该地址,才能在数据文件中找到目的数据 。
PS:InnoDB支持聚簇索引 , MyISAM不支持聚簇索引
4、SQL事务隔离级别
ACID的四个特性
原子性(Atomicity):把多个操作放到一个事务中,保证这些操作要么都成功,要么都不成功
一致性(Consistency):理解成一串对数据进行操作的程序执行下来,不会对数据产生不好的影响,比如凭空产生 , 或消失
隔离性(Isolation,又称独立性):隔离性的意思就是多个事务之间互相不干扰,即使是并发事务的情况下,他们只是两个并发执行没有交集,互不影响的东西;当然实现中,也不一定需要这么完整隔离性,即不一定需要这么的互不干扰,有时候还是允许有部分干扰的 。所以MySQL可以支持4种事务隔离性
持久性(Durability):当某个操作操作完毕了,那么结果就是这样了,并且这个操作会持久化到日志记录中
PS:ACID中C与CAP定理中C的区别
ACID的C着重强调单数据库事务操作时,要保证数据的完整和正确性,数据不会凭空消失跟增加 。CAP 理论中的C指的是对一个数据多个备份的读写一致性
事务操作可能会出现的数据问题
1、脏读(dirty read):B事务更改数据还未提交,A事务已经看到并且用了 。B事务如果回滚,则A事务做错了
2、 不可重复读(non-repeatable read):不可重复读的重点是修改: 同样的条件, 你读取过的数据, 再次读取出来发现值不一样了,只需要锁住满足条件的记录
3、 幻读(phantom read):事务A先修改了某个表的所有纪录的状态字段为已处理,未提交;事务B也在此时新增了一条未处理的记录,并提交了;事务A随后查询记录 , 却发现有一条记录是未处理的造成幻读现象,幻读仅专指新插入的行 。幻读会造成语义上的问题跟数据一致性问题 。
4、 在可重复读RR隔离级别下,普通查询是快照读,是不会看到别的事务插入的数据的 。因此,幻读在当前读下才会出现 。要用间隙锁解决此问题 。
在说隔离级别之前 , 你首先要知道,你隔离得越严实,效率就会越低 。因此很多时候,我们都要在二者之间寻找一个平衡点 。SQL标准的事务隔离级别由低到高如下: 上图从上到下的模式会导致系统的并行性能依次降低,安全性依次提高 。
读未提交:别人改数据的事务尚未提交 , 我在我的事务中也能读到 。
读已提交(Oracle默认):别人改数据的事务已经提交,我在我的事务中才能读到 。
可重复读(MySQL默认):别人改数据的事务已经提交,我在我的事务中也不去读,以此保证重复读一致性 。
串行:我的事务尚未提交,别人就别想改数据 。
标准跟实现:上面都是关于事务的标准 , 但是每一种数据库都有不同的实现 , 比如MySQL InnDB 默认为RR级别,但是不会出现幻读 。因为当事务A更新了所有记录的某个字段,此时事务A会获得对这个表的表锁 , 因为事务A还没有提交,所以事务A获得的锁没有释放,此时事务B在该表插入新记录,会因为无法获得该表的锁,则导致插入操作被阻塞 。只有事务A提交了事务后,释放了锁,事务B才能进行接下去的操作 。所以可以说 MySQL的RR级别的隔离是已经实现解决了脏读,不可重复读和幻读的 。
5、MySQL中的锁
无论是Java的并发编程还是数据库的并发操作都会涉及到锁,研发人员引入了悲观锁跟乐观锁这样一种锁的设计思想 。
悲观锁:
优点:适合在写多读少的并发环境中使用,虽然无法维持非常高的性能,但是在乐观锁无法提更好的性能前提下 , 可以做到数据的安全性
缺点:加锁会增加系统开销,虽然能保证数据的安全,但数据处理吞吐量低,不适合在读书写少的场合下使用
乐观锁:
优点:在读多写少的并发场景下,可以避免数据库加锁的开销 , 提高DAO层的响应性能,很多情况下ORM工具都有带有乐观锁的实现,所以这些方法不一定需要我们人为的去实现 。
缺点:在写多读少的并发场景下,即在写操作竞争激烈的情况下,会导致CAS多次重试,冲突频率过高,导致开销比悲观锁更高 。
实现:数据库层面的乐观锁其实跟CAS思想类似,通数据版本号或者时间戳也可以实现 。
数据库并发场景主要有三种:
读-读:不存在任何问题,也不需要并发控制
读-写:有隔离性问题,可能遇到脏读,幻读,不可重复读
写-写:可能存更新丢失问题,比如第一类更新丢失,第二类更新丢失
两类更新丢失问题:
第一类更新丢失:事务A的事务回滚覆盖了事务B已提交的结果 第二类更新丢失:事务A的提交覆盖了事务B已提交的结果
为了合理贯彻落实锁的思想 , MySQL中引入了杂七杂八的各种锁:
锁分类
MySQL支持三种层级的锁定,分别为
表级锁定
MySQL中锁定粒度最大的一种锁,最常使用的MYISAM与INNODB都支持表级锁定 。
页级锁定
是MySQL中锁定粒度介于行级锁和表级锁中间的一种锁,表级锁速度快,但冲突多,行级冲突少,但速度慢 。所以取了折衷的页级 , 一次锁定相邻的一组记录 。
行级锁定
Mysql中锁定粒度最细的一种锁,表示只针对当前操作的行进行加锁 。行级锁能大大减少数据库操作的冲突 。其加锁粒度最小,但加锁的开销也最大行级锁不一定比表级锁要好:锁的粒度越细,代价越高,相比表级锁在表的头部直接加锁,行级锁还要扫描找到对应的行对其上锁,这样的代价其实是比较高的,所以表锁和行锁各有所长 。
MyISAM中的锁
虽然MySQL支持表 , 页,行三级锁定,但MyISAM存储引擎只支持表锁 。所以MyISAM的加锁相对比较开销低,但数据操作的并发性能相对就不高 。但如果写操作都是尾插入,那还是可以支持一定程度的读写并发
从MyISAM所支持的锁中也可以看出,MyISAM是一个支持读读并发 , 但不支持通用读写并发,写写并发的数据库引擎 , 所以它更适合用于读多写少的应用场合,一般工程中也用的较少 。
InnoDB中的锁
该模式下支持的锁实在是太多了,具体如下:
共享锁和排他锁 (Shared and Exclusive Locks)
意向锁(Intention Locks)
记录锁(Record Locks)
间隙锁(Gap Locks)
临键锁 (Next-Key Locks)
插入意向锁(Insert Intention Locks)
主键自增锁 (AUTO-INC Locks)
空间索引断言锁(Predicate Locks for Spatial Indexes)
举个栗子 , 比如行锁里的共享锁跟排它锁:lock in share modle 共享读锁:
为了确保自己查到的数据没有被其他的事务正在修改,也就是说确保查到的数据是最新的数据,并且不允许其他人来修改数据 。但是自己不一定能够修改数据 , 因为有可能其他的事务也对这些数据使用了 in share mode 的方式上了S 锁 。如果不及时的commit 或者rollback 也可能会造成大量的事务等待 。
for update排它写锁:
为了让自己查到的数据确保是最新数据 , 并且查到后的数据只允许自己来修改的时候,需要用到for update 。相当于一个 update 语句 。在业务繁忙的情况下,如果事务没有及时的commit或者rollback 可能会造成其他事务长时间的等待,从而影响数据库的并发使用效率 。
Gap Lock间隙锁:
1、行锁只能锁住行,如果在记录之间的间隙插入数据就无法解决了 , 因此MySQL引入了间隙锁(Gap Lock) 。间隙锁是左右开区间 。间隙锁之间不会冲突 。
2、间隙锁和行锁合称NextKeyLock,每个NextKeyLock是前开后闭区间 。
间隙锁加锁原则(学完忘那种):
1、加锁的基本单位是 NextKeyLock,是前开后闭区间 。
2、查找过程中访问到的对象才会加锁 。
3、索引上的等值查询 , 给唯一索引加锁的时候,NextKeyLock退化为行锁 。
4、索引上的等值查询,向右遍历时且最后一个值不满足等值条件的时候,NextKeyLock退化为间隙锁 。
5、唯一索引上的范围查询会访问到不满足条件的第一个值为止 。
如何用C语言连接MYSQL数据库1、配置ODBC数据源 。
2、使用SQL函数进行连接 。
对于1、配置数据源,配置完以后就可以编程操作数据库了 。
对于2、使用SQL函数进行连接,参考代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
#includewindows.h
#includesql.h
#includesqlext.h
void main()
{
HENV henv; //环境句柄
HDBC hdbc; //数据源句柄
HSTMT hstmt; //执行语句句柄
unsigned char datasource[]="数据源名称"; //即源中设置的源名称
unsigned char user[]= "用户名"; //数据库的帐户名
unsigned char pwd[]= "密码"; //数据库的密码
unsigned char search[]="select xm from stu where xh=0";
SQLRETURN retcode; //记录各SQL函数的返回情况
// 分配环境句柄
retcode= SQLAllocEnv(henv); // 等介于 SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL
, henv);
// 设置ODBC环境版本号为3.0
retcode= SQLSetEnvAttr(henv, SQL_ATTR_ODBC_VERSION, (void*)SQL_OV_ODBC3, 0);
// 分配连接句柄
retcode= SQLAllocConnect(henv,hdbc); // 等介于 SQLAllocHandle(SQL_HANDLE_DBC, henv, hdbc);
//设置连接属性,登录超时为*rgbValue秒(可以没有)
// SQLSetConnectAttr(hdbc, SQL_LOGIN_TIMEOUT, (SQLPOINTER)(rgbValue), 0);
//直接连接数据源
// 如果是windows身份验证 , 第二、三参数可以是
【mysql怎么编程 mysql编程基础】关于mysql怎么编程和mysql编程基础的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站 。
推荐阅读
- hbase导入优化,hbase scan优化
- 刘畅直播唱的歌是什么,刘畅唱歌视频
- raft手机怎么拿起水杯,raft怎么拿杯子
- php数据库信息删除 php删除记录
- 安卓能用css中透明度属性吗,安卓透明度代码
- asp.netmvc无刷新分页,aspnet分页控件
- 视频号0粉丝直播热门,视频号没有粉丝吗
- linux命令上传 linux命令
- 包含java求数相加代码大全的词条