Fork me on Gitee

Python爬虫--数据存储

数据存储

txt文件

占坑
代码


实例代码

Json文件

实例代码

csv文件

数据库存储

数据量多且复杂,存储到文件的数据管理不方便,效率较低。

将数据存储到数据库中,方便存储和管理。

关系型数据库

概念:

  • 关系型数据库是建立在关系模型基础的数据库。
  • 借助于集合代数等数学概念和方法处理数据库中的数据。
  • 由多张能互相连接的二维行列表格组成。

MYSQL


mysql优势
  • 开源(可根据需要修改),免费
  • 支持大型数据库,可以处理拥有上千万条记录的大型数据库
  • 核心线程完全是多线程,支持多处理器

mysql安装
sudo apt install mysql-server
启动服务器
  • 启动服务
sudo service mysql start
  • 重启服务
sudo service mysql restart
  • 查看服务进程
ps -axu|grep mysqld
启动客户端

-u 表示用户名, -p 表示密码

mysql -uroot -p  # -p后输入root用户密码
配置支持远程连接

MySQL默认不支持远程连接,只能本主机连接

配置:

  • 找到MySQL配置文件并修改
sudo vim /etc/mysql/mysql.conf.d/mysqld.cnf
将bind-address=127.0.0.1注释
  • 登录MySQL,修改Host值
登录
mysql -uroot -p
选择使用MySQL数据库
use mysql;
更新,将root原来的 Host值(localhost) 改成 %
localhost 表示只能本地登录,% 表示任何机器都可以登录
update user set Host="%" where User = "root";
刷新权限信息让所做的设置马上生效
flush privileges;
退出登录
exit;
  • 重启服务
sudo service mysql restart
  • 测试远程连接

-h 参数 指定客户端登录的MySQL主机名,登录本机,该参数可省略

如果是远程连接,则必须指定其值是远程主机的IP地址,这里的192.168.1.150就是主机的IP地址

mysql -h192.168.1.150 -uroot -p

数据库命令

查看数据库

show databases;

创建数据库

create database 数据库名 charset=utf8;

删除数据库

drop database 数据库名;

切换数据库

use 数据库名;

查看当前选择的数据库

select database();

表命令

查看当前数据库中所有表

show tables;

创建表

create table 表名(列及类型);

修改表

alter table 表名 add|modify|drop 列名 类型;

删除表

drop table 表名;

查看表结构

desc 表名;

更改表名称

rename table 原表名 to 新表名;

查看表的创建语句

show create table 表名;
增删改查命令

新增

insert into 表名 (field, field, ..., fieldN) values(value, value, ..., valueN);

删除

delete from 表名 [where 条件];

更新

update table_name set field = new-value, field = new-value [where 条件];

查询

select field, field from table_name [where 条件];

使用命令

流程

service mysql start  # 启动mysql
mysql -uroot -p1234 # 登录mysql
create database python default charset=utf8;  # 创建python数据库
use python; # 切换到python数据库
-- 创建grade表(主键表)
create table grade(
    id int primary key auto_increment,
    name varchar(100) not null
);
-- 创建student表(外键表)
create table student(
    id int primary key auto_increment,
    name varchar(100) not null,
    sex char(1) not null,
    phone char(11) unique not null,
    address varchar(100) default '郑州',
    birthday date not null,
    gid int not null,
    foreign key(gid) references grade(id)
);
-- 年级(grade)表中插入数据
insert into grade(name) values('一年级'); 
insert into grade(name) values('二年级');
-- 学生(student)表中插入数据
insert into student(name, sex, phone, address, birthday, gid) values('王强', '男' ,'15583678666' , '开封', '1990-2-4',  1);
insert into student(name, sex, phone, address, birthday, gid) values('李丽', '女', 16683678659'', '郑州', '1991-3-12', 2);
select * from grade;  # 查询年级(grade)表中的数据
select * from student;  # 查询学生表数据
update student set phone = '16683678657' where id = 2;  # 更改id为2的学生电话号码
delete from student where id = 1;  # 删除id为1的学生

Python与MySQL交互

安装pymysql模块

pip install pymysql -i https://pypi.tuna.tsinghua.edu.cn/simple

创建连接

'''Python与MySQL交互-创建连接'''
import pymysql
获取连接对象
conn = pymysql.Connect(host='localhost', port=3306, db='python', user='root', passwd='1234', charset='utf8')
print(conn)
关闭
conn.close()

说明:

  • MySQL的服务器必须先启动才可以连接成功
  • pymysql.Connect用来创建数据库连接对象
  • 参数
