Redis和nosql简介,api调用;Redis数据功能(String类型的数据处理);List数据结构(及Java调用处理);Hash数据结构;Set数据结构功能;sortedSet(有序集合)数

本文涉及的产品
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
云数据库 Tair(兼容Redis),内存型 2GB
简介: 1、Redis和nosql简介,api调用14.1/ nosql介绍 NoSQL:一类新出现的数据库(not only sql),它的特点:1、  不支持SQL语法2、  存储结构跟传统关系型数据库中的那种关系表完全不同,nosql中存储的数据都是KV形式3、  NoSQL的世界中没有一种通用的语言,每种nosql数据库都有自己的api和语法,以及擅长的业务场景4、  NoSQ

1、Redis和nosql简介,api调用

14.1/ nosql介绍

 

NoSQL:一类新出现的数据库(not only sql),它的特点:

1、  不支持SQL语法

2、  存储结构跟传统关系型数据库中的那种关系表完全不同,nosql中存储的数据都是KV形式

3、  NoSQL的世界中没有一种通用的语言,每种nosql数据库都有自己的api和语法,以及擅长的业务场景

4、  NoSQL中的产品种类相当多:

a)        Mongodb  文档型nosql数据库,擅长做CMS系统(内容管理系统)

b)        Redis        内存数据库,数据结构服务器,号称瑞士军刀(精巧),只要你有足够的想象力,它可以还给你无限惊喜

c)        Hbase  hadoop生态系统中原生的一种nosql数据库,重量级的分布式nosql数据库,用于海量数据的场景

d)        Cassandra  hadoop生态系统中原生的一种分布式nosql数据库,后起之秀

 

NoSQL和SQL数据库的比较:

1、适用场景不同:sql数据库适合用于关系特别复杂的数据查询场景,nosql反之

2、“事务”特性的支持:sql对事务的支持非常完善,而nosql基本不支持事务

3、两者在不断地取长补短,呈现融合趋势

 

 

14.2/ redis介绍

14.2.1 简述

Redis是一个高性能的kv缓存和内存数据库(存的不像mysql那样的表)

Redis的存储结构就是key-value,形式如下:


注: redis中的value内部可以支持各种数据结构类型,比如可以存入一个普通的string,还可以存listsethashmapsortedSet(有序的set

 

14.2.2 redis应用场景

A、用来做缓存(ehcache/memcached)——redis的所有数据是放在内存中的(内存数据库)

B、可以在某些特定应用场景下替代传统数据库——比如社交类的应用

C、在一些大型系统中,巧妙地实现一些特定的功能:session共享、购物车

只要你有丰富的想象力,redis可以用在可以给你无限的惊喜…….

 

 

14.2.3 redis的特性

1、redis数据访问速度快(数据在内存中)

2、redis有数据持久化机制(持久化机制有两种:1、定期将内存数据dump到磁盘;2、aof(append only file)持久化机制——用记日志的方式记录每一条数据更新操作,一旦出现灾难事件,可以通过日志重放来恢复整个数据库)

3、redis支持集群模式(容量可以线性扩展)

4、redis相比其他缓存工具(ehcach/memcached),有一个鲜明的优势:支持丰富的数据结构

14.3.Redis的api客户端连接

新建一个maven工程,导入jedis的maven依赖坐标

<dependency>

<groupId>redis.clients</groupId>

<artifactId>jedis</artifactId>

<version>2.7.2</version>

<type>jar</type>

<scope>compile</scope>

</dependency>

然后写一个类用来测试服务器跟客户端的连通性:

public class RedisClientConnectionTest {

         public static void main(String[] args) {

                  // 构造一个redis的客户端对象

                  Jedis jedis = new Jedis("pinshutang.zicp.net", 6379);

                  String ping = jedis.ping();

                  System.out.println(ping);

         }

}

 

15、Redis数据功能(String类型的数据处理)

15.1 String类型的数据

(常作为缓存使用)

1/插入和读取一条string类型的数据

redis notrue-centos:6379> set sessionid-0001 "zhangsan"

OK

redis notrue-centos:6379> get sessionid-0001

"zhangsan"

 

2/对string类型数据进行增减(前提是这条数据的value可以看成数字)

192.168.106.80:6379> DECR key

(integer) -1

192.168.106.80:6379> DECR key

(integer) -2

192.168.106.80:6379> DECR key

(integer) -3

192.168.106.80:6379> INCR key

(integer) -2

192.168.106.80:6379> INCR key

(integer) -1

 

192.168.106.80:6379> DECRBY key 4

(integer) -5

192.168.106.80:6379> DECRBY key -7

(integer) 2

192.168.106.80:6379> INCRBY key 2

(integer) 4

192.168.106.80:6379> INCRBY key -9

(integer) -5

 

3/一次性插入或者获取多条数据

192.168.106.80:6379> MSET key1 1 key2 2 key3 3

OK

192.168.106.80:6379> MGET key1 key2 key3

1) "1"

