redis学习摘录

## 5.    Redis的数据类型

### 5.1    Redis的5种数据类型

```
redis是一种高级的key-value的存储系统,其中value支持五种数据类型:

·        字符串(**String)**

·        哈希(**hash)**

·        字符串列表(**list)**

·        字符串集合(**set)**

·        有序字符串集合(sorted set)

在日常开发中主要使用比较多的有字符串、哈希、字符串列表、字符串集合四种类型,其中最为常用的是字符串类型。

 关于key的定义,注意如下几点:

·        key不要太长,最好不要超过1024个字节,这不仅会消耗内存还会降低查找效率

·        key不要太短,如果太短会降低key的可读性 

·        在项目中,key最好有一个统一的命名规范


```

### 5.2    字符串类型string

**使用场景:**

```
常规key-value缓存应用。常规计数: 微博数, 粉丝数,json格式的数据。
```

#### 5.2.1    字符串类型string概述

```
字符串类型是Redis中最为基础的数据存储类型,字符串在Redis中是二进制保存,因此是安全的,这便意味着该类型存入和获取的数据相同。

在Redis中字符串类型的Value最多可以容纳的数据长度是512M。
```

#### 5.2.2    字符串类型string常用命令

·        命令 : **set key value**

设定key持有指定的字符串value,如果该key存在则进行覆盖操作。总是返回”OK”

```cmd
127.0.0.1:6379> set company "itcast"

OK

127.0.0.1:6379>
```

·       获取保存的数据命令:  **get key**

获取key的value。如果与该key关联的value不是String类型,redis将返回错误信息,因为get命令只能用于获取String value;如果该key不存在,返回(nil)。

```cmd
127.0.0.1:6379> get company

"itcast"
```

·        **del key**

删除指定key   delete 

```cmd
127.0.0.1:6379> del company

(integer) 1

127.0.0.1:6379> get company

(nil)
```

### 5.3    哈希类型hash

#### 5.3.1.    哈希类型hash概述

```
Redis中的Hash类型可以看成具有String Key和String Value的map容器。

所以该类型非常适合于存储值对象的信息。

如username、password和age等。

如果Hash中包含很少的字段,那么该类型的数据也将仅占用很少的磁盘空间。

每一个Hash可以存储4294967295个键值对。

```

![1531535474245](assets/1531535474245.png)

```
Key是用户ID, value是一个Map,这个Map的key是成员的属性名,value是属性值,这样对数据的修改和存取都可以直接通过其内部Map的Key(Redis里称内部Map的key为field), 
也就是通过 key(用户ID) + field(属性标签) 就可以操作对应属性数据了,既不需要重复存储数据,也不会带来序列化和并发修改控制的问题
```

#### 5.3.2.    哈希类型hash常用命令

 **使用场景:**

```
存储部分变更数据,如用户信息等。 
```

·       使用命令: **hset key field value**

为指定的key设定field/value对(键值对)。

```cmd
127.0.0.1:6379> hset myhash username haohao

(integer) 1

127.0.0.1:6379> hset myhash age 23
```

·        **hget key field**

返回指定的key中的field的值

```cmd
127.0.0.1:6379> hset myhash username haohao

(integer) 1

127.0.0.1:6379> hget myhash username

"haohao"
```

·        **hdel key field [field … ]** 

可以删除一个或多个字段,返回值是被删除的字段个数

```cmd
127.0.0.1:6379> hdel myhash email tps163.com

(integer) 1

127.0.0.1:6379> hget myhash password 123

(nil)

127.0.0.1:6379> hgetall myhash
1) "email"
2) "tps@163.com"
3) "password"
4) "123"

127.0.0.1:6379> hmset myhash sex mail job teacher
OK
//  hmset 设置多个key value数据值。
127.0.0.1:6379> hgetall myhash
 1) "email"
 2) "tps@163.com"
 3) "password"
 4) "123"
 5) "sex"
 6) "mail"
 7) "job"
 8) "teacher"
```

补充:  一次获取key的所有map的key和value:hgetall key

### 5.4.   列表类型list

**应用场景:**  

