mysql第二天

Mysql 第二天

  • 补充

  • 索引

  • 增删改查

  • 多表联合查询

 

补充

表字段改名 change

1.修改字段类型  modify 
2.添加字段 add first/after
3.删除字段 drop
4.表改名   rename
补充  
5.字段改名 change  

alter table users change 旧字段 新字段 类型(长度);

MySQL 查看帮助

? 关键字 

 

数据类型 排序

最大到最小 字符串类型->时间类型->数值类型

性别 一般存数值类型

时间可以存时间戳

如果精度要求高 选择 decimal 字符串型的浮点数

 

索引

比如新华字典的 目录 比如书的目录

目的是 快速找到制定的内容

如果不用索引 需要从头到尾轮询 效率慢 索引 就是 先划定一个范围

类型

  • 普通索引

  • 唯一索引

  • 主键索引

  • 全文索引

  • 复合索引

普通索引

最基本的索引没有任何限制的

? index 
发现创建索引有两种方式  
1.alter table  
2.create index

区别: create index 不能创建主键索引

alter table 表名 add index(字段名);#创建索引 这个没有制定索引名称 那么名称跟字段名相同
alter table 表名 add index 索引名称(字段名); #创建索引并且指定索引的名称
show index from 表名\G #查看索引  
alter table 表名 drop index 索引名字; #删除索引


create index 索引名字 on 表名(字段名); #创建索引  
drop index 索引名称 on 表名; #删除索引
show index from 表名\G

 

唯一索引 unique

要求这一列不能有重复值 比如年龄 性别 这样的列

alter table 表名 add unique 索引名称(username);
alter table 表名 drop index 索引名称;

 

主键索引

每个表一定有个主键 不能有重复值 不能有空值 一般创建表的时候 我们就创建主键索引 比如 id

特殊的唯一索引

alter table users drop primary key; #删除主键索引 如果自动递增这一步报错 
ERROR 1075 (42000): Incorrect table definition; there can be only one auto column and it must be defined as a key

alter table users id int(11) unsigned not null;#消除自动递增
alter table users drop primary key; #再次执行成功


添加主键索引  

alter table users add primary key(id);#创建主键索引
alter table users modify id int(11) unsigned not null auto_increment; #让主键自动递增

 

全文索引

文章中 或者电商 或者社交网站 对需要全局搜索的内容 进行全文索引

alter table 表名 add fulltext 索引名字(字段名);

 

复合索引

对多个字段 同时添加索引

text 类型不能添加 普通索引、复合索引

alter table users add index a_test(username,age);

 

总结 建表的同时添加索引

create table if not exists in_test(
id int(11) unsigned not null primary key auto_increment,
username varchar(50) not null,
password varchar(50) not null,
content text not null,
index pw(password),
unique(username),
fulltext(content)
)engine=myisam default charset=utf8;

 

增删改查

增 insert

mysql> desc in_test;
+----------+------------------+------+-----+---------+----------------+
| Field    | Type             | Null | Key | Default | Extra          |
+----------+------------------+------+-----+---------+----------------+
| id       | int(11) unsigned | NO   | PRI | NULL    | auto_increment |
| username | varchar(50)      | NO   | UNI | NULL    |                |
| password | varchar(50)      | NO   | MUL | NULL    |                |
| content  | text             | NO   | MUL | NULL    |                |
+----------+------------------+------+-----+---------+----------------+
4 rows in set (0.00 sec)
#上面有几个字段 就要写 对应字段的内容 一个都不能落下  

insert into in_test values(1,'abc','123456','kangbazi666'); #主键 username不能出现重复值 否则报错  
#只是插入指定字段对应的内容即可  主键自增  sex 默认值 可以不用插入   其它字段不能为空
insert into in_test(username,password,content) values('zhangsan','123456','kangbazi888');


insert into money(username,password,balance,province,age,sex) values('laowang','123456','1000000','湖北',28,0),('laoshen','aaa',123.45,'湖北',40,0),('laoning','11111',999999999,'辽宁',38,0),('caixukun','xxxxxxs',23456.12,'日本',29,1); #一次性插入多条记录