2) "2"

3) "3"

 

4/在插入一条string类型数据的同时为它指定一个存活期限

setex key seconds value

 

例如:

# bancao这条数据就只会存活10秒钟,过期会被redis自动清除

192.168.106.80:6379> setex bancao 10 testvalue

OK

192.168.106.80:6379> get bancao

"testvalue"

192.168.106.80:6379> get bancao

"testvalue"

192.168.106.80:6379> get bancao

(nil)

192.168.106.80:6379> get bancao

(nil)

 

案例:

其中maven项目的pom文件的内容如下:

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">

    <modelVersion>4.0.0</modelVersion>

    <groupId>cn.toto.redis</groupId>

    <artifactId>redistest</artifactId>

    <version>0.0.1-SNAPSHOT</version>

 

    <dependencies>

 

       <!-- <dependency> <groupId>redis.clients</groupId> <artifactId>jedis</artifactId>

           <version>2.7.2</version> <type>jar</type> <scope>compile</scope> </dependency> -->

 

       <dependency>

           <groupId>redis.clients</groupId>

           <artifactId>jedis</artifactId>

           <version>2.8.0</version>

       </dependency>

 

       <dependency>

           <groupId>com.google.code.gson</groupId>

           <artifactId>gson</artifactId>

           <version>2.3.1</version>

       </dependency>

      

       <dependency>

           <groupId>junit</groupId>

           <artifactId>junit</artifactId>

           <version>4.7</version>

           <scope>test</scope>

       </dependency>

    </dependencies>

</project>

代码:

package cn.toto.redis.client;

 

import java.io.ByteArrayInputStream;

import java.io.ByteArrayOutputStream;

import java.io.ObjectInputStream;

import java.io.ObjectOutputStream;

 

import org.junit.Before;

import org.junit.Test;

 

import com.google.gson.Gson;

 

import cn.toto.redis.bean.ProductInfo;

 

import redis.clients.jedis.Jedis;

 

public class StringStructureData {

 

    private Jedis jedis = null;

 

    @Before

    public void init() {

       //这里的是hadoop2可以用ip地址替换

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

      

    }

   

    @Test

    public void testString() {

       jedis.set("user02:name", "ruhua");

       jedis.set("user03:name", "滑板鞋");

      

       String u02 = jedis.get("user02:name");

       String u03 = jedis.get("user03:name");

      

       System.out.println(u02);

       System.out.println(u03);

    }

   

    @Test

    public void testObjectCache() throws Exception {

       ProductInfo p = new ProductInfo();

      

       p.setName("username");

       p.setDescription("备注信息");

       p.setCatelog("unknow");

       p.setPrice(10.8);

      

       //将对象序列化成字节数组

       ByteArrayOutputStream ba = new ByteArrayOutputStream();

       ObjectOutputStream oos = new ObjectOutputStream(ba);

      

       //用对象序列化流来将p对象序列化,然后把序列化之后的二进制数据写到ba流中

       oos.writeObject(p);

      

       //ba流转成byte数组

       byte[] pBytes = ba.toByteArray();

      

       //将对象序列化之后的byte数组存到redisstring结构数组中

       jedis.set("product:01".getBytes(), pBytes);

      

       //根据keyredis中取出对象的byte数据

       byte[] pBytesResp = jedis.get("product:01".getBytes());

      

      

       //byte数据反序列化列出对象

       ByteArrayInputStream bi = new ByteArrayInputStream(pBytesResp);

       ObjectInputStream oi = new ObjectInputStream(bi);

      

       //从对象读取流中取出对象

       ProductInfo pResp = (ProductInfo) oi.readObject();

       System.out.println(pResp);

    }

   

    @Test