host='localhost', 表示连接主机是本机,也可指定IP地址
port=3306,主机端口号,默认是3306

host='localhost',连接主机,localhost表示连接主机是本机,也可使用指定IP地址

port=3306, 主机端口号,mysql默认端口号为3306,一般不用更改

db='python', 连接的数据库,这里连接的数据库是python

user='root', 用户名

passwd='1234',密码

charset='utf8', 编码格式

  • 连接对象使用完毕后要关闭,释放相关资源

查询和修改

Python与MySQL交互_查询与修改

import pymysql 

def select():
    '''查询'''
    try:
        # 获取连接对象
        conn = pymysql.Connect(host='localhost', port=3306, db='python', user='root', passwd='1234', charset='utf8')
        # 创建可执行对象,可以执行SQL语句
        cur = conn.cursor()
        # 执行SQL语句,并传递参数
        cur.execute('select * from student where id=%s', [2])
        # 查询一个结果
        result = cur.fetchone()
        print(result)
        # 关闭
        conn.close()
    except Exception as ex:
        print(ex)
  
        
def update():
    '''修改'''
    try:
        # 创建连接
        conn = pymysql.Connect(host='localhost', port=3306, db='python', user='root', passwd='1234', charset='utf8')
        # 创建可以执行SQL语句的对象
        cur = conn.cursor()
        # 执行SQL语句,并传递参数
        count = cur.execute('update student set name=%s  where id=%s', ['张三', 2])
        # 判断结果
        if count > 0:
            print('成功')
        else:
            print('失败')
        # 提交    
        conn.commit()
        # 关闭
        conn.close()
    except Exception as ex:
        print(ex)

if __name__ == '__main__':
    select()
    update()

说明:

  • 如果是增删改,cur执行SQL语句返回值是受影响的行数。
  • 如果是查询,则需要继续调用cur的查询方法,fetchone是查询符合条件的第一个对象,fetchall是查询符合条件的结果集

非关系型数据库

NoSQL(Not Only SQL)

特点

  • 不支持SQL语法
  • 存储数据KV形式
  • NoSQL没有通用语言,每种NoSQL数据库都有各自的API和语法,以及擅长的业务场景

对比

  • 适用场景:关系型数据库适用于关系特别复杂的数据查询场景,非关系型数据库反之
  • 事务特性的支持:关系型数据库对事务支持完善,非关系型数据基本不支持事务
  • 两者取长补短。对于爬虫的数据,可能有缺失值,可能结构不同。

用关系型数据库,建数据库,数据库表,关系约束,数据筛选,或拆分,这样使用效率较低;而非关系型数据库,直接以键值对存储,简单且效率高。

Redis

概述

用key-value存储,NoSQL的一种

值可以是字符串(String),哈希(Hash),列表(List),集合(Sets),有序集合(Sorted Sets)等类型

优势

  • 性能极高:读的速度110000次/s,写的速度是81000次/s
  • 支持的数据类型多:支持Strings,Lists,Hashes,Sets及Ordered Sets等数据类型的操作
  • 原子:Redis的所有操作都是原子性的,同时还支持对几个操作合并后的原子性执行
  • 丰富的特性:Redis还支持publish/subscribe,通知,key过期等特性
安装

1.打开Redis官方网站,下载稳定版本(stable),放到家目录下 cd /home

2.解压安装包

tar -zxvf ./redis.tar.gz  # 解压

3.移动

sudo mv ./redis/ /usr/local/

4.编译

cd /usr/local/redis/
sudo make

5.测试

sudo make test

6.将Redis的命令安装到/usr/local/bin/目录

sudo make install

启动

redis-server  # 启动服务器
redis-cli  # 启动客户端

测试连接

ping  # 打印PONG表示连接成功

远程访问配置

修改redis.conf配置文件

vim /usr/local/redis/redis.conf

注释bind

protected-mode改为no

bind 127.0.0.1
protected-mode no

使用配置文件方式启动

-- 复制配置文件到etc下的redis文件夹中
sudo mkdir /etc/redis/
sudo cp /usr/local/redis/redis.conf /etc/redis
-- 指定配置文件启动
sudo redis-server /etc/redis/redis.conf

停止Redis服务

ps ajx|grep redis  # 找到Redis的进程id
sudo kill -9 redis的进程id

命令操作

Redis地理位置命令

