五月综合激情婷婷六月,日韩欧美国产一区不卡,他扒开我内裤强吻我下面视频 ,无套内射无矿码免费看黄,天天躁,日日躁,狠狠躁

新聞動態(tài)

Redis整合MySQL主從集群的示例代碼

發(fā)布日期:2022-01-29 20:34 | 文章來源:源碼中國

Redis作為承擔緩存作用的數(shù)據(jù)庫,一般會應(yīng)用在高并發(fā)的場景里,而在這些高并發(fā)應(yīng)用場景的數(shù)據(jù)庫層面還會用到其他數(shù)據(jù)庫的組件或集群以提升性能,比如用MySQL主從集群實現(xiàn)讀寫分離效果、用MyCAT組件實現(xiàn)分庫分表的功能。另外,Redis本身會以集群的形式對外提供緩存服務(wù)。

1、用Docker搭建MySQL主從集群

這里用Docker容器搭建如下圖所示的MySQL主從集群。

  • 在主MySQL服務(wù)器里操作的動作會自動同步到從nysql服務(wù)器,比如在主服務(wù)器里發(fā)起的“建數(shù)據(jù)庫”“通過insert語句插入數(shù)據(jù)”和“通過delete語句刪除數(shù)據(jù)”的動作都會同步到從服務(wù)器,并且這些操作都會在從服務(wù)器上被執(zhí)行。通過這種同步的動作能確保主從數(shù)據(jù)庫間的數(shù)據(jù)一致性。
  • 在項目里,一般是向“主服務(wù)器”里寫數(shù)據(jù),從“從服務(wù)器”里讀數(shù)據(jù),用這種“讀寫分離”的操作方式提升數(shù)據(jù)庫性能。

具體搭建的步驟如下:

1.1 拉取mysql鏡像

  開啟一個命令窗口,在其中運行docker pull mysql:latest,下載最新的mysql鏡像。下載完成后,通過docker images mysql能看到如下圖所示的鏡像i信息。

1.2 創(chuàng)建配置文件夾

新建/root/redisconf/masterMySQL/conf/root/redisconf/masterMySQL/data兩個目錄,在其中將會保存主mysql服務(wù)器的配置信息和數(shù)據(jù)。同時新建/root/redisconf/slaveMySQL/conf/root/redisconf/slaveMySQL/data兩個目錄,在其中將會保存從MySQL服務(wù)器的配置信息和數(shù)據(jù)。當然,目錄可自行更改。

1.3 編寫主服務(wù)器的配置文件信息

  在/root/redisconf/masterMySQL/conf目錄里新建一個my.cnf文件,在其中編寫針對主mysql服務(wù)器的配置信息,主mysql服務(wù)器在啟動時會讀取其中的配置,具體代碼如下所示:

[mysqld]
pid-file =/var/run/mysqld/mysqld.pid
socket =/var/run/mysqld/mysqld.sock
datadir =/var/lib/mysql
server-id =1
log-bin=mysql-master-bin

  第二行到第四行給出了MYSQL運行時的參數(shù),在第五行里定義了該服務(wù)器的id(這個id需要和之后編寫的從MySQL服務(wù)器的server-id不一樣,否則會出錯),在第6行里制定了二進制文件的名字(為了搭建主從集群,建議加上這行配置)

1.4 啟動mysql主服務(wù)器的容器

docker run -itd --privileged=true -p 3306:3306 \
--name myMasterMysql -e MYSQL_ROOT_PASSWORD=123456\
-v /root/redisconf/masterMySQL/conf:/etc/mysql/conf.d\
 -v /root/redisconf/masterMySQL/data:/var/lib/mysql mysql:latest

-p3306:3306參數(shù)指定Docker容器里MySQL的工作端口3306映射到主機的3306端口
-itd參數(shù)指定該容器以后臺交互模式的方式啟動
--name參數(shù)指定該容器的名字
通過-e MYSQL_ROOT_PASSWORD=123456參數(shù)指定該容器運行時的環(huán)境變量,具體到這個場景,配置以用戶名root登錄到MySQL服務(wù)器時所用到的密碼123456.
兩個-v參數(shù)指定外部主機和Docker容器間映射的目錄。由于在第三步把MySQL啟動時需要加載的my.cnf文件放在了/root/redisconf/masterMySQL/conf目錄里,因此這里需要把/root/redisconf/masterMySQL/conf目錄映射成容器內(nèi)部MYSQL服務(wù)器的相關(guān)路徑。
通過mysql:latest參數(shù)指定該容器是基于這個鏡像生成的。