    public void testObjectToJsonCache() {

       ProductInfo p = new ProductInfo();

      

       p.setName("ABC");

       p.setDescription("刘亦菲专用");

       p.setCatelog("化妆品");

       p.setPrice(10.8);

      

       //利用gson将对象转成json

       Gson gson = new Gson();

       String pJson = gson.toJson(p);

      

       //json串存入redis

       jedis.set("prodcut:02", pJson);

      

       //redis中取出对象的json

       String pJsonResp = jedis.get("prodcut:02");

      

       //将返回的json解析成对象

       ProductInfo pResponse = gson.fromJson(pJsonResp, ProductInfo.class);

      

       //显示对象的属性

       System.out.println(pResponse);

    }

}

 

16、List数据结构(及Java调用处理)

16.1 List图示

 

16.2 List功能演示

#从头部(左边)插入数据

redis>LPUSH  key  value1  value2  value3   

#从尾部(右边)插入数据

redis>RPUSH  key  value1  value2  value3  

#读取list中指定范围的values

redis>LRANGE  key  start   end

redis> lrange  task-queue  0  -1      读取整个list

#从头部弹出一个元素

LPOP  key

#从尾部弹出一个元素

RPOP  key

#从一个list的尾部弹出一个元素插入到另一个list

RPOPLPUSH   key1    key2     ## 这是一个原子性操作

例如下面的案例:

#从头部(左边)插入数据

192.168.106.81:6379> lpush task task0 task1 task2

(integer) 3

192.168.106.81:6379> lrange task 0 2

1) "task2"

2) "task1"

3) "task0"

192.168.106.81:6379> lrange task 0 -2

1) "task2"

2) "task1"

192.168.106.81:6379> lrange task 0 -1

1) "task2"

2) "task1"

3) "task0"

#从尾部(右边)插入数据

192.168.106.81:6379> rpush ids id0 id1 id2

(integer) 3

192.168.106.81:6379> lrange ids 0 -1

1) "id0"

2) "id1"

3) "id2"

192.168.106.81:6379> lpop ids

"id0"

192.168.106.81:6379> lpop ids

"id1"

192.168.106.81:6379> rpop task

"task0"

192.168.106.81:6379> rpoplpush task ids

"task1"

192.168.106.81:6379> lrange ids 0 -1

1) "task1"

2) "id2"

192.168.106.81:6379>

 

16.3 List的应用案例demo

1 需求描述

任务调度系统:

生产者不断产生任务,放入task-queue排队

消费者不断拿出任务来处理,同时放入一个tmp-queue暂存,如果任务处理成功,则清除tmp-queue,否则,将任务弹回task-queue

 

2 代码实现

1/生产者

——模拟产生任务

package cn.toto.redis.productAndCustomer;

 

import java.util.Random;

import java.util.UUID;

 

import redis.clients.jedis.Jedis;

 

public class TaskProducer {

    //获取一个redis的客户端连接对象

    public static Jedis getRedisConnection(String host,int port) {

       Jedis jedis = new Jedis(host,port);

       return jedis;

    }

   

    public static void main(String[] args) {

       Jedis jedis = getRedisConnection("hadoop3", 6379);

      

       Random random = new Random();

       //生成任务

       while(true) {

           try {

              //生成任务的速度有一定的随机性,在1-2秒之间

              Thread.sleep(random.nextInt(1000) + 1000);

              //生成一个任务

              String taskid = UUID.randomUUID().toString();

              //往任务队列"task-queue"中插入,第一次插入时,"task-queue"还不存在

              //但是lpush方法会在redis库中创建一条新的list数据

              jedis.lpush("task-queue", taskid);

              System.out.println("向任务队列中插入了一个新的任务:" + taskid);

           } catch (Exception e) {

              e.printStackTrace();

           }

       }

    }

}

 

2/消费者

——模拟处理任务,并且管理暂存队列

package cn.toto.redis.productAndCustomer;

 

import java.util.Random;

 

import redis.clients.jedis.Jedis;

 

public class TaskConsumer {

 