geoadd # 将指定的地理空间位置(纬度,经度,名称添加到指定的key中)
geodist  # 返回两个给定位置之间的距离
geohash  # 返回一个或多个位置元素的Geohash表示
geopos  # 从key中返回所有给定位置元素的位置
georadius  # 以给定的经纬度为中心,找出某一半径内的元素
georadiusbymember # 找出位于指定范围内的元素中心点由给定的位置元素决定

Redis键(key)命令

del  # 该命令用于在key存在时删除key
dump  # 序列化给定key,并返回被化的值
move  # 将当前数据库中的key移动到给定数据库db当中
rename  # 修改key的名称
renamenx  # 仅当newkey不存在时,将key改名为newkey

exists  # 检查给定key是否存在 
type  # 返回key所储存的值的类型
keys  # 查找所有符合给定模式的key
randomkey  # 从当前数据库中随机返回一个key

expire  # seconds为给定key设置过期时间
expireat  # 作用与expire类似,设置key过期时间,接受的时间参数是unix时间戳
persist  # 移除key的过期时间,key将持久保持
pttl  # 以毫秒为单位,返回key的剩余过期时间
ttl  # 以秒为单位,返回给定key的剩余生存时间

Redis字符串(String)命令

set  # 设置指定key的值
get  # 获取指定key的值


getrange  # 返回key中字符值的子字符

mget  # 获取所有(一个或多个)给定key的值
setbit  # 对key所储存的字符串值,设置或清除指定偏移量上的位
setex  # 将值value关联到key,并将key的过期时间设为seconds(以秒为单位)
setnx  # 只有在key不存在时设置key的值
setrange  # 用value参数覆写给定key所储存的字符串值,从偏移量offset开始
strlen  # 返回key所储存的字符串值的长度
getset  # 将给定key的值设为value,并返回key的旧值

mset   # 同时设置一个或多个key-value对
msetnx  # 同时设置一个或多个key-value对,当且仅当所有给定key都不存在

psetex  # 与setex相似,以毫秒为单位设置key的生存时间

incr # 将key中储存的数字值增一
incrby  # 将key所储存的值加上给定的增量值
incrbyfloat  # 将key所储存的值加上给定的浮点增量值

decr  # 将key中储存的数字值减一
decrby  # 将key所储存的值减去给定的减量值

append  #如果key已经存在并且是一个字符串,则append将value追加到key原来值的末尾

Redis哈希(hash)命令

hdel  # 删除一个或多个哈希表字段
hexists  # 查看哈希表key中,指字的字段是否存在 
hget  # 获取存储在哈希表中指定key的所有字段和值
hgetall  # 获取在哈希表中指定keyr所有字段和值
hincrby  # 为哈希表key中指定字段的整数值加上增量
hincrbyfloat  # 为哈希表上指定字段的浮点数值加上增量
hkeys  # 获取所有哈希表中的字段
hlen  # 获取哈希表中的字段的数量 
hmget  # 获取所有给定字段的值
hmset  # 同时将多个field-value(域-值)对设置到哈希表key中
hset  # 将哈希表中的key中的字段field的值设为value
hsetnx  # 只有在字段field不存在时,设置哈项表字段的值
hvals  # 获取哈希表中所有值

记忆: hash命令和string命令的区别在于,hash前面多了个h,去了h命令简写都和string命令差不多

Redis列表(List)命令

blpop  # 移出并获取列表的第一个元素,如果列表中没有元素,则会阻塞列表直到等待超时或发现可弹出元素为止
brpop  # 移出并获取列表的最后一个元素,如果列表中没有元素,则会阻塞列表直到等待超时或发现可弹出元素为止
brpoplpush  # 从列表弹出一个值,将弹出的元素插入另外一个列表蹉返回它;如果列表中没有元素,则会阻塞列表直到等待超时或发现可弹出元素为止
lindex  # 通过索引获取列表中的元素
linsert  # 在列表的元素前或后插入元素
llen  # 获取列表长度
lpop  # 移出并获取列表的第一个元素
lpush  # 将一个或多个值插入列表头部
lpushx  # 将一个或多个值插入已存在的列表头部
lrange  # 获取列表指定范围内的元素
lrem  # 移除列表元素
lset  # 通过索引设置列表元素的值
ltrim  # 对一列表进行修剪,也就让列表只保留指定区内的元素,不在指定区间内的元素都将被删除
rpop  # 移除并获取列表最后一个元素
rpoplpush  # 移除列表的最后一个元素,并将该元素添加到另一个列表并返回
rpush  # 在列表中添加一个或多个值
rpushx  # 为已存在的列表添加值

Redis集合(set)命令

sadd  # 向集合添加一个或多个成员
srem  # 移除集合中一个或多个成员