查 select

 select * from 表名; #   *代表 所有的字段的意思  
 select 字段1,字段2 from 表名; # 查看指定的字段 
 
 mysql> select username,content from in_test;
+----------+-------------+
| username | content     |
+----------+-------------+
| abc      | kangbazi666 |
+----------+-------------+
1 row in set (0.00 sec)

mysql> select username as 姓名,content as 内容 from in_test; #as 显示结果的别名
+--------+-------------+
| 姓名   | 内容        |
+--------+-------------+
| abc    | kangbazi666 |
+--------+-------------+

删 delete

delete from 表名#清空表 

原来有3条记录  下次再插入数据  id 从4开始  除非指定id的值  

delete from 表名 where 字段=值;#删除指定的内容  where 是条件 
 

 

truncate table 清空表

 truncate table 表名; #下次插入数据 id下标就从1开始   

 

改 update

update 表名 set 字段='新内容' where 字段=值; #更新指定的内容 

数据库查询 扩展

create table if not exists money(
	id int(11) unsigned not null primary key auto_increment,
	username varchar(64) not null,
	password char(64) not null,
	province varchar(16) not null,
	balance float not null,
	age tinyint default 18,
	sex tinyint default 0
)engine=myisam default charset=utf8;


insert into money(username,password,balance,province,age,sex) values('laowang','123456','1000000','湖北',28,0),('laoshen','aaa',123.45,'湖北',40,0),('laoning','11111',999999999,'辽宁',38,0),('caixukun','xxxxxxs',23456.12,'日本',29,1); #一次性插入多条记录

 

distinct 查询单个字段不重复记录
 select distinct age from money; #查询money 表年龄唯一的结果
  select distinct 字段 from 表名;
where 条件
 select * from money where age=18;
 select 字段 from 表名 where 字段=值;
运算符 说明
>  
<  
>=  
<=  
!=  
=  
or 或者
and 并且
   
select username as '姓名',balance as '余额',province as '省份' from money where id<10 and province='湖北';

结果集排序 order by
  • asc 升序

  • desc 降序

select id,username,balance,province from money order by balance desc;#按照余额降序排列
select id,username,balance,province from money order by balance asc;#升序
select 字段1,字段2,字段n from 表名 order by 字段 关键词;
多个字段排序  如果第一个字段已经将结果排序ok,那么第二个字段将会被忽略

mysql> select id,username,balance,province,age from money order by balance desc,age asc;

结果集限制 limit
mysql> select * from money limit 5;
限制结果集并且排序
mysql> select id,username,balance,province,age from money order by balance desc,age asc limit 5;

结果集区间选择 limit
select 字段 from 表名 limit 偏移量,数量;
下标从0开始

mysql> select * from money limit 2,5; #下标为2 第三条记录开始 取5条结果
用户从前端提交	  下标 数量(程序员自己设定)   
第1页  		   0    3     
第2页            3    3
第3页            6    3
第4页            9    3

下标 =  (第几页-1)* 每页显示的数量

 

统计函数
函数 作用
count() 统计总数
sum() 求和
avg() 平均数
max() 最大值
min() 最小值
   
mysql> select count(id) from money; #统计总数
+----------+
| count(id) |
+----------+
|        8 |
+----------+
1 row in set (0.00 sec)

mysql> select sum(balance) from money;
+-------------------+
| sum(balance)      |
+-------------------+
| 9756671443.651169 |
+-------------------+
1 row in set (0.00 sec)

mysql> select avg(balance) from money;
+-------------------+
| avg(balance)      |
+-------------------+
| 1219583930.456396 |
+-------------------+
1 row in set (0.01 sec)



mysql> select max(balance) from money;
+--------------+
| max(balance) |
+--------------+
|   7666670080 |
+--------------+
1 row in set (0.00 sec)

mysql> select min(balance) from money;
+--------------------+
| min(balance)       |
+--------------------+
| 123.44999694824219 |
+--------------------+
1 row in set (0.00 sec)

 

分组 group by
进入mysql   
select @@sql_mode;
会发现有 ONLY_FULL_GROUP_BY 表名这是按照Oracle的分组规则 也就是说 select 所有的列都要在分组中  
mysql> select @@sql_mode;
+-------------------------------------------------------------------------------------------------------------------------------------------+
| @@sql_mode                                                                                                                                |
+-------------------------------------------------------------------------------------------------------------------------------------------+
| ONLY_FULL_GROUP_BY,STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION |
+-------------------------------------------------------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)