    public static void main(String[] args) {

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

       Random random = new Random();

       

       while(true) {

           try {

              //task-queue中取一个任务,同时放入tmp-queue

              String taskid = jedis.rpoplpush("task-queue", "tmp-queue");

             

              //模拟处理任务

              Thread.sleep(1000);

             

              //模拟有成功又有失败的情况

              int nextInt = random.nextInt(13);

              if (nextInt % 7 == 0) {//模拟失败的情况

                  //在失败的情况下,需要将任务从tmp-queue弹回"task-queue"

                  jedis.rpoplpush("tmp-queue", "task-queue");

                  System.out.println("----任务处理失败:" + taskid);

              } else {//模拟成功的情况

                  //成功的情况下,将任务从"tmp-queue"清除

                  jedis.rpop("tmp-queue");

                  System.out.println("任务处理成功:" + taskid);

              }

           } catch (Exception e) {

              e.printStackTrace();

           }

       }

    }

}

 

上述机制是一个简化版,真实版的任务调度系统会更加复杂,如下所示:

(增加了一个专门用来管理暂存队列的角色,以便就算消费者程序失败退出,那些处理失败的任务依然可以被弹回task-queue)

 

17、Hash数据结构

17.1 Hash图示

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

 

17.2 Hash功能演示

1、往redis库中插入一条hash类型的数据

redis> HSET  key  field  value

举例:

redis 127.0.0.1:6379> hset user001:zhangsan  iphone   6

(integer) 1

redis 127.0.0.1:6379> hset user001:zhangsan   xiaomi   7

(integer) 1

redis 127.0.0.1:6379> hset user001:zhangsan   meizu   8

(integer) 1

在redis库中就形成了这样一条数据:

 

2、从redis库中获取一条hash类型数据的value

ü 取出一条hash类型数据中所有field-value对

redis 127.0.0.1:6379> hgetall user001:zhangsan

1) "iphone"

2) "6"

3) "xiaomi"

4) "7"

5) "meizu"

6) "8"

 

ü 取出hash数据中所有fields

redis 127.0.0.1:6379> HKEYS user001:zhangsan

1) "iphone"

2) "xiaomi"

3) "meizu"

 

ü 取出hash数据中所有的value

redis 127.0.0.1:6379> hvals user001:zhangsan

1) "6"

2) "7"

3) "8"

 

ü 取出hash数据中一个指定field的值

redis 127.0.0.1:6379> hget user001:zhangsan xiaomi

"8"

 

ü 为hash数据中指定的一个field的值进行增减

redis 127.0.0.1:6379> HINCRBY user001:zhangsan xiaomi 1

(integer) 8

 

ü 从hash数据中删除一个字段field及其值

redis 127.0.0.1:6379> hgetall user001:zhangsan

1) "iphone"

2) "6"

3) "xiaomi"

4) "7"

5) "meizu"

6) "8"

redis 127.0.0.1:6379> HDEL user001:zhangsan iphone

(integer) 1

redis 127.0.0.1:6379> hgetall user001:zhangsan

1) "xiaomi"

2) "7"

3) "meizu"

4) "8"

 

18、Set数据结构功能

集合的特点:无序、无重复元素

1、  插入一条set数据

redis 127.0.0.1:6379> sadd frieds:zhangsan  bingbing baby fengjie furong ruhua tingting

(integer) 6

redis 127.0.0.1:6379> scard frieds:zhangsan

(integer) 6

redis 127.0.0.1:6379>

 

 

2、获取一条set数据的所有members

redis 127.0.0.1:6379> smembers frieds:zhangsan

1) "fengjie"

2) "baby"

3) "furong"

4) "bingbing"

5) "tingting"

6) "ruhua"

 

3、判断一个成员是否属于某条指定的set数据

redis 127.0.0.1:6379> sismember frieds:zhangsan liuyifei     #如果不是,则返回0

(integer) 0

redis 127.0.0.1:6379> sismember frieds:zhangsan baby       #如果是,则返回1

(integer) 1

 

4、求两个set数据的差集

#求差集

redis 127.0.0.1:6379> sdiff  frieds:zhangsan  friends:xiaotao

1) "furong"

2) "fengjie"

3) "ruhua"

4) "feifei"

#求差集,并将结果存入到另一个set

redis 127.0.0.1:6379> sdiffstore zhangsan-xiaotao frieds:zhangsan friends:xiaotao

(integer) 4

#查看差集结果

redis 127.0.0.1:6379> smembers zhangsan-xiaotao

1) "furong"

2) "fengjie"

3) "ruhua"

4) "feifei"

 

5、  求交集,求并集

#求交集

redis 127.0.0.1:6379> sinterstore zhangsan:xiaotao frieds:zhangsan friends:xiaotao