sismember  # 判断member元素是否集合key的成员
smembers  # 返回集合中的所有成员
scard # 获取集合的成员数

sscan #迭代集合中的元素

srandmember  # 返回集合中一个或多个随机数

spop  # 移除并返回集合中的一个随机元素
smove  # 将member元素从source集合到destination集合

sdiff  # 返回所有给定集合的差集
sdiffstore  # 返回所有给定集合的差集并存储在destination集合中
sinter  # 返回所有给定集合的交集
sinterstore  # 返回所有给定集合的交集并存储在destination集合中
sunion  # 返回所有给定集合的并集
sunionstore  # 返回所有给定集合的并集要并存储在destination集合中

记忆:s开头命令代表集合(set)命令

Redis有序集合(Sorted Set)命令

zadd  # 添加一个或多个成员,或者更新已存在成员的分数
zrem  # 移除一个或多个成员


zscan  # 迭代元素

�zscore  # 返回成员分数值
zcard  # 获取成员数

zincraby  # 对指定成员的分数加上增量increment
zranbyscore  # 通过分数返回指定区间内成员
zrangebylex  # 通过字典区间返回成员

zinterstore  # 计算给定的一个或多个的有序集合的交集并存储在新的key中
zunionstore  # 计算给定的一个或多个有序集合并集并存储在新的key中

zrank  # 返回指定成员的索引
zcount  # 计算在指定区间分数的成员数
zlexcount  # 计算指定字典区间内成员变量
zrange  # 通过索引区间返回指定区间内成员数量

zremrangebylex  # 移除给定的字典区间所有成员
zremrangebyrank  # 移除给定的排名区间所有成员
zremrangebyscore  # 移除给定的分数区间所有成员

zrevrank  # 返回指定成员,按分数值递减(从大到小)
zrevrange  # 返回指定区间内成员,通过索引,分数从高到低排序
zrevrangebyscore  # 返回指定分数区间内成员,分数从高到低排序

Redis HyperLogLog命令

pfadd  # 添加指定元素到HyperLogLog中
pfcount  # 返回给定HyperLogLog的基数估算值
pgmerge  # 将多个HyperLogLog合并成一个

Redis发布订阅命令

publish  # 将信息发送到指定的频道

pubsub  # 查看订阅与发布系统状态

subscribe  # 订阅给定的一个或多个频道的信息
psubscribe  # 订阅一个或多个符合给定模式的频道
unsubscribe  # 只退订给定的频道
punsubscribe  # 退订所有给定模式的频道

Redis事务命令

exec  # 执行所有事务块内的命令
discard  # 取消事务,放弃执行事务块内的所有命令

multi  # 标记一个事务块的开始 

watch  # 监视一个(或多个)key,如果在事务执行之前这个(这些)key被其他命令所改动,则事务将被打断
unwatch  # 取消watch命令对所有key的监视 

Redis脚本命令

eval  # 执行Lua脚本
evalsha # 执行Lua脚本 

scriptexists  # 查看指定的脚本是否已经被在缓存中
scriptflush  # 从脚本缓存中移除所有脚本 
scriptload  # 将脚本script添加到脚本缓存中,但并不立即执行这个脚本 

scriptkill  # 杀死当前正在运行的Lua脚本

Redis连接命令

bgrewriteaof  # 异步执行一个AOF(Append Only File)文件重写操作
bgsave  # 在后台异步保存当前数据库的数据到磁盘

client kill  # 关闭客户端连接
client list  # 获取连接到服务器的客户端连接列表
client getname  # 获取连接的名称
client pause  # 在指定时间内终止运行来自客户端的命令
client setname  # 设置当前连接的名称
cluster slots  # 获取集群节点的映射数组
command  # 获取详情数组
command count  # 获取总数
command getkeys  # 获取给定的所有键
time  # 返回当前服务器时间
command info  # 获取指定描述的数组
config get  # 获取指定配置参数的值
config rewrite  # 对启动服务器时所指定的.conf配置文件进行改写
config set   # 修改配置参数,无须重启
config resetstat  # 重置INFO中的某些统计数据
dbsize  # 返回当前数据库的key的数量
debug object  # 获取key的调试信息
debug segfault  # 让服务器崩溃
flushall  # 删除所有数据库的所有key
flushdb  # 删除当前数据库的所有key
info  # 获取服务器的各种信息和统计数值 , db=
lastsave  # 返回最近一次成功将数据保存到磁盘上的时间,以UNIX时间戳格式表示 
monitor  # 实时打印出服务器接收到的命令,调试用
role  # 返回主从实例所属的角色
save  # 异步保存数据到硬盘
shutdown  # 异步保存数据到硬盘,并关闭服务器
slaveof  # 将当前服务器转变为指定服务器的从属服务器
showlog  # 管理Redis的慢日志
sync  # 用于复制功能的内部命令
Python与Redis交互

