mongodb作为一个数据库与所有的数据库一样都面临一个问题:当只存在一个实例时,数据是不安全的,如果当前实例挂掉程序从其中读取不了数据不说。如果装载此实例的载体遭到破坏有可能会导致数据的永远丢失。这在如今这个基本上所有互联网公司都要靠数据吃饭的时代是不能接受的。所以了解一下如何让数据进行备份与多实例读取时很重要的。
本文旨在让读者能简单的实现mongodb的主从复制与集群复制,废话不多说,直接开始操作。
1.主从复制
①主机配置文件:
dbpath=d:\mongodb\data\db
logpath=d:\mongodb\logs\mongodb.log
port=27017
#auth=true
master=true
dbpath指向的是mongodb存储数据的位置
logpath指向mongodb产生日志的位置
port说明将会启用的端口号
auth代表是否开启认证
master是最关键的,为true代表指定此机器为主机
②从机配置文件:
dbpath=d:\mongodb\data2\db
logpath=d:\mongodb\logs2\mongodb.log
port=27018
#auth=true
slave=true
source=127.0.0.1:27017
slave为true代表指定此机器为主机
source指向的是主机的ip:端口号
③分别启动两个实例此时
④操作主机端插入并查询数据:
⑤操作从机端写入与读取数据:
⑥主机端读写都是没有问题的,但是从机读写都有问题,这并不是我们想要的,从机对于我们来说应该具有读的功能。其实并不是配置什么的出现了问题。只需要在从机端执行一个命令:
rs.slaveOk()
之后就能进行读操作了,但是由于不是主机,所有还是不能进行写入的操作。
至此,一个简单的主从复制就完成了。接下来我们来进行集群复制的操作。
2.集群复制
①配置文件(每个节点配置文件都只需要增加一个参数replSet=你设置的集群名字)
dbpath=d:\mongodb\data4\db
logpath=d:\mongodb\logs4\mongodb.log
port=27017
#auth=true
replSet=testColony
②启动节点
③此时集群配置还未生效,需要进行一下配置。需要在任意节点进行进一步的配置(暂时只加两个节点,剩下一个节点待会儿在进行说明):
先将配置赋值给rscfg
rscfg={_id:'testColony',members:[{_id:0,host:'127.0.0.1:27017'},{_id:1,host:'127.0.0.1:27018'}]}
在调用rs.initiate(),将rscfg传入
rs.initiate(rscfg)
然后调用rs.status()查看当前集群状态:
> rscfg={_id:'testColony',members:[{_id:0,host:'127.0.0.1:27017'},{_id:1,host:'127.0.0.1:27018'}]}
{
"_id" : "testColony",
"members" : [
{
"_id" : 0,
"host" : "127.0.0.1:27017"
},
{
"_id" : 1,
"host" : "127.0.0.1:27018"
}
]
}
> rs.initiate(rscfg)
{ "ok" : 1 }
testColony:OTHER> rs.status()
{
"set" : "testColony",
"date" : ISODate("2017-10-19T07:37:34.444Z"),
"myState" : 1,
"term" : NumberLong(1),
"heartbeatIntervalMillis" : NumberLong(2000),
"optimes" : {
"lastCommittedOpTime" : {
"ts" : Timestamp(1508398647, 1),
"t" : NumberLong(1)
},
"appliedOpTime" : {
"ts" : Timestamp(1508398647, 1),
"t" : NumberLong(1)
},
"durableOpTime" : {
"ts" : Timestamp(1508398647, 1),
"t" : NumberLong(1)
}
},
"members" : [
{
"_id" : 0,
"name" : "127.0.0.1:27017",
"health" : 1,
"state" : 1,
"stateStr" : "PRIMARY",
"uptime" : 1372,
"optime" : {
"ts" : Timestamp(1508398647, 1),
"t" : NumberLong(1)
},
"optimeDate" : ISODate("2017-10-19T07:37:27Z"),
"infoMessage" : "could not find member to sync from",
"electionTime" : Timestamp(1508398645, 1),
"electionDate" : ISODate("2017-10-19T07:37:25Z"),
"configVersion" : 1,
"self" : true
},
{
"_id" : 1,
"name" : "127.0.0.1:27018",
"health" : 1,
"state" : 2,
"stateStr" : "SECONDARY",
"uptime" : 19,
"optime" : {
"ts" : Timestamp(1508398647, 1),
"t" : NumberLong(1)
},
"optimeDurable" : {
"ts" : Timestamp(1508398647, 1),
"t" : NumberLong(1)
},
"optimeDate" : ISODate("2017-10-19T07:37:27Z"),
"optimeDurableDate" : ISODate("2017-10-19T07:37:27Z"),
"lastHeartbeat" : ISODate("2017-10-19T07:37:33.977Z"),
"lastHeartbeatRecv" : ISODate("2017-10-19T07:37:32.578Z"),
"pingMs" : NumberLong(0),
"syncingTo" : "127.0.0.1:27017",
"configVersion" : 1
}
],
"ok" : 1
}
④此时集群配置已经做好了命令提示符也发生了变化:
先试试查看所有数据库的命令:
show dbs
此时发现在secondary上不能查询,执行以下命令:
rs.slaveOk()
之后就可以查询了
但是还是不能插入数据
主机插入是没问题的
⑤增加仲裁节点:
rs.addArb("127.0.0.1:27019")
此时就算停掉primary节点secondary会自动成为primary节点。