(integer) 2

redis 127.0.0.1:6379> smembers zhangsan:xiaotao

1) "bingbing"

2) "baby"

 

#求并集

redis 127.0.0.1:6379> sunion  frieds:zhangsan friends:xiaotao

 1) "fengjie"

 2) "tangwei"

 3) "liuyifei"

 4) "bingbing"

 5) "ruhua"

 6) "feifei"

 7) "baby"

 8) "songhuiqiao"

 9) "furong"

10) "yangmi"

案例代码:

package cn.toto.redis.hashcart;

 

import java.util.Map;

import java.util.Map.Entry;

import java.util.Set;

 

import org.junit.Before;

import org.junit.Test;

 

import redis.clients.jedis.Jedis;

 

public class BuyCartServiceImpl {

   

    private Jedis jedis = null;

    private static final String CART_PRIFIX = "cart:";

   

    @Before

    public void init() {

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

    }

   

    /**

     * 添加商品到购物车

     */

    @Test

    public void testAddItemToCart() {

       jedis.hset(CART_PRIFIX + "user02", "上衣", "2");

       jedis.hset(CART_PRIFIX + "user02", "鞋子", "1");

       jedis.hset(CART_PRIFIX + "user02", "毛衣", "3");

       jedis.hset(CART_PRIFIX + "user02", "羽绒服", "4");

      

       jedis.close();

    }

   

    /**

     * 查看购物车信息,当本类中的其它单元测试执行完成之后,可以通过执行这个单元测试方法查看内容

     * @author toto

     * @date 2017-1-14

     * @note  begin modify by 涂作权 2017-1-14  原始创建

     */

    @Test

    public void testGetCartInfo() {

       Map<String, String> cart = jedis.hgetAll(CART_PRIFIX + "user02");

       Set<Entry<String, String>> entrySet = cart.entrySet();

       for(Entry<String, String> ent : entrySet) {

           System.out.println(ent.getKey() + ":" + ent.getValue());

       }

       jedis.close();

    }

   

    /**

     * 编辑缓存中的内容

     */

    @Test

    public void editCart() {

       //给蜡烛商品项的数量加1

       jedis.hincrBy(CART_PRIFIX + "user02", "羽绒服", 1);

       jedis.close();

    }

   

    /**

     * 删除内存中上衣这个内容

     */

    @Test

    public void delItemFromCart() {

       jedis.hdel(CART_PRIFIX + "user02", "上衣");

       jedis.close();

    }

}

生产者消费者:

生产者:

package cn.toto.redis.listqueue;

 

import java.util.Random;

import java.util.UUID;

 

import redis.clients.jedis.Jedis;

 

public class TaskProducer {

    private static Jedis jedis = null;

   

    public static void main(String[] args) throws Exception {

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

       Random random = new Random();

      

       while(true) {

           int nextInt = random.nextInt(1000);

           Thread.sleep(1000 + nextInt);

          

           //生成一个任务的id

           String taskid = UUID.randomUUID().toString();

          

           jedis.lpush("task-queue", taskid);

          

           System.out.println("生成了一个任务:" + taskid);

       }

    }

}

消费者:

package cn.toto.redis.listqueue;

 

import java.util.Random;

 

import redis.clients.jedis.Jedis;

 

/**

 * @brief TaskProcessor.java 任务处理模块

 */

public class TaskProcessor {

   

    private static Jedis jedis = null;

   

    public static void main(String[] args) throws Exception {

       Random random = new Random();

      

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

      

       while(true) {

           Thread.sleep(1500);

          

           //从任务队列中取出一个任务,同时放入到暂存队列中

           String taskid = jedis.rpoplpush("task-queue", "tmp-queue");

          

           //处理任务

           if (random.nextInt(19) % 9 == 0) {

              //模拟失败

              //失败的情况下,需要将任务从暂存队列弹回任务队列

              jedis.rpoplpush("tmp-queue", "task-queue");

              System.out.println("该任务处理失败:" + taskid);

           } else {

              //模拟成功

              //成功的情况下,只需要将任务从暂存队列清除

              jedis.rpop("tmp-queue");

              System.out.println("任务处理成功:" + taskid);

           }

       }

    }

}

Set的案例:

package cn.toto.redis.set;

 

import java.util.Set;

 

import redis.clients.jedis.Jedis;

 