安装模块

pip install redis 

创建连接

'''Python与Redis交互-创建连接'''
import redis
client = redis.StrictRedis(host='localhost', port=6379, db=0)
print(client)
# 简写
# client = redis.StrictRedis()

说明:
Redis的服务器必须启动才可以连接成功
redis默认无密
redis.StrictRedis用来创建数据库连接对象
参数,
host:连接主机
port:主机端口号
db:要连接的数据库
连接对象不需要手动关闭。
当用Redis和StrictRedis创建连接时,内部实现没有主动创建一个连接,获得的连接是连接池提供的,由连接池管理,所以无法关注连接是否需要主动释放。
连接池有自己关闭连接的接口,一旦调用该接口,所有连接都将被关闭。

增删改查

# Python与Redis交互-增删改查

from redis import *

    
def insert_update():
    '''新增/修改'''
    try:
        # 创建StrictRedis对象,与Redis服务器建立连接
        sr = StrictRedis()
        # 添加键name,值为python
        # 如果键name不存在,则为新增,否则为修改
        result = sr.set('name', 'python')
        # 输出响应结果,如果添加成功,则返回True,否则返回False
        print(result)
    except Exception as e:
        print(e)
    
def select():
    '''查询'''
    try:
        # 连接
        sr = StrictRedis()
        # 获取键name的值
        result = sr.get('name')
        # 输出键的值,如果键不存在,则返回None
        print(result)
    except Exception as e:
        print(e)
            
def delete():
    '''删除'''
    try:
        # 连接
        sr = StrictRedis()
        # 删除
        result = sr.delete('name')
        # 输出响应结果,删除成功,返回受影响的键数,否则返回0
        print(result)
    except Exception as e:
        print(e)
            
        
if __name__ == '__main__':
    insert_update()
    select()
    delete()

MongoDB

概念

基于分布式(主从复制,负载均衡)文件存储的NoSQL数据库,由C++语言编写,运行稳定,性能高,旨在为Web应用提供可扩展的高性能数据存储解决方案

特性

  • 模式自由:可以把不同结构的文档存储在同一个数据库中
  • 面向集合存储:适合存储对象及JSON形式的数据
  • 完整的索引支持:对任何属性可索引
  • 复制和高可用性:支持服务器之间的数据复制,支持主-从模式及服务器之间的相互复制,复制的主要目的是提供冗余及自动故障转移
  • 自动分片以支持云级别的伸缩性:自动分片功能 支持水平的数据库集群,可动态添加额外的机器
  • 丰富的查询:支持丰富的查询表达方式,查询指令使用形式的标记,可轩昂查询文档中内嵌的对象及数组
  • 快速就地更新:查询优化器会分析查询表达式,并生成一个高效的查询计划
  • 高效的传统存储方式:支持二进制数据及大型对象(如照片或图片)

优势

  • 更高的写负载
  • 处理很大规模的单表。数据库扩展是非常有挑战性的,当表格大小达到5-10GB时,MySQL表格性能会毫无疑问的降低。如果需要分片并分割数据库,MongoDB很容易实现
  • 不可靠环境保证高可用性。设置副本集(主-从服务器设置)不仅方便,而且很快。此外,使用MongoDB还可以快速,安全及自动化地实现节点(或数据中心)故障转移
  • 使用基于位置的数据查询,查得更快。MongoDB支持二维空间索引,如管道,因此可以快速及精确地从指定位置获取数据据。MongoDB在启动后会将数据库中的数据以文件映射的方式加载到内存中。如果内存资源相当丰富,那么将极大地提高数据库的查询速度,毕竟内存I/O效率比磁盘高很多。
  • 非结构化数据的爆发增长 。增加列在有些情况下可能锁定整个数据库,或者增加负载从而导致性能下降,这个问题通常发生在表格大于1GB的情况。鉴于MongoDB的弱数据结构模式,添加一个新字段不会对旧表格有任何影响,整个过程会非常快速;因此,在应用程序发生改变时,不需要专门的一个DBA(数据库管理员)去修改数据库模式。
  • 技术门槛相对低。如果没有专业的DBA,同时也不需要结构化数据及做join查询,那么MongoDB将会是首选。MongoDB非常适合类的持久化,类可以被序列化成JSON并储存在MongoDB中。需要注意的是,如果期望获得一个更大的规模,就必须要了解一些最佳实践来避免走入误区。