查看啟動的容器:docker ps


由此能確認myMasterMysql啟動成功。
查看該Docker容器的IP地址:docker inspect myMasterMysql


可以看到,這里是172.17.0.2,這也是主mysql服務(wù)器所在的Ip地址。

1.5 觀察主服務(wù)器狀態(tài)

  運行docker exec -it myMasterMysql /bin/bash命令后進入該myMasterMysql容器的命令行窗口,再運行mysql -u root -p命令,進入MYSQL服務(wù)器的命令行窗口,在這個Mysql命令里,以-u參數(shù)指定用戶名,隨后需要輸入密碼(剛才設(shè)置的123456).

進入mysql服務(wù)器之后,再運行show master status命令觀察主服務(wù)器的狀態(tài)。

  

可以看到,主從集群同步所用到的日志文件是mysql-master-bin.000003,當前同步的位置是156,每次運行這個命令看到的結(jié)果未必相同,請記住這兩個值,在設(shè)置從mysql服務(wù)器的主從同步關(guān)系時會用到。

1.6 配置mysql從服務(wù)器

  在/root/redisconf/slaveMaster/conf目錄里,新建一個名為my.cnf的文件,編寫針對從MYSQL服務(wù)器的配置信息。同樣的,從Mysql服務(wù)器在啟動時也會讀取其中的配置,具體代碼如下所示。

[mysqld]
pid-file=/var/run/mysqld/mysqld.pid
socket=/var/run/mysqld/mysqld.sock
datadir=/var/lib/mysql
server-id=2
log-bin=mysql-slave-bin

  該配置文件和第三步創(chuàng)建的主服務(wù)器的配置文件很相似,只不過在第5行更改了server-id(這里的取值不能和主mysql服務(wù)器的一致)。在第6行也是設(shè)置二進制文件的名字

1.7 啟動mysql從服務(wù)器

docker run -itd --privileged=true -p 3316:3306\
 --name mySlaveMysql -e MYSQL_ROOT_PASSWORD=123456\
 -v /root/redisconf/slaveMySQL/conf:/etc/mysql/conf.d\
 -v /root/redisconf/slaveMySQL/data:/var/lib/mysql\
  mysql:latest

  這里在-p參數(shù)之后使用主機的3316端口映射Docker容器的3306端口,因為之前主mysql服務(wù)器的Docker容器已經(jīng)映射到了3306端口,其他的參數(shù)和之前創(chuàng)建myMasterMysql容器時很相似,就不再重復了。

  隨后docker exec -it mySlaveMysql /bin/bash進入容器,進入后可以運行mysql -h 172.17.0.2 -u root -p命令,嘗試在這個容器里連接主MySQL服務(wù)器。其中172.17.0.2是主服務(wù)器的地址。隨后輸入root用戶的密碼123456,即可確認連接。


  確認鏈接后,通過exit命令退出指向myMasterMysql的連接,再通過mysql -h 127.0.0.1 -u root -p命令連接到本Docker容器包含的從MySQL服務(wù)器上。

1.8 確認主從關(guān)系

change master to master_host='172.17.0.2',master_port=3306,\
master_user='root',master_password='123456',\
master_log_pos=156,\
master_log_file='mysql-master-bin.000003';

   本命令運行在mySlaveMysql容器中的從Mysql服務(wù)器里,通過master_host和master_port指定主服務(wù)器的ip地址和端口號,通過master_user和master_password設(shè)置了連接所用的用戶名和密碼。
  注意master_logpos和master_log_file兩個參數(shù)的值需要和第5步圖中的結(jié)果一致。

  運行完成后,需要再運行start slave命令啟動主從復制的動作。運行后可以通過show slave status\G;命令查看主從復制的狀態(tài),如果Slave_IO_Running和Slave_SQL_Running這兩項都是Yes,并且沒有其他異常,就說明配置主從復制成功。


  此時如果再到主mysql服務(wù)器里運行create database redisDemo創(chuàng)建一個數(shù)據(jù)庫,那么從庫里雖然沒有運行命令,但是也能看到redisDemo數(shù)據(jù)庫,這說明已經(jīng)成功地搭建了MySQL主從復制集群。其中,主庫地IP地址和端口號是172.17.0.2:3306,從庫是172.17.0.3:3306.