public class TestSet {

   

    public static void main(String[] args) {

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

       

        jedis.sadd("friends:shuangshuang", "dandan","lulu","lili");

        jedis.sadd("friends:laobi", "laowang","laodu","laoli","lili","lulu");

       

        //判断一个成员是否属于指定的集合

        Boolean isornot = jedis.sismember("friends:laobi", "shuangshuang");

        System.out.println(isornot);

       

        //求两个集合的差并交集

        Set<String> ssDiffbb = jedis.sdiff("friends:shuangshuang","friends:laobi");

       Set<String> ssUnionbb = jedis.sunion("friends:shuangshuang","friends:laobi");

       Set<String> ssInterbb = jedis.sinter("friends:shuangshuang","friends:laobi");

      

       //打印结果

       for (String mb : ssUnionbb) {

           System.out.println(mb);

       }

    }

}

 

 

19、sortedSet(有序集合)数据结构

19.1 sortedSet图示

sortedset中存储的成员都有一个附带的分数值

而redis就可以根据分数来对成员进行各种排序(正序、倒序)

 

1、  sortedSet存储内容示意图:

 

19.2 SortedSet功能演示

1、往redis库中插入一条sortedset数据

redis 127.0.0.1:6379> zadd nansheng:yanzhi:bang  70 liudehua  90 huangbo  100 weixiaobao  250 yangwei  59 xiaotao

(integer) 5

 

 

2、  从sortedset中查询有序结果

#正序结果

redis 127.0.0.1:6379> zrange nansheng:yanzhi:bang  0  4

1) "xiaotao"

2) "liudehua"

3) "huangbo"

4) "weixiaobao"

5) "yangwei"

#倒序结果

redis 127.0.0.1:6379> zrevrange nanshen:yanzhi:bang 0 4

1) "yangwei"

2) "weixiaobao"

3) "huangbo"

4) "liudehua"

5) "xiaotao"

 

3、  查询某个成员的名次

#在正序榜中的名次

redis 127.0.0.1:6379> zrank nansheng:yanzhi:bang  xiaotao

(integer) 0

 

#在倒序榜中的名次

redis 127.0.0.1:6379> zrevrank nansheng:yanzhi:bang xiaotao

(integer) 4

 

4、修改成员的分数

redis 127.0.0.1:6379> zincrby nansheng:yanzhi:bang  300  xiaotao

"359"

redis 127.0.0.1:6379> zrevrank nansheng:yanzhi:bang xiaotao

(integer) 0

案例:

下面的这个先运行:

package cn.toto.redis.sortedset;

 

import java.util.Random;

 

import redis.clients.jedis.Jedis;

 

public class LolBoxPlayer {

   

    public static void main(String[] args) throws Exception {

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

      

       Random random = new Random();

       String[] heros = {"易大师","德邦","剑姬","盖伦","阿卡丽","金克斯","提莫","猴子","亚索"};

       while(true) {

           int index = random.nextInt(heros.length);

           //选择一个英雄

           String hero = heros[index];

          

           //开始玩游戏

           Thread.sleep(1000);

          

           //给集合中的该英雄的出场次数加1

           //第一次添加的时候,集合不存在,zincrby方法会创建

           jedis.zincrby("hero:ccl:phb", 1, hero);

          

           System.out.println(hero + "出场了......");

       }

    }

}

下面的这个后运行:

package cn.toto.redis.sortedset;

 

import java.util.Set;

 

import redis.clients.jedis.Jedis;

import redis.clients.jedis.Tuple;

 

public class LolBoxViewer {

   

    public static void main(String[] args) throws Exception {

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

      

       int i = 1;

       while(true) {

           //每隔3秒查看一次榜单

           Thread.sleep(3000);

           System.out.println("" + i + "次查看榜单-----------");

          

           //redis中查询榜单的前N

           Set<Tuple> topHeros = jedis.zrevrangeWithScores("hero:ccl:phb", 0, 4);

          

           for (Tuple t : topHeros) {

              System.out.println(t.getElement() + "  " + t.getScore());

           }

          

           i++;

          

           System.out.println("");

           System.out.println("");

           System.out.println("");

       }

    }

}