使用

安装
从官网下载安装包,选择合适的版本下载
32bit的MongoDB最大只能存放2GB的数据,64bit就没有限制
mongodb-linux-x86_64-ubuntu.tgz

解压安装包

tar -zxvf mongodb-linux-x86_64-ubuntu.tgz

移动

sudo mv mongodb-linux-x86_64-ubuntu.tgz/ /usr/local/mongodb

新建配置文件mongod.conf,放到/etc目录下,写入6行命令

port=27017
dbpath=/var/lib/mongodb/
logpath=/var/log/mongodb/mongodb.log
fork=true
logappend=true
noauth=true  # 无密

创建相关文件和文件

sudo mkdir /var/lib/mongodb/
sudo mkdir /var/log/mongodb/
sudo touch /var/log/mongodb/mongodb.log

创建软链接,方便调用mongod和mongo命令

sudo ln -sf /usr/local/mongodb/bin/mongod /usr/local/sbin/mongod
sudo ln -sf /usr/local/mongodb/bin/mongo /usr/local/sbin/mongo

启动
启动服务器端

mongod

sudo mongod -f /etc/mongod.conf

启动客户端

sudo mongo

查看数据库的状态信息

db.stats()

关闭服务器

use admin 
db.shutdownServer()
常用的数据类型

object id # 文档id
string # 字符串,最常用,必须是有效的UTF-8
boolean # 布尔值,True或False
integer # 整数可以是32位或64位,取决于服务器
double # 浮点值
array # 数组或列表,多个值存储到一个键
object # 用于嵌入式的文档,即一个值为一个文档
null # 存储null值
timestamp # 时间戳
date # 存储当前日期或时间的UNIX时间格式

说明:
object id。每个文档都有一个属性,为_id,保证每个文档的唯一性,可以自己去设置_id插入文档。
如果没有提供,MonoDB为每个文档提供了一个独特的_id,类型为ObjectID.
ObjectID是一个12字节的十六进制数,前4个字节为当前时间戳,后3个字节为机器ID,后2个字节为MongoDB的服务进程id,最后3个字节为简单的增量值

命令操作

use mydb
db.stu.insert([
		{
		'name': "郭靖",
        "hometown": '蒙古',
        "age": 20,
        'gender': true
    },
	{
		'name': '黄蓉',
        'hometown': '桃花岛',
        'age': 18,
        'gender': false
    }
])

数据库命令

db  # 查看当前数据库名称

show dbs  # 查看所有数据库名称,列出所有在物理上存在的数据库

use 数据库名称  # 切换数据库
�/*如果数据库不存在,则指向数据库,直到插入数据或创建集合时数据库才被创建。
默认的数据库为test,如果没有创建新的数据库,则集合存放在test数据库中*/

db.dropDatabase()  # 删除当前指向的数据库,如果数据库不存在,则什么也不做

db相当于MySQL中的database

集合命令

db.createCollection(name, options)  # 创建集合  
/*name:要创建的集合名称,必须有。db.createCollection('stu')
选项参数options:文档,用于指定集合的配置。
capped:默认False,不设置上限,Ture,设置(集合大小)上限
size:capped值为True时指定,上限大小,文档达到上限时,会覆盖之前数据,单位字节 
db.createCollection("sub", {capped:true, size:10})
*/

show collections  # 查看当前数据库的集合

db.集合名称.drop()  # 删除集合

这里的集合相当于MySQL的表

数据操作命令

db.集合名称.insert(document)  # 新增一条文档信息
/*
db.stu.insert({name:'qj', gender:1})
*/

db.集合名称.find()  # 查询当前集合所有文档信息

db.集合名称.update(<query>, <update>, {multi:<boolean>})  
/*
参数:
query:查询条件
update:更新操作符
multi:可选,默认False,只更新找到的第一条记录,True,把满足条件的文档全部更新
示例:
全文档更新
db.stu.update({name:'hr', {name:'mnc'}})
指定属性更新
�db.stu.insert({'name':'hr', gender:0})
db.stu.update({name:'hr',{$set:{name:'hys'}}})
修改多条匹配到的数据
db.stu.update({}, {$set:{gender:0}}, {multi:true})��*/
�
db.集合名称.save(document)  # 保存
/*
如果文档的_id已存在,则修改;如果文档的_id不存在,则添加 
db.stu.save({_id:'20160102', 'name':'yk', gender:1})
db.stu.save({_id:'20160102', 'name': 'wyk'})*/