主庫

從庫

2、準備數(shù)據(jù)

  由于已經(jīng)成功地設(shè)置了主從復制模式,因此如下地建表和插入語句都只需要在主庫里運行。

2.1 創(chuàng)建數(shù)據(jù)庫

create database redisDemo

進入redisDemo數(shù)據(jù)庫use redisDemo

2.2 創(chuàng)建student數(shù)據(jù)表

create table student(
    id int not null primary key,
    name char(20),
     age int,
     score float
    );

2.3 向student表插入幾條數(shù)據(jù)

insert into student(id,name,age,score) values(1,'Peter',18,100);
insert into student(id,name,age,score) values(2,'Tom',17,98);
insert into student(id,name,age,score) values(3,'John',17,99);


從庫里查看

3、用Java代碼讀寫MySQL集群和Redis

3.1 引入redis和mysql依賴

   <!-- https://mvnrepository.com/artifact/redis.clients/jedis -->
        <dependency>
            <groupId>redis.clients</groupId>
            <artifactId>jedis</artifactId>
            <version>3.3.0</version>
        </dependency>
        <!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java -->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.21</version>
        </dependency>

  Java應(yīng)用程序是向主mySQL服務(wù)器寫數(shù)據(jù),這樣寫入地數(shù)據(jù)會自動同步到從mysql服務(wù)器上,而讀數(shù)據(jù)時會先從Redis緩存里讀,讀不到時再到從mysql里讀。以下用代碼實現(xiàn)

3.2 代碼整合

MySQLClusterDemo.java

import redis.clients.jedis.Jedis;
import java.sql.*;
public class MySQLClusterDemo {
    //創(chuàng)建操作Redis和數(shù)據(jù)庫的對象
    private Jedis jedis;
    private Connection masterConn;  //連接主庫的對象
    private Connection slaveConn;   //連接從庫的對象
    PreparedStatement masterPs=null;    //對主庫進行操作的對象
    PreparedStatement slavePs=null;     //對從庫進行操作的對象
    //初始化環(huán)境
    private void init(){
        //MYSQL的連接參數(shù)
        String mySQLDriver="com.mysql.cj.jdbc.Driver";
        String masterUrl="jdbc:mysql://192.168.159.33:3306/redisDemo?useSSL=false&serverTimezone=UTC&allowPublicKeyRetrieval=true";
        String slaveUrl="jdbc:mysql://192.168.159.33:3316/redisDemo?useSSL=false&serverTimezone=UTC&allowPublicKeyRetrieval=true";
        String user="root";
        String pwd="123456";
        try{
            Class.forName(mySQLDriver);
            masterConn= DriverManager.getConnection(masterUrl,user,pwd);
            slaveConn= DriverManager.getConnection(slaveUrl,user,pwd);
            jedis=new Jedis("192.168.159.33",6379);
        }catch (SQLException e){
            e.printStackTrace();
        }catch (Exception e){
            e.printStackTrace();
        }
    }
    private void insertData(){
        //是向主MySQL服務(wù)器插入數(shù)據(jù)
        try{
        masterPs=masterConn.prepareStatement("insert into student(id,name,age,score) values(10,'Frank',18,100)");
        masterPs.executeUpdate();
        }catch (SQLException e){
            e.printStackTrace();
        }catch (Exception e){
            e.printStackTrace();
        }
    }
    private String getNameByID(String id){
        String key="Stu"+id;
        String name="";
        //如果存在于Redis,就先從Redis里獲取
        if(jedis.exists(key)){
            System.out.println("ID:"+key+" exists in Redis");
            name=jedis.get(key);
            System.out.println("Name is :"+jedis.get(key));
            return name;
        }else{  //如果沒在Redis里,就到從MySQL里去讀
            try {
                slavePs=slaveConn.prepareStatement("select name from student where id=10");
                ResultSet rs=slavePs.executeQuery();
                if(rs.next()){
                    System.out.println("ID: "+key+" exists in Slave MySQL");
                    name=rs.getString("name");
                    System.out.println("Name is: "+name);
                    //放入Redis緩存
                    jedis.set(key,name);
                }
                return name;
            }catch (SQLException e){
                e.printStackTrace();
            }catch (Exception e){
                e.printStackTrace();
            }
        }
        return name;
    }
    public static void main(String[] args) {
        MySQLClusterDemo tool=new MySQLClusterDemo();
        tool.init();
        tool.insertData();
        //場景1 沒有從Redis中找到,就到從MySQL服務(wù)器中去讀
        System.out.println(tool.getNameByID("10"));
        //場景2,當前ID=10的數(shù)據(jù)已存在于Redis,所有直接讀緩存
        System.out.println(tool.getNameByID("10"));
    }
}