修改配置文件 

sudo vim /etc/mysql/mysqld.conf.d/mysqld.cnf  

[mysqld]

sql_mode = STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION


service mysql restart   


重新登录mysql  
select @@sql_mode; 
就发现 ONLY_FULL_GROUP_BY 取消了 


mysql> select * from money group by province; # 相同的省份只显示一个
+----+------------+----------+------------+----------+------+------+
| id | username   | password | balance    | province | age  | sex  |
+----+------------+----------+------------+----------+------+------+
|  6 | 郭德纲     | 1112212  | 1000000000 | 天津     |   50 |    0 |
|  7 | bingbing   | 1122345  |   88888900 | 山东     |   45 |    1 |
|  4 | caixukun   | xxxxxxs  |    23456.1 | 日本     |   29 |    1 |
|  5 | xiaoyueyue | 11233    |    88888.1 | 河南     |   18 |    0 |
|  1 | laowang    | 123456   |    1000000 | 湖北     |   28 |    0 |
|  3 | laoning    | 11111    | 1000000000 | 辽宁     |   38 |    0 |
+----+------------+----------+------------+----------+------+------+
6 rows in set (0.00 sec)



mysql> select count(province) as '人数',province from money group by province;
#统计省份的数量以后再显示  
+--------+----------+
| 人数   | province |
+--------+----------+
|      1 | 天津     |
|      1 | 山东     |
|      1 | 日本     |
|      1 | 河南     |
|      3 | 湖北     |
|      1 | 辽宁     |
+--------+----------+
6 rows in set (0.00 sec)



mysql> select count(province) as '人数',province from money group by province with rollup;
# 相比较上面结果 多了同一总数的统计    
+--------+----------+
| 人数   | province |
+--------+----------+
|      1 | 天津     |
|      1 | 山东     |
|      1 | 日本     |
|      1 | 河南     |
|      3 | 湖北     |
|      1 | 辽宁     |
|      8 | NULL     |  #这里多了一行总的结果 
+--------+----------+
7 rows in set (0.01 sec)


分组的结果过滤一下 having
mysql> select count(province) as result,province from money group by province having result>1; 
+--------+----------+
| result | province |
+--------+----------+
|      3 | 湖北     |
+--------+----------+

 

总结

select 你选择的列 [as "显示的别名"]
from 表
where 条件
group by 分组 [having 分组过滤的条件]
order by 字段 
limit 偏移量,数量或者 数量

 

多表联合查询

  • 内连接 inner join on

  • 外连接

    • 左连接 left join on

    • 右连接 right join on

  • 子查询

准备

mysql> create table if not exists users(
    uid int(11) unsigned not null primary key auto_increment,
    username varchar(64) not null,
    password varchar(64) not null
)engine=innodb default charset=utf8;
Query OK, 0 rows affected (0.01 sec)

mysql> create table if not exists order_goods(
    oid int(11) unsigned not null primary key auto_increment,
    uid int(11) not null,
    name varchar(64) not null,
    buytime int(11) not null
)engine=innodb default charset=utf8;
Query OK, 0 rows affected (0.01 sec)

 

内连接  

1.select 表1.字段1,表1.字段2 as '别名',表2.字段1 as '别名',表2.字段2 from 表1,表2 where 表1.字段=表2.字段;


mysql> select users.uid,users.username as '用户名',order_goods.name as '商品名',order_goods.buytime from users,order_goods where users.uid=order_goods.uid;
+-----+-----------+------------+---------+
| uid | 用户名    | 商品名     | buytime |
+-----+-----------+------------+---------+
|   2 | zhicheng  | iPhone11   | 1111112 |
|   4 | huwei     | 杜蕾斯     |   12345 |
|   6 | Lee       | wawa       |  123455 |
|   3 | daoke     | jinpingmei | 1121212 |
+-----+-----------+------------+---------+

select u.uid,u.username as '用户名',o.name as '商品名',o.buytime from users u,order_goods o where u.uid=o.uid; #上面的简化版   字段显示别名 as 表起别名 空格即可 