```
Redis list的应用场景非常多,也是Redis最重要的数据结构之一,比如twitter的关注列表,粉丝列表等都可以用Redis的list结构来实现。

List 就是链表,相信略有数据结构知识的人都应该能理解其结构。使用List结构,我们可以轻松地实现最新消息排行等功能。List的另一个应用就是消息队列,

可以利用List的PUSH操作,将任务存在List中,然后工作线程再用POP操作将任务取出进行执行。Redis还提供了操作List中某一段的api,你可以直接查询,删除List中某一段的元素。

```

#### 5.4.1.    列表类型list概述

![img](assets/clip_image016.jpg)

在Redis中,List类型是按照插入顺序排序的字符串链表。和数据结构中的普通链表一样,我们可以在其头部(left)和尾部(right)添加新的元素。

在插入时,如果该键不存在,Redis将为该键创建一个新的链表。

如果链表中所有的元素均被移除,那么该键也将会被从数据库中删除。

List中可以包含的最大元素数量是4294967295

#### 5.4.2.    列表类型list

·        **lpush key value1 value2 …**

在指定的key所关联的list的**头部插入所有的values**,

如果该key不存在,该命令在插入的之前创建一个与该key关联的空链表,之后再向该链表的头部插入数据。

插入成功,返回元素的个数。

```cmd
127.0.0.1:6379> lpush mylist a b c

(integer) 3

127.0.0.1:6379>
```

**·**        **rpush key value value value**

在指定的key对应的list的尾部插入所有的value,

如果该key不存在,该命令在插入之前创建一个与该key对应的空链表,再从尾部插入数据。

插入成功,返回元素的个数。

![img](assets/clip_image018.jpg)

·        **lpop key**

返回并弹出指定的key关联的链表中的第一个元素,即头部元素。

如果该key不存在,返回nil;

若key存在,则返回链表的头部元素。

```cmd
127.0.0.1:6379> lpush mylist a b c

(integer) 3

127.0.0.1:6379> lpop mylist

"c"

127.0.0.1:6379> lpop mylist

"b"
```

·        **rpop key**

从尾部弹出元素。

```cmd
127.0.0.1:6379> lpush mylist a b c

(integer) 3

127.0.0.1:6379> rpop mylist

"a"
```

### 5.5.   集合类型set

**应用场景:**

```
     Redis set对外提供的功能与list类似是一个列表的功能,
     特殊之处在于set是可以自动排重的,当你需要存储一个列表数据,又不希望出现重复数据时,
     set是一个很好的选择,并且set提供了判断某个成员是否在一个set集合内的重要接口,
     这个也是list所不能提供的。
```

#### 5.5.1.    集合类型set

在Redis中,我们可以将Set类型看作为没有排序的字符集合,和List类型一样,我们也可以在该类型的数据值上执行添加、删除或判断某一元素是否存在等操作。需要说明的是,这些操作的时间复杂度为O(1),即常量时间内完成次操作。Set可包含的最大元素数量是4294967295,和List类型不同的是,Set集合中不允许出现重复的元素。

#### 5.5.2.    集合类型set的常用命令

·        **sadd key values[value1****、****value2…]**

向set中添加数据,如果该key的值已有则不会重复添加

```cmd
127.0.0.1:6379> sadd myset a b c

(integer) 3
```

·        **smembers key**

获取set中所有的成员

```cmd
127.0.0.1:6379> sadd myset a b c

(integer) 3

127.0.0.1:6379> smembers myset

1) "c"

2) "a"

3) "b"
```

·        srem key members[member1、member2…]

删除set中指定的成员

```cmd
127.0.0.1:6379> srem myset a b

(integer) 2

127.0.0.1:6379> smembers myset

1) "c"

127.0.0.1:6379>


小结:
字符串   set   get    del  (json格式字符串)
hash    hset  hget   hgetall  hmset
list    lpush  rpush  lpop  rpop 
set     sadd  smemebers   srem

```

## 6.   Redis的通用命令

·        **keys pattern**

获取所有与pattern匹配的key,返回所有与该key匹配的keys。*表示任意一个或多个字符,?表示任意一个字符

```cmd
127.0.0.1:6379> keys *

1) "company"

2) "mylist"

3) "myhash"

4) "myset"
```

·        **del key1 key2…**

删除指定的key

```cmd
127.0.0.1:6379> del company

(integer) 1
```

·        **exists key**

判断该key是否存在,1代表存在,0代表不存在