db.集合名称.remove(<query>, {justOne:<boolean>}) # 删除
/*
参数:
query:删除文档的条件
justOne:设为True或1,则只删除一条,默认为False,删除多条
示例:
只删除一条
db.stu.remove({gender:0}, {justOne:true})
全部删除
db.stu.remove({})*/

这里的文档document相当于MySQL的一行数据,其语法格式相当于JavaScript的JSON对象

数据查询命令

db.集合名称.find({条件文档})  # 根据条件查询,返回结果集

db.集合名称.findOne({条件文档})  # 根据条件查询,返回第一个

db.集合名称.find({条件文档}).pretty()  # 将查询的结果格式化

$lt, $lte, $gt, $gte, $ne  
/*
等于,默认是等于判断,没有运算符
小于$lt
小于或等于$lte
大于$gt
大于或等于$gte
不等于$ne
示例:
查询名称等于'qj'的学生
db.stu.find({name:'gj'})
查询年龄大于或等于18的学生
db.stu.find({age:{\$gte:18}})
*/


and, or   # 逻辑与,逻辑或  默认是逻辑与
/*
示例:
查询年龄大于或等于18,并且性别为1的学生
db.stu.find({age:{$gte:18},gender:1})
查询年龄大于18或性别为1的学生
db.stu.find({$or:[{age:{$gt:18}}, {gender:1}]})
查询年龄大于18或性别为1的学生,并且学生的姓名为gj
db.stu.find({$or[{age:{$gte:18}}, {gender:1}], name:'gj'})
*/

$in, $nin  # 是否在某个范围内
/*
查询年龄为18,28的学生
db.stu.find({age:{$in:[18, 28]}})
*/

$regex, //  # 编写正则表达式
/*
查询姓黄的学生
db.stu.find({name:/^黄/})
db.stu.find({name:{$regex:'^黄'}})
*/


$where  # 后写一个函数,返回满足条件的数据
/*查询年龄大于20的学生
db.stu.find({$where:function(){return this.age>20}})
*/

limit()  # 用于读取指定数量的文档
/*语法:db.集合名称.find().limit(NUMBER)
参数NUMBER表示要获取文档的条数
未指定,则显示集合跌所有文档
例:查询2条学生信息
db.stu.find().limit(2)
*/

skip()  # 用于跳过指定数量的文档
/*语法:db.集合名称.find()/skip(NUMBER)
参数number表示跳过的记录条数,默认为0
示例:
查询从第3条开始的学生信息
db.stu.find().skip(2)
方法limit()和skip()可以一起使用,不分先后
sb.stu.find().limit(4).skip(1)
db.stu.find().skip(1).limit(4)
*/
投影   # 在查询的结果中,只选择必要的字段,而不是一个文档的整个字段 
/*
参数为字段与值,值1为显示,值0为不显示
db.stu.find({}, {name:1, gender:1})
db.stu.find({}, {_id:0, name:1, gender:1})
*/

sort()  # 用于对结果集进行排序
/*
语法:db.集合名称.find().sort({字段1:1, 字段2:-1,...})
参数1为升序,参数-1为降序
db.stu.find().sort({gender:-1, age:1})
*/
count()  # 用于统计结果集中文档条数
/*
语法:db.集合名称.find({条件}).count()
db.集合名称.count({条件})
示例:
db.stu.find({gender:1}).count()
db.stu.count({age:{$gt:20}, gender:1})
*/
distinct()  # 对数据进行去重
/*
语法:db.集合名称.distinct('去重字段', {条件})
例:查找年龄大于18的性别(去重)
db.stu.distinct('gender', {age:{$gt:18}})
*/

管道命令

linux:将当前命令的输出结果作为下一个命令的输入
mongodb:具有同样作用,文档处理完毕,通过管道进行下一次处理

$group  # 将集合的文档分组,可用于统计结果
/*
_id表示分组的依据,使用某个字段的格式为'$字段'db.stu.aggregate([{$group:{_id:'$gender', conter:{$sum:1}}}])
*/
$match  # 用于过滤数据,只输出符条件的文档
$project  # 修改输入文档的结构,如重命名,增加,删除字段,创建计算结果
$sort  # 将输入文档排序后输出
$limit  # 限制聚合管道返回的文档数
$skip  # 路过指定数量的文档,并返回余下的文档
$limit和$skip  # 先写skip,再写limit
$unwind  # 将文档中的某一个数组类型字段拆分成多条,每条包含数组中的一个值
Python与MongoDB交互