相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore &nbsp; &nbsp; ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库&nbsp;ECS 实例和一台目标数据库&nbsp;RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&amp;RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
目录
相关文章
|
2月前
|
Java API Maven
如何使用Java开发抖音API接口?
在数字化时代,社交媒体平台如抖音成为生活的重要部分。本文详细介绍了如何用Java开发抖音API接口,从创建开发者账号、申请API权限、准备开发环境,到编写代码、测试运行及注意事项,全面覆盖了整个开发流程。
337 10
|
2月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
107 2
|
9天前
|
JSON Java 数据挖掘
利用 Java 代码获取淘宝关键字 API 接口
在数字化商业时代,精准把握市场动态与消费者需求是企业成功的关键。淘宝作为中国最大的电商平台之一,其海量数据中蕴含丰富的商业洞察。本文介绍如何通过Java代码高效、合规地获取淘宝关键字API接口数据,帮助商家优化产品布局、制定营销策略。主要内容包括: 1. **淘宝关键字API的价值**:洞察用户需求、优化产品标题与详情、制定营销策略。 2. **获取API接口的步骤**:注册账号、申请权限、搭建Java开发环境、编写调用代码、解析响应数据。 3. **注意事项**:遵守法律法规与平台规则,处理API调用限制。 通过这些步骤,商家可以在激烈的市场竞争中脱颖而出。
|
25天前
|
JSON Java Apache
Java基础-常用API-Object类
继承是面向对象编程的重要特性,允许从已有类派生新类。Java采用单继承机制,默认所有类继承自Object类。Object类提供了多个常用方法,如`clone()`用于复制对象,`equals()`判断对象是否相等,`hashCode()`计算哈希码,`toString()`返回对象的字符串表示,`wait()`、`notify()`和`notifyAll()`用于线程同步,`finalize()`在对象被垃圾回收时调用。掌握这些方法有助于更好地理解和使用Java中的对象行为。
|
1月前
|
算法 Java API
如何使用Java开发获得淘宝商品描述API接口?
本文详细介绍如何使用Java开发调用淘宝商品描述API接口,涵盖从注册淘宝开放平台账号、阅读平台规则、创建应用并申请接口权限,到安装开发工具、配置开发环境、获取访问令牌,以及具体的Java代码实现和注意事项。通过遵循这些步骤,开发者可以高效地获取商品详情、描述及图片等信息,为项目和业务增添价值。
83 10
|
1月前
|
存储 Java 数据挖掘
Java 8 新特性之 Stream API:函数式编程风格的数据处理范式
Java 8 引入的 Stream API 提供了一种新的数据处理方式,支持函数式编程风格,能够高效、简洁地处理集合数据,实现过滤、映射、聚合等操作。
77 6
|
1月前
|
Java API 开发者
Java中的Lambda表达式与Stream API的协同作用
在本文中,我们将探讨Java 8引入的Lambda表达式和Stream API如何改变我们处理集合和数组的方式。Lambda表达式提供了一种简洁的方法来表达代码块,而Stream API则允许我们对数据流进行高级操作,如过滤、映射和归约。通过结合使用这两种技术,我们可以以声明式的方式编写更简洁、更易于理解和维护的代码。本文将介绍Lambda表达式和Stream API的基本概念,并通过示例展示它们在实际项目中的应用。
|
2月前
|
安全 Java API
告别SimpleDateFormat:Java 8日期时间API的最佳实践
在Java开发中,处理日期和时间是一个基本而重要的任务。传统的`SimpleDateFormat`类因其简单易用而被广泛采用,但它存在一些潜在的问题,尤其是在多线程环境下。本文将探讨`SimpleDateFormat`的局限性,并介绍Java 8引入的新的日期时间API,以及如何使用这些新工具来避免潜在的风险。
45 5
|
2月前
|
开发框架 Java 关系型数据库
Java哪个框架适合开发API接口?
在快速发展的软件开发领域,API接口连接了不同的系统和服务。Java作为成熟的编程语言,其生态系统中出现了许多API开发框架。Magic-API因其独特优势和强大功能,成为Java开发者优选的API开发框架。本文将从核心优势、实际应用价值及未来展望等方面,深入探讨Magic-API为何值得选择。
87 2
|
2月前
|
缓存 监控 Java
如何运用JAVA开发API接口?
本文详细介绍了如何使用Java开发API接口,涵盖创建、实现、测试和部署接口的关键步骤。同时,讨论了接口的安全性设计和设计原则,帮助开发者构建高效、安全、易于维护的API接口。
242 4