```cmd
127.0.0.1:6379> exists compnay

(integer) 0

127.0.0.1:6379> exists mylist

(integer) 1

127.0.0.1:6379>
```

·        **type key**

获取指定key的类型。该命令将以字符串的格式返回。 返回的字符串为string、list、set、hash,如果key不存在返回none

```cmd
127.0.0.1:6379> type company

string

127.0.0.1:6379> type mylist

list

127.0.0.1:6379> type myset

set

127.0.0.1:6379> type myhash

hash

127.0.0.1:6379>
```

## 7.   Jedis的基本使用

### 7.1.   jedis的介绍

- jedis是官方首选的java客户端开发包

```java
Redis不仅是使用命令来操作,现在基本上主流的语言都有客户端支持,比如java、C、C#、C++、php、Node.js、Go等。 

在官方网站里列一些Java的客户端,有Jedis、Redisson、Jredis、JDBC-Redis、等其中官方推荐使用Jedis和Redisson。 

在企业中用的最多的就是Jedis,Jedis同样也是托管在github上,

地址:https://github.com/xetorthio/jedis。
下载jedis解压后得到jar包如下:java操作redis数据库API(Jedis)

```

### 7.2.   jedis的基本操作

#### 7.2.1.    jedis常用API

| **方法**              | **解释**                                                     |
| --------------------- | ------------------------------------------------------------ |
| new Jedis(host, port) | 创建jedis对象,参数host是redis服务器地址,参数port是redis服务端口 |
| set(key,value)        | 设置字符串类型的数据                                         |
| get(key)              | 获得字符串类型的数据                                         |
| hset(key,field,value) | 设置哈希类型的数据                                           |
| hget(key,field)       | 获得哈希类型的数据                                           |
| lpush(key,values)     | 设置列表类型的数据                                           |
| lpop(key)             | 列表左面弹栈                                                 |
| rpop(key)             | 列表右面弹栈                                                 |
| del(key)              | 删除指定的key                                                |

#### 7.2.2.    jedis的基本操作

导入开发包:

![1531536384483](assets/1531536384483.png)

```java
public void testJedisSingle(){

    //1 设置ip地址和端口

    Jedis jedis = new Jedis("localhost", 6379);

    //2 设置数据

         jedis.set("name", "itheima");
    
        jedis.psetex("tom",90000L,"黑马41");
        //  保存数据 持续90秒
        jedis.close();//  释放资源
        System.out.println("保存数据完成");

    //3 获得数据

    String name = jedis.get("name");

    System.out.println(name);

    //4 释放资源

    jedis.close();

}
```

### 7.3.   jedis连接池的使用

#### 7.3.1.          jedis连接池的基本概念

**jedis连接资源的创建与销毁是很消耗程序性能**,所以jedis为我们提供了jedis的**池化技术**,

jedisPool在创建时初始化一些连接资源存储到连接池中,使用jedis连接资源时不需要创建,

而是从连接池中获取一个资源进行redis的操作,使用完毕后,不需要销毁该jedis连接资源,

而是将该资源归还给连接池,供其他请求使用。

#### 7.3.2.    jedisPool的基本使用

```java
public void testJedisPool(){

    //1 获得连接池配置对象,设置配置项

    JedisPoolConfig config = new JedisPoolConfig();

    // 1.1 最大连接数

    config.setMaxTotal(30);

    // 1.2  最大空闲连接数

    config.setMaxIdle(10);

    

    //2 获得连接池

    JedisPool jedisPool = new JedisPool(config, "localhost", 6379);

    

    //3 获得核心对象

    Jedis jedis = null;

    try {

        jedis = jedisPool.getResource();

        

        //4 设置数据

        jedis.set("name", "itcast");

        //5 获得数据

        String name = jedis.get("name");

        System.out.println(name);

        

    } catch (Exception e) {

        e.printStackTrace();

    } finally{

        if(jedis != null){

            jedis.close();

        }

        // 虚拟机关闭时,释放pool资源

        if(jedisPool != null){

            jedisPool.close();

        }

    }

}
```

#### 7.4.   案例-编写jedis连接池工具类