安装pymongo模块

pip install pymongo

创建连接

# Python与MongoDB交互--创建连接

import pymongo 
myclient = pymongo.MongoClient('mongodb://localhost:27017/')
mydb = myclient['mydb']
print(mydb)
myclient.close()

说明:
MongoDB的服务器必须启动才可以连接成功
创建数据库需要使用MongoClient对象,并且指定连接的URL地址和要创建的数据库名
localhost是要连接的主机,也可以用ip地址访问,27017是主机端口号,mydb是要选择的数据库

增删改查

# MongoDB-增删改查

import pymongo

def is_having(mydb):
    dblist = myclient.list_database_names()
    if 'mydb' in dblist:
        print('数据库已存在!')
    else:
        print('数据库不存在!')
    
    return mydb
    
def insert(mydb):
    stu = mydb['stu']
    _id = stu.insert({
        'name': '扫地僧',
        'hometown':'少林寺',
        'age': 66,
        'gender': True
        
    })
    
    print(_id)
    
def select(mydb):
    stu = mydb['stu']
    ret = stu.find()
    for i in ret:
        print(i)

        
def update(mydb):
    stu = mydb['stu']
    x = stu.update_many({'age': {'$gt': 20}}, {'$inc': {'age': 1}})
    print(x.modified_count, '文档已修改')
    

def delete(mydb):
    stu = mydb['stu']
    x = stu.delete_many({'age': {'$gt': 20}})
    print(x.deleted_count, '个文档已删除')
    myclient.close()
    
if __name__ == '__main__':
    myclient = pymongo.MongoClient('mongodb://localhost:27017/')
    mydb = myclient['mydb']
    is_having(mydb)
    insert(mydb)
    select(mydb)
    update(mydb)
    delete(mydb)
    myclient.close()

说明:
在MongoDB中,数据库只有在内容插入后才会创建。也就是说,数据库创建后要创建集合(数据表)并插入一个文档(记录),数据库才会真正创建。
MongoClient对象的list_database_names()获取所有数据

爬豆瓣电影,保存到MongoDB

# 爬豆瓣电影-保存到MongoDB

import requests
import random 
import time 
import threading
import json
import csv
import pymongo
from lxml import etree
from queue import Queue 


class DouBanSpider:
    '''爬虫类'''
    def __init__(self):
        self.headers = {'User-Agent': 'Mozilla/5.0'}
        self.baseURL = 'https://movie.douban.com/top250'
        # MongoDB客户端对象
        self.client = pymongo.MongoClient('mongodb://localhost:27017/')['mydb']['douban']
        
        
    def loadPage(self, url):
        '''向url发送请求,获取响应内容'''
        
        # 随机休眠0-2秒,避免爬虫过快,会导致爬虫封禁
        time.sleep(random.random() * 2)
        return requests.get(url, headers=self.headers).content
        
        
    def parsePage(self, url):
        '''根据起始url提取所有的url'''
        # 获取url对应的响应内容
        content = self.loadPage(url)
        html = etree.HTML(content)
        # 所有电影节点
        node_list = html.xpath('//div[@class="info"]')
        # 遍历
        for node in node_list:
            # 使用字典存储数据
            item = {}
            # 每部电影的标题
            item['title'] = node.xpath('.//span[@class="title"]/text()')[0]
            # 每部电影的评分
            item['score'] = node.xpath('.//span[@class="rating_num"]/text()')[0]
            self.client.insert(item)
            print(item)
        # 只有在第一页时才获取所有url组成的列表,其他页就不获取    
        if url == self.baseURL:
            return [self.baseURL + link for link in html.xpath('//div[@class="paginator"]/a/@href')]
            
            
    def startWork(self):
        '''开始'''
        print('begin....')
        # 第一个页面的请求,需要返回所有页面链接,并提取第一页的电影信息
        link_list = self.parsePage(self.baseURL)
        thread_list = []
        # 循环发送每个页面的请求,并获取所有电影信息
        for link in link_list:
            # 循环创建了9个线程,每个线程都执行一个任务
            thread = threading.Thread(target=self.parsePage, args=[link])
            thread.start()
            thread_list.append(thread)
        # 父线程等待所有子线程结束,自己再结束
        for thread in thread_list:
            thread.join()
        print('end...')
        
        
        
if __name__ == '__main__':
    # 创建爬虫对象
    spider = DouBanSpider()    
    # 开始爬虫
    spider.startWork()
posted @ 2021-09-26 12:43  Veryl  阅读(159)  评论(0编辑  收藏  举报
动态线条
动态线条end