2. select 表1.字段1 [as 别名],表n.字段 from 表1 inner join 表n on 条件;


mysql> select users.uid as ID,users.username as '姓名',order_goods.name as "商品名",order_goods.buytime from users inner join order_goods on users.uid=order_goods.uid;

 

外连接

1.左连接   select 表1.字段1 [as 别名],表n.字段 from 表1 left join 表n on 条件;
以lef join 左边的表为准  用户表 买东西的 和没买的都给你显示  

mysql> select users.uid as ID,users.username as '姓名',order_goods.name as "商品名",order_goods.buytime from users left join order_goods on users.uid=order_goods.uid;
+----+----------+------------+---------+
| ID | 姓名     | 商品名     | buytime |
+----+----------+------------+---------+
|  2 | zhicheng | iPhone11   | 1111112 |
|  4 | huwei    | 杜蕾斯     |   12345 |
|  6 | Lee      | wawa       |  123455 |
|  3 | daoke    | jinpingmei | 1121212 |
|  1 | xuelin   | NULL       |    NULL |
|  5 | xinxin   | NULL       |    NULL |
+----+----------+------------+---------+
6 rows in set (0.00 sec)



2.右连接    select 表1.字段1 [as 别名],表n.字段 from 表1 right join 表n on 条件;
以right join 右边的表为准   
mysql> select users.uid as ID,users.username as '姓名',order_goods.name as "商品名",order_goods.buytime from users right join order_goods on users.uid=order_goods.uid;
+------+----------+------------+---------+
| ID   | 姓名     | 商品名     | buytime |
+------+----------+------------+---------+
|    2 | zhicheng | iPhone11   | 1111112 |
|    4 | huwei    | 杜蕾斯     |   12345 |
|    6 | Lee      | wawa       |  123455 |
|    3 | daoke    | jinpingmei | 1121212 |
+------+----------+------------+---------+
4 rows in set (0.00 sec)


3.如果上面左连接为例子  null 可以显示默认值    

(case when 表.字段 is null then 默认值 else 表.字段 end) as '别名'

mysql> SELECT users.uid,users.username as 用户名,(case when order_goods.`name` is NULL THEN '没买东西' ELSE order_goods.name END) as 商品名 from users LEFT JOIN order_goods on users.uid=order_goods.uid;
+-----+-----------+--------------+
| uid | 用户名    | 商品名       |
+-----+-----------+--------------+
|   2 | zhicheng  | iPhone11     |
|   4 | huwei     | 杜蕾斯       |
|   6 | Lee       | wawa         |
|   3 | daoke     | jinpingmei   |
|   1 | xuelin    | 没买东西     |
|   5 | xinxin    | 没买东西     |
+-----+-----------+--------------+
6 rows in set (0.00 sec)


4.如果上面左连接为例子  null 可以显示默认值  

ifnull(表.字段,'默认值')

mysql> SELECT users.uid,users.username as 用户名, IFNULL(order_goods.name,"空空 如也") as 商品名 from users LEFT JOIN order_goods on users.uid=order_goods.uid;
+-----+-----------+--------------+
| uid | 用户名    | 商品名       |
+-----+-----------+--------------+
|   2 | zhicheng  | iPhone11     |
|   4 | huwei     | 杜蕾斯       |
|   6 | Lee       | wawa         |
|   3 | daoke     | jinpingmei   |
|   1 | xuelin    | 空空如也     |
|   5 | xinxin    | 空空如也     |
+-----+-----------+--------------+
6 rows in set (0.00 sec)

 

子查询

1.SELECT * from users where uid in(2,3,4,6);# 查询购买用户的详细信息   括号中的值是写死的


2.SELECT * from users where uid in(select uid from order_goods);#先从一个表中查出记录再 将结果作为条件 从另外表中查询 

in 
not in 
= 
!= 
exists 
not exists 

 

 

union union all

两个表的数据按照一定条件查询出来 结果并到一起显示

mysql> select uid from users union all select uid from order_goods;
+-----+
| uid |
+-----+
|   1 |
|   2 |
|   3 |
|   4 |
|   5 |
|   6 |
|   2 |
|   4 |
|   6 |
|   3 |
+-----+
10 rows in set (0.00 sec)