運行結(jié)果

4、MySQL主從集群整合Redis主從集群

  上面的mysql主從集群至整合了一個Redis主機,在這種模式里如果Redis服務(wù)器失效了,那么整個緩存可能都會失效。可以在次基礎(chǔ)上引入Redis主從復制集群,以提升緩存的可用性以及性能,改進后的框架圖如下所示。

應(yīng)用程序同樣是向主mysql服務(wù)器里寫數(shù)據(jù),這些數(shù)據(jù)同步到從mysql數(shù)據(jù)庫里。

應(yīng)用程序先到“從Redis服務(wù)器”里讀取緩存,如果找不到,就再到從mysql數(shù)據(jù)庫里去讀。

如果從“從mysql數(shù)據(jù)庫”里讀到數(shù)據(jù),那么需要寫入“主Redis”,而根據(jù)Redis集群的主從復制機制,該數(shù)據(jù)會被寫入“從Redis服務(wù)器”。這種針對Redis集群的讀寫分離機制能提升讀寫緩存的性能。

4.1 搭建Redis主從復制集群

4.1.1 創(chuàng)建redis-master容器

docker run -itd --name redis-master -p 6379:6379 redis:latest

4.1.2 創(chuàng)建resis-slave容器

docker run -itd --name redis-slave -p 6380:6379 redis:latest

4.1.3 查看redis服務(wù)器的ip

docker inspect redis-master


可以看到,redis-master的ip地址為172.17.0.4

4.1.4 主從配置

  在redis-slave容器的窗口里,通過docker exec -it redis-slave /bin/bash命令進入容器的命令行窗口。運行如下的slaveof命令,指定當前服務(wù)器為從服務(wù)器,該命令的格式是slaveof IP地址 端口號,這里指向172.17.0.2:6379所在的主服務(wù)器。

slaveof 172.17.0.4 6379

運行完該命令后,在redis-slave客戶端里再次運行info replication

  可以看到,該redis-slave已經(jīng)成為從服務(wù)器,從屬于172.17.0.2:6379所在的Redis服務(wù)器。

4.2、代碼整合

MySQLClusterImprovedDemo.java