**JedisUtils.java**

 ```java
package util;

import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;

import java.util.ResourceBundle;

/**
 * Jedis工具类
 */
public final class JedisUtil {
    private JedisUtil(){}
    private static JedisPool jedilPool;
    private static int maxtotal;
    private static int maxwaitmillis;
    private static String host;
    private static int port;;
    
    /**
     * 读取jedis.properties配置文件
     */
    static{
        ResourceBundle rb = ResourceBundle.getBundle("jedis");
        maxtotal = Integer.parseInt(rb.getString("maxtotal"));
        maxwaitmillis = Integer.parseInt(rb.getString("maxwaitmillis"));
        host = rb.getString("host");
        port = Integer.parseInt(rb.getString("port"));
    }
    /**
     * 创建连接池
     */
    static{
        JedisPoolConfig jedisPoolConfig = new JedisPoolConfig();
        jedisPoolConfig.setMaxTotal(maxtotal);
        jedisPoolConfig.setMaxWaitMillis(maxwaitmillis);
        jedilPool = new JedisPool(jedisPoolConfig,host,port);
    }
    /**
     * 获取Jedis
     */
    public static Jedis getJedis(){
        return jedilPool.getResource();
    }
    /**
     * 关闭Jedis
     */
    public static void close(Jedis jedis){
        if(jedis!=null){
            jedis.close();
        }
    }
}

 ```

xml  : 配置复杂数据结构  描述对象  解析XML文件,DOM for java   dom4j

properties  配置简单数据结构  key=value  java解析properties 文件数据 api简单

**jedis.properties**(src目录下配置文件,编写配置文件)

 ```properties
maxtotal=100
maxwaitmillis=3000
host=127.0.0.1
port=6379

 ```

如何快速读取properties配置文件信息

使用jdk提供 ResourceBundle加载 通过getBundle("文件名")   通过 getString("key")获取目标数据

src下添加Properties配置文件

![1531732250370](assets/1531732250370.png)

内容:

```properties
username=lisi
password=123
url=http://xxx
driverclass=com.mysql.jdbc.Driver

```

加载代码实现 : `ResourceBundle.java`使用说明

```java
package cn.itheima.jedis.demo;

import java.util.ResourceBundle;

public class PropertiesDemo {
    //  java  专属类   ResourceBundle 对象  jdk 提供
    public static void main(String[] args) {
        //  getBundle源码查询 src路路径下的properties文件 传递文件名默认  .properties 所有不需要写扩展名
        String s = ResourceBundle.getBundle("db").getString("driverclass");
        String url = ResourceBundle.getBundle("db").getString("url");
        String aa = ResourceBundle.getBundle("db").getString("username");
        String bb = ResourceBundle.getBundle("db").getString("password");
        System.out.println(s);
        System.out.println(url);
        System.out.println(aa);
        System.out.println(bb);
    }
}

```

## 8.     **案例:Redis实战之查询所有省份(重点)**

![1532859900897](assets/1532859900897.png)

需求:访问index.html页面,使用ajax请求加载省份列表,用户第一次访问数据库获取,以后都从redis里面获取。

 资料:准备省市区表和数据

![1531536905277](assets/1531536905277.png)

 表设计说明:

```sql
CREATE TABLE `pcd` (
  `id` int(11) NOT NULL,
  `pid` int(11) DEFAULT NULL,
  `name` varchar(10) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
```

![1531734227377](assets/1531734227377.png)

案例分析:

![1531734606495](assets/1531734606495.png)

代码实现:

1.  搭建项目环境

导入jar

mysql  c3p0    jdbcTemplate   redis /pool  / fastjson   

![1531734815074](assets/1531734815074.png)

配置文件 src

: c3p0  / jedis 连接池配置

![1531734916672](assets/1531734916672.png)

工具类: JDBCUtils   /    JedisUtils

![1531735030937](assets/1531735030937.png)

实体类

```java
package cn.itheima.jedis.domain;

public class City {

    private int id;
    private int pid;
    private String name;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public int getPid() {
        return pid;
    }

    public void setPid(int pid) {
        this.pid = pid;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

```

首页

```html
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<h2>省市区三级联动</h2>

省&nbsp;
<select name="province">
    <option value="">--请选择--</option>

</select>

市&nbsp;
<select name="city">
    <option value="">--请选择--</option>

</select>

区&nbsp;
<select name="area">
    <option value="">--请选择--</option>

</select>

</body>
</html>

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值