mysql> select uid from users union select uid from order_goods; #去除重复记录  
+-----+
| uid |
+-----+
|   1 |
|   2 |
|   3 |
|   4 |
|   5 |
|   6 |
+-----+
6 rows in set (0.00 sec)

 

DCL 数据库控制语句

首先选中 mysql

每个数据库服务器都有一个mysql数据库

mysql 数据库存放着 权限用户等 重要信息 每个数据库都是不同的

grant 权限 on 数据库.数据表 to '用户名'@'主机' identified by '密码' with grant option;

权限: all 所有权限 
	 select 
	 insert 
	 update
	 delete
数据库: 
	 * 所有的数据库
数据表:
	* 所有的数据表
	
用户名:
	用户名如果不是存在的 那么就会新建 
主机: 
	%  任意主机
	
flush privileges; #刷新权限  


mysql> grant select,insert on test.users to 'mxl'@'%' identified by '666666';
Query OK, 0 rows affected, 1 warning (0.00 sec)

mysql> flush privileges;
Query OK, 0 rows affected (0.01 sec)




revoke 权限 on 数据库.数据表 from '用户名'@'主机';
flush privileges; #刷新权限  



mysql> revoke insert on test.users from 'mxl'@'%';
Query OK, 0 rows affected (0.00 sec)

mysql> flush privileges;
Query OK, 0 rows affected (0.00 sec)



 

数据库的备份及恢复 Linux命令

备份 mysqldump

mysqldump -u root -p test > /tmp/test.sql

 

恢复 mysql

 mysql -u root -p test < /tmp/test.sql

计划任务 备份数据库

which mysqldump
crontab -e  
0 2 * * *  /usr/bin/mysqldump -u root -p123456 数据库名 > /tmp/test.sql

 

 

Python 操作mysql

python不能直接操作mysql 需要通过 扩展

pymysql(开源)

https://github.com/PyMySQL/PyMySQL

安装pymysql

1.pip install pymysql 

2.sudo apt-get install git 
  git clone https://github.com/PyMySQL/PyMySQL.git
  cd PyMySQL/
  python setup.py install

Python 操作 mysql 五行拳

  • 连接数据库

  • 创建游标

  • 执行sql 语句

  • 获取结果集

  • 关闭连接

#encoding:utf-8
import pymysql

#连接数据库
#主机地址 
#用户名
#密码
#数据库名字

#五行拳第一拳 
db = pymysql.connect("127.0.0.1",'root','123456','test')

#第二拳  
cursor =db.cursor()


#准备sql语句 
sql = 'select * from money order by balance limit 3'

#执行sql语句
cursor.execute(sql)

#获取结果集  
data = cursor.fetchone()
print(data)

#释放句柄对象  
cursor.close()

#关闭mysql 连接 
db.close()

 

创建数据表

#encoding:utf-8
import pymysql

#连接数据库
#主机地址 
#用户名
#密码
#数据库名字

#五行拳第一拳 
db = pymysql.connect("127.0.0.1",'root','123456','test')

#第二拳  
cursor =db.cursor()


#准备sql语句 
sql = 'create table if not exists kangbazi(id int(11) unsigned not null primary key auto_increment,username varchar(64) not null)engine=innodb default charset=utf8'

#执行sql语句
cursor.execute(sql)

#释放句柄对象  
cursor.close()

#关闭mysql 连接 
db.close()

 

插入数据

#encoding:utf-8
import pymysql

#连接数据库
#主机地址 
#用户名
#密码
#数据库名字

#五行拳第一拳 
db = pymysql.connect("127.0.0.1",'root','123456','test')

#第二拳  
cursor =db.cursor()


#准备sql语句 
sql = 'insert into kangbazi(username) values("kangbazi1"),("kangbazi2")'
try:
    #执行sql语句
    cursor.execute(sql)
    db.commit()
except:
    db.rollback() #只有innodb引擎才支持事务
#释放句柄对象  
cursor.close()

#关闭mysql 连接 
db.close()

 

作业

1.封装一个类实现增删改查  

你可能感兴趣的:(mysql第二天)