import redis.clients.jedis.Jedis;
import java.sql.*;
public class MySQLClusterImprovedDemo {
    //創(chuàng)建操作Redis和數(shù)據(jù)庫的對象
    private Jedis masterJedis;  //指向主Redis服務(wù)器
    private Jedis slaveJedis;   //指向從Redis服務(wù)器
    private Connection masterConn;  //連接主庫的對象
    private Connection slaveConn;   //連接從庫的對象
    PreparedStatement masterPs=null;    //對主庫進行操作的對象
    PreparedStatement slavePs=null;     //對從庫進行操作的對象
    private void init(){
        //MYSQL的連接參數(shù)
        String mySQLDriver="com.mysql.cj.jdbc.Driver";
        String masterUrl="jdbc:mysql://192.168.159.33:3306/redisDemo?useSSL=false&serverTimezone=UTC&allowPublicKeyRetrieval=true";
        String slaveUrl="jdbc:mysql://192.168.159.33:3316/redisDemo?useSSL=false&serverTimezone=UTC&allowPublicKeyRetrieval=true";
        String user="root";
        String pwd="123456";
        try{
            Class.forName(mySQLDriver);
            masterConn= DriverManager.getConnection(masterUrl,user,pwd);
            slaveConn= DriverManager.getConnection(slaveUrl,user,pwd);
            masterJedis=new Jedis("192.168.159.33",6379);
            slaveJedis=new Jedis("192.168.159.33",6380);
        }catch (SQLException e){
            e.printStackTrace();
        }catch (Exception e){
            e.printStackTrace();
        }
    }
    private void insertData(){
        //是向主MySQL服務(wù)器插入數(shù)據(jù)
        try{
            masterPs=masterConn.prepareStatement("insert into student(id,name,age,score) values(10,'Frank',18,100)");
            masterPs.executeUpdate();
        }catch (SQLException e){
            e.printStackTrace();
        }catch (Exception e){
            e.printStackTrace();
        }
    }
    private String getNameByID(String id){
        String key="Stu"+id;
        String name="";
        //如果存在于Redis,就先從Redis里獲取
        if(slaveJedis.exists(key)){ //到從Redis服務(wù)器去找
            System.out.println("ID: "+key+" exists in Redis.");
            name=slaveJedis.get(key);//找到后到從Redis里讀
            System.out.println("Name is: "+slaveJedis.get(key));
            return name;
        }else{  //沒在Redis,就到從MySQL去讀
            try{
                slavePs=slaveConn.prepareStatement("select name from student where id=10");
                ResultSet rs=slavePs.executeQuery();
                if(rs.next())
                {
                    System.out.println("ID: "+key+" exists in Slave MySQL");
                    name=rs.getString("name");
                    System.out.println("Name is: "+name);
                    //放入主Redis緩存
                    masterJedis.set(key,name);
                }
                return name;
            }catch (SQLException e){
                e.printStackTrace();
            }catch (Exception e){
                e.printStackTrace();
            }
        }
        return name;
    }
    public static void main(String[] args) {
        MySQLClusterImprovedDemo tool=new MySQLClusterImprovedDemo();
        tool.init();
        tool.insertData();
        //場景1 在主Redis中沒有讀到,則到從MySQL服務(wù)器中讀
        System.out.println(tool.getNameByID("10"));
        //場景2 當前ID=10已經(jīng)存在于Redis,所以直接讀緩存
        System.out.println(tool.getNameByID("10"));
    }
}

   為了突出重點,這里我并沒有設(shè)置“緩存失效時間”和“防止緩存穿透”等方面的實施代碼,但是這些要點同樣重要。

到此這篇關(guān)于Redis整合MySQL主從集群的示例代碼的文章就介紹到這了,更多相關(guān)Redis整合MySQL主從集群內(nèi)容請搜索本站以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持本站!

香港穩(wěn)定服務(wù)器

版權(quán)聲明:本站文章來源標注為YINGSOO的內(nèi)容版權(quán)均為本站所有,歡迎引用、轉(zhuǎn)載,請保持原文完整并注明來源及原文鏈接。禁止復制或仿造本網(wǎng)站,禁止在非maisonbaluchon.cn所屬的服務(wù)器上建立鏡像,否則將依法追究法律責任。本站部分內(nèi)容來源于網(wǎng)友推薦、互聯(lián)網(wǎng)收集整理而來,僅供學習參考,不代表本站立場,如有內(nèi)容涉嫌侵權(quán),請聯(lián)系alex-e#qq.com處理。

實時開通

自選配置、實時開通

免備案

全球線路精選!

全天候客戶服務(wù)

7x24全年不間斷在線

專屬顧問服務(wù)

1對1客戶咨詢顧問

在線
客服

在線客服:7*24小時在線

客服
熱線

400-630-3752
7*24小時客服服務(wù)熱線

關(guān)注
微信

關(guān)注官方微信
頂部