范文健康探索娱乐情感热点
投稿投诉
热点动态
科技财经
情感日志
励志美文
娱乐时尚
游戏搞笑
探索旅游
历史星座
健康养生
美丽育儿
范文作文
教案论文

Java基于springredis集成SpringBoot操作Redis

  本文章为系列文章,如需系统观看请按照文章顺序阅读,如有所了解,解决针对性问题,可直接阅读,谢谢支持!前言
  Java操作Redis有很多种方式,主要有Jedis,redisson,lettuce等还有Spring家族的spring-data-redisJedis
  是Redis的Java实现客户端,提供了比较全面的Redis命令的支持Redisson
  实现了分布式和可扩展的Java数据结构,提供很多分布式相关操作服务,例如,分布式锁,分布式集合,延迟队列Lettuce
  用于线程安全同步,异步和响应使用,支持集群,Sentinel,管道和编码器,主要在一些分布式缓存框架上使用比较多spring-data-redisSpring-Data-Redis项目(简称SDR)对Redis的Key-Value数据存储操作提供了更高层次的抽象类似于Spring Framework对JDBC支持一样项目主页: http://projects.spring.io/spring-data-redis/项目文档: http://docs.spring.io/spring-data/redis/docsSpring Data Redis使得在Spring应用中读写Redis数据库更加容易。Spring Data Redis提供了四种Redis服务的Java客户端包的集成,分别是 Jedis ,JRedis , SRP and Lettuce
  一般我们都会选择 spring-data-redis 核心对象RedisConnectionFactory
  redis的连接工厂类,根据Redis配置连接上Redis服务RedisTemplate
  熟悉Spring操作JDBC都知道JdbcTemplate作用,这个类也一样,spring 封装了 RedisTemplate 对象来进行对redis的各种操作,它支持所有的 redis 原生的 apiOperation
  针对Jedis客户端中大量api进行了归类封装,将同一类型操作封装为operation接口ValueOperations:String类型数据操作HashOperations:针对map类型的数据操作ListOperations:针对list类型的数据操作SetOperations:set类型数据操作ZSetOperations:zset类型数据操作BoundKeyOperations
  提供了对key的bound(绑定)便捷化操作API,可以通过bound封装指定的key,然后进行一系列的操作而无须"显式"的再次指定KeyBoundValueOperations:Redis String/Value key 约束BoundHashOperations:Redis Hash key 约束BoundListOperations:Redis List key 约束BoundSetOperations:Redis Set key 约束BoundZSetOperations:Redis Sort Set key 约束RedisTemplate序列化问题介绍
  当数据存储到Redis时,键(key)和值(value)都是通过Spring提供的Serializer序列化到数据库的。RedisTemplate默认使用的是JdkSerializationRedisSerializer ,StringRedisTemplate默认使用的是StringRedisSerializer 问题
  spring-data-redis中redisTemplate默认使用JDK的序列化策略,会出现两个问题使用redis-cli或者可视化工具查看数据时,不管是key还是value头部都会携带很多特殊字符,不易查看 ,Java代码中获取数据时不受影响,但是对运维来说就不友好了JDK Serializer太费资源解决使用Jackson serializer 替代JDK SerializerStringRedisSerializer结合GenericJackson2JsonRedisSerializerspring-data-redis支持的序列化策略
  spring-data-redis默认采用的序列化策略有两种,一种是String的序列化策略,一种是JDK的序列化策略GenericToStringSerializer:可以将任何对象泛化为字符串并序列化Jackson2JsonRedisSerializer:跟JacksonJsonRedisSerializer实际上是一样的JacksonJsonRedisSerializer:序列化object对象为json字符串JdkSerializationRedisSerializer:序列化java对象(被序列化的对象必须实现Serializable接口)StringRedisSerializer:简单的字符串序列化GenericToStringSerializer:类似StringRedisSerializer的字符串序列化GenericJackson2JsonRedisSerializer:类似Jackson2JsonRedisSerializer,但使用时构造函数不用特定的类
  key和hashKey:推荐使用 StringRedisSerializer: 简单的字符串序列化
  value和hashValue:推荐使用Jackson2JsonRedisSerializerRedisTemplate和StringRedisTemplate的区别两者的关系是StringRedisTemplate继承RedisTemplate。两者的数据是不共通的;也就是说StringRedisTemplate只能管理StringRedisTemplate里面的数据,RedisTemplate只能管理RedisTemplate中的数据SDR默认采用的序列化策略有两种,一种是String的序列化策略,一种是JDK的序列化策略StringRedisTemplate默认采用的是String的序列化策略,保存的key和value都是采用此策略序列化保存RedisTemplate默认采用的是JDK的序列化策略,保存的key和value都是采用此策略序列化保存
  当你的redis数据库里面本来存的是字符串数据或者你要存取的数据就是字符串类型数据的时候,那么你就使用StringRedisTemplate即可,但是如果你的数据是复杂的对象类型,而取出的时候又不想做任何的数据转换,直接从Redis里面取出一个对象,那么使用RedisTemplate是更好的选择开搞开发环境
  SpringBoot2.1.5、Redis5.0.5、Maven3.5.0、mybatis-Plus3.2.0pom依赖       org.springframework.boot     spring-boot-starter-web         org.springframework.boot     spring-boot-starter-data-redis       com.alibaba     fastjson     1.2.7        org.springframework.boot     spring-boot-starter-test     test application.yml文件spring:   #Redis配置   redis:     host: test201     port: 6379     password:     #这些都有默认配置,如果没有特殊需求可不写     #连接超时时间(2.0中该参数的类型为Duration,这里在配置的时候需要指明单位     timeout: 2000ms     #连接池最大连接数(使用负值表示没有限制)     max-active: 8     #连接池最大阻塞等待时间(使用负值表示没有限制)     max-wait: -1ms     #连接池中的最大空闲连接     max-idle: 8     #连接池中的最小空闲连接     min-idle: 0RedisConfig配置类
  上边咱们说默认的RedisTemplate使用的序列化方式是JDK的,会存在特殊字符的问题而且性能损耗较大,我们创建Java Config修改默认实现的RedisTemplate序列化方式import com.fasterxml.jackson.annotation.JsonAutoDetect; import com.fasterxml.jackson.annotation.PropertyAccessor; import com.fasterxml.jackson.databind.ObjectMapper; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.data.redis.connection.RedisConnectionFactory; import org.springframework.data.redis.core.RedisTemplate; import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer; import org.springframework.data.redis.serializer.StringRedisSerializer;  /**  * Redis的配置类  * SDR项目操作Redis的话需要使用RedisTemplate对象  * 但是该对象默认使用的数据序列化方法是JDK的,可能会存在特殊字符  *  * key和hashKey 我们推荐使用String序列化  * value 我们推荐是JSON存储,使用Json序列化  */ //添加Configuration注解,否则该类不会被加载 @Configuration public class RedisConfig {     /**      * @Bean:创建对象放到IOC容器中      * RedisConnectionFactory:Redis的连接工厂,根据application.yml文件中的Redis的配置做Redis连接和连接池的管理      该对象在项目初始化时被创建,有一个实现类是  LettuceConnectionFactory       */     @Bean     public RedisTemplate redisTemplate(RedisConnectionFactory connectionFactory){         //创建原生的RedisTemplate对象         RedisTemplate redisTemplate = new RedisTemplate();         //设置连接工厂         redisTemplate.setConnectionFactory(connectionFactory);         // 使用Jackson2JsonRedisSerialize替换 默认的JDK序列化         Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);         //创建JSON序列化的对象,value使用JSON序列化方式         ObjectMapper om = new ObjectMapper();         //设置序列化的策略         om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);         //启用默认的类型         om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);         //引用JSON序列化         jackson2JsonRedisSerializer.setObjectMapper(om);         //创建String序列化,我们的key使用String序列化方式         StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();         // key采用String的序列化方式         redisTemplate.setKeySerializer(stringRedisSerializer);         // hash的key也采用String的序列化方式         redisTemplate.setHashKeySerializer(stringRedisSerializer);         // value序列化方式采用JSON         redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);         // hash的value序列化方式采用JSON         redisTemplate.setHashValueSerializer(jackson2JsonRedisSerializer);         //使用该配置         redisTemplate.afterPropertiesSet();         //返回修改后的RedisTemplate对象         return redisTemplate;     } }测试类package com.tian.redis;  import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.data.redis.core.*; import org.springframework.test.context.junit4.SpringRunner; import java.util.*; import java.util.concurrent.TimeUnit;  @RunWith(SpringRunner.class) @SpringBootTest public class Day0512RedisApplicationTests {      @Test     public void contextLoads() {     }      //注入RedisTemplate     @Autowired     private RedisTemplate redisTemplate;      /**      * 操作String      */     @Test     public void test1(){         //获取操作String"类型的对象         ValueOperations valueOperations = redisTemplate.opsForValue();         //存储数据         try {             valueOperations.set("user1","添哥");             valueOperations.set("user2","小添哥哥");             System.out.println("存储数据成功");             //获取数据             String user1 = (String)valueOperations.get("user1");             String user2 = (String)valueOperations.get("user2");             System.out.println(user1 + "==>" + user2);         }catch (Exception e){             e.printStackTrace();         }     }     /**      * 操作list      */     @Test     public void test2(){          ListOperations operations = redisTemplate.opsForList();          Long num = operations.leftPushAll("l1", "l11", "l22", "l33", "l44");         System.out.println("插入数据个数===》" + num);         //取值         List list = operations.range("l1", 0, -1);         for (Object o : list) {             System.out.println("list---》"+o);         }      }     /**      * 操作set      */     @Test     public void test3(){          SetOperations operations = redisTemplate.opsForSet();          Long num = operations.add("s1", "s11", "s22", "s33", "s44", "s11", "s22", "s55");         System.out.println(num);         //获取         Set set = operations.members("s1");         for (Object o : set) {             System.out.println(o);         }     }      /**      * 操作zset      */     @Test     public void test4(){         ZSetOperations operations = redisTemplate.opsForZSet();         operations.add("z1","z10",100.00);         operations.add("z1","z20",200.00);         operations.add("z1","z5",50.00);         //获取值         Set set = operations.range("z1", 0, -1);         for (Object o : set) {             System.out.println(o);         }     }      /**      * 操作Hash      */     @Test     public void test5(){         HashOperations operations = redisTemplate.opsForHash();         //存储学生1         operations.put("stu:1:info","name","小添");         operations.put("stu:1:info","classRoom","快刀班");         //存储学生2         HashMap map = new HashMap<>();         //添加学生信息         map.put("name","晓添");         map.put("classRoom","吃瓜班");         operations.putAll("stu:2:info",map);          //获取数据1         Object name = operations.get("stu:1:info","name");         Object classRoom = operations.get("stu:1:info","classRoom");         System.out.println("学生1==》"+name+"==>"+classRoom);         //获取数据2         Map entries = operations.entries("stu:2:info");         //遍历map         Set set = entries.keySet();         for (Object o : set) {             System.out.println(o + "==>" + entries.get(o));         }     }      /**      * 对key的操作      */     @Test     public void test6(){         //判断key是否存在         Boolean hasKey = redisTemplate.hasKey("stu:2:info");         System.out.println("stu:2:info===>"+hasKey);         //设置过期时间,k3键10秒过期         redisTemplate.expire("k3",10, TimeUnit.SECONDS);         //设置l1键在2020 5 13 7点过期         redisTemplate.expireAt("l1",new Date(2020,5,13,7,00,00));     } } //下方两个测试是使用RedisUtil中封装使用boundHashOps方式操作 @Test public void t1(){    Student student = new Student();    student.setId("100");    student.setName("石靓仔");    student.setSex("男");    student.setBirth("1995-06-26");    redisUtil.putSome(Student.class,"student",student.getId(),student); } @Test public void t2(){    List studentList = new ArrayList<>();    Cursor> student = redisUtil.getSome("student", Student.class);    while (student.hasNext()){        System.out.println(student.getCursorId());        Map.Entry next = student.next();        Student student1 = (Student) next.getValue();        studentList.add(student1);    }    for (Student student1 : studentList) {        System.out.println(student1);    } }RedisUtil类
  封装一些常用的操作,大家可以根据自己的需求进行封装,这个作为参考吧package com.tian.springbootredis.util;  import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis.core.RedisTemplate; import org.springframework.stereotype.Component; import org.springframework.util.CollectionUtils; import java.util.List; import java.util.Map; import java.util.Set; import java.util.concurrent.TimeUnit; /**  * ClassName: RedisUtils  * Description:  * date: 2019/11/11 0011 下午 19:59  *  * @author stt  * @since JDK 1.8  */ @Component public class RedisUtil {      @Autowired     public RedisTemplate redisTemplate;      // =============================common============================     /**      26      * 指定缓存失效时间      27      * @param key 键     28      * @param time 时间(秒)     29      * @return     30      */     public boolean expire(String key, long time) {         try {             if (time > 0) {                 redisTemplate.expire(key, time, TimeUnit.SECONDS);             }             return true;         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 根据key 获取过期时间      * @param key 键 不能为null      * @return 时间(秒) 返回0代表为永久有效      */     public long getExpire(String key) {         return redisTemplate.getExpire(key, TimeUnit.SECONDS);     }     /**      * 判断key是否存在      * @param key 键      * @return true 存在 false不存在      */     public boolean hasKey(String key) {         try {             return redisTemplate.hasKey(key);         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 删除缓存      * @param key 可以传一个值 或多个      */     @SuppressWarnings("unchecked")     public boolean del(String... key) {         if (key != null && key.length > 0) {             if (key.length == 1) {                 return redisTemplate.delete(key[0]);             }             return redisTemplate.delete(CollectionUtils.arrayToList(key)) > 0 ? true : false;         }         return false;     }      /**      * 匹配所有的key      * @param pettern      * @return      */     public Set keys(String pettern){         if(pettern.trim() != "" && pettern != null){             return redisTemplate.keys(pettern);         }         return null;     }      // ============================String=============================     /**      * 普通缓存获取      * @param key 键      * @return 值      */     public Object get(String key) {         return key == null ? null : redisTemplate.opsForValue().get(key);     }     /**      * 普通缓存放入      * @param key 键      * @param value 值      * @return true成功 false失败      */     public boolean set(String key, Object value) {         try {             redisTemplate.opsForValue().set(key, value);             return true;         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 普通缓存放入并设置时间      * @param key 键      * @param value 值      * @param time 时间(秒) time要大于0 如果time小于等于0 将设置无限期      * @return true成功 false 失败      */     public boolean set(String key, Object value, long time) {         try {             if (time > 0) {                 redisTemplate.opsForValue().set(key, value, time, TimeUnit.SECONDS);             } else {                 set(key, value);             }             return true;         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 递增      * @param key 键      * @param delta 要增加几(大于0)      * @return      */     public long incr(String key, long delta) {         if (delta < 0) {             throw new RuntimeException("递增因子必须大于0");         }         return redisTemplate.opsForValue().increment(key, delta);     }     /**      * 递减      * @param key 键      * @param delta 要减少几(小于0)      * @return      */     public long decr(String key, long delta) {         if (delta < 0) {             throw new RuntimeException("递减因子必须大于0");         }         return redisTemplate.opsForValue().increment(key, -delta);     }     // ================================Map=================================     /**      * HashGet      * @param key 键 不能为null      * @param item 项 不能为null      * @return 值      */     public Object hget(String key, String item) {          return redisTemplate.opsForHash().get(key, item);     }     /**      * 获取hashKey对应的所有键值      * @param key 键      * @return 对应的多个键值      */     public List hmget(String key) {          return redisTemplate.opsForHash().values(key);     }     /**      * HashSet      * @param key 键      * @param map 对应多个键值      * @return true 成功 false 失败      */     public boolean hmset(String key, Map map) {         try {             redisTemplate.opsForHash().putAll(key, map);             return true;         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * HashSet 并设置时间      * @param key 键      * @param map 对应多个键值      * @param time 时间(秒)      * @return true成功 false失败      */     public boolean hmset(String key, Map map, long time) {         try {             redisTemplate.opsForHash().putAll(key, map);             if (time > 0) {                 expire(key, time);             }             return true;         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 向一张hash表中放入数据,如果不存在将创建      * @param key 键      * @param item 项      * @param value 值      * @return true 成功 false失败      */     public boolean hset(String key, String item, Object value) {         try {             redisTemplate.opsForHash().put(key, item, value);             return true;         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 向一张hash表中放入数据,如果不存在将创建      * @param key 键      * @param item 项      * @param value 值      * @param time 时间(秒) 注意:如果已存在的hash表有时间,这里将会替换原有的时间      * @return true 成功 false失败      */     public boolean hset(String key, String item, Object value, long time) {         try {             redisTemplate.opsForHash().put(key, item, value);             if (time > 0) {                 expire(key, time);             }             return true;         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 删除hash表中的值      * @param key 键 不能为null      * @param item 项 可以使多个 不能为null      */     public void hdel(String key, Object... item) {         redisTemplate.opsForHash().delete(key, item);     }     /**      * 判断hash表中是否有该项的值      * @param key 键 不能为null      * @param item 项 不能为null      * @return true 存在 false不存在      */     public boolean hHasKey(String key, String item) {         return redisTemplate.opsForHash().hasKey(key, item);     }     /**      * hash递增 如果不存在,就会创建一个 并把新增后的值返回      * @param key 键      * @param item 项      * @param by 要增加几(大于0)      * @return      */     public double hincr(String key, String item, double by) {         return redisTemplate.opsForHash().increment(key, item, by);     }     /**      * hash递减      * @param key 键      * @param item 项      * @param by 要减少记(小于0)      * @return      */     public double hdecr(String key, String item, double by) {         return redisTemplate.opsForHash().increment(key, item, -by);     }      /**      * 存储对象      * @param key      * @param field      * @param value      */     public void putSome(Class cls,String key,String field,Object value){         redisTemplate.setHashValueSerializer(new Jackson2JsonRedisSerializer<>(cls));         redisTemplate.boundHashOps(key).put(field,value);     }      /**      * 获取对象      * @param key      * @param cls      * @return      */     public Cursor> getSome(String key,Class cls){         redisTemplate.setHashValueSerializer(new Jackson2JsonRedisSerializer<>(cls));         Cursor> scan = redisTemplate.boundHashOps(key).scan(ScanOptions.scanOptions().build());         return scan;     }       // ============================set=============================     /**      * 根据key获取Set中的所有值      * @param key 键      * @return      */     public Set sGet(String key) {         try {             return redisTemplate.opsForSet().members(key);         } catch (Exception e) {             e.printStackTrace();             return null;         }     }     /**      * 根据value从一个set中查询,是否存在      * @param key 键      * @param value 值      * @return true 存在 false不存在      */     public boolean sHasKey(String key, Object value) {         try {             return redisTemplate.opsForSet().isMember(key, value);         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 将数据放入set缓存      * @param key 键      * @param values 值 可以是多个      * @return 成功个数      */     public long sSet(String key, Object... values) {         try {             return redisTemplate.opsForSet().add(key, values);         } catch (Exception e) {             e.printStackTrace();             return 0;         }     }     /**      * 将set数据放入缓存      * @param key 键      * @param time 时间(秒)      * @param values 值 可以是多个      * @return 成功个数      */     public long sSetAndTime(String key, long time, Object... values) {         try {             Long count = redisTemplate.opsForSet().add(key, values);             if (time > 0)                 expire(key, time);             return count;         } catch (Exception e) {             e.printStackTrace();             return 0;         }     }     /**      * 获取set缓存的长度      * @param key 键      * @return      */     public long sGetSetSize(String key) {         try {             return redisTemplate.opsForSet().size(key);         } catch (Exception e) {             e.printStackTrace();             return 0;         }     }     /**      * 移除值为value的      * @param key 键      * @param values 值 可以是多个      * @return 移除的个数      */     public long setRemove(String key, Object... values) {         try {             Long count = redisTemplate.opsForSet().remove(key, values);             return count;         } catch (Exception e) {             e.printStackTrace();             return 0;         }     }     // ===============================list=================================     /**      * 获取list缓存的内容      * @param key 键      * @param start 开始      * @param end 结束 0 到 -1代表所有值      * @return      */     public List lGet(String key, long start, long end) {         try {             return redisTemplate.opsForList().range(key, start, end);         } catch (Exception e) {             e.printStackTrace();             return null;         }     }     /**      * 获取list缓存的长度      * @param key 键      * @return      */     public long lGetListSize(String key) {         try {             return redisTemplate.opsForList().size(key);         } catch (Exception e) {             e.printStackTrace();             return 0;         }     }     /**      * 通过索引 获取list中的值      * @param key 键      * @param index 索引 index>=0时, 0 表头,1 第二个元素,依次类推;index<0时,-1,表尾,-2倒数第二个元素,依次类推      * @return      */     public Object lGetIndex(String key, long index) {         try {             return redisTemplate.opsForList().index(key, index);         } catch (Exception e) {             e.printStackTrace();             return null;         }     }     /**      * 将list放入缓存      * @param key 键      * @param value 值      * @return      */     public boolean lSet(String key, Object value) {         try {             redisTemplate.opsForList().rightPush(key, value);             return true;         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 将list放入缓存      * @param key 键      * @param value 值      * @param time 时间(秒)      * @return      */     public boolean lSet(String key, Object value, long time) {         try {             redisTemplate.opsForList().rightPush(key, value);             if (time > 0)                 expire(key, time);             return true;         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 将list放入缓存      * @param key 键      * @param value 值      * @return      */     public boolean lSet(String key, List value) {         try {             redisTemplate.opsForList().rightPushAll(key, value);             return true;         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 将list放入缓存      *      * @param key 键      * @param value 值      * @param time 时间(秒)      * @return      */     public boolean lSet(String key, List value, long time) {         try {             redisTemplate.opsForList().rightPushAll(key, value);             if (time > 0)                 expire(key, time);             return true;         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 根据索引修改list中的某条数据      * @param key 键      * @param index 索引      * @param value 值      * @return      */     public boolean lUpdateIndex(String key, long index, Object value) {         try {             redisTemplate.opsForList().set(key, index, value);             return true;         } catch (Exception e) {             e.printStackTrace();             return false;         }     }     /**      * 移除N个值为value      * @param key 键      * @param count 移除多少个      * @param value 值      * @return 移除的个数      */     public long lRemove(String key, long count, Object value) {         try {             Long remove = redisTemplate.opsForList().remove(key, count, value);             return remove;         } catch (Exception e) {             e.printStackTrace();             return 0;         }     } }
  收藏等于学会,多多支持我吧!长期稳定输出优质系列文章!!!
越南穷?看看人家咖啡店的装修,韩式咖啡成了越南年轻人的最爱色调柔和的咖啡厅,空间布置得简单但很精致,吸引了很多游客前来打卡。如果您正在西贡,想体验下越南的咖啡馆,千万不要错过这篇文章。这里有一些很棒的建议,供您选择!西贡火爆的咖啡馆1。T太极古村呈坎,藏于山谷中的山水田园风光,探究巷弄里的历史痕迹呈坎原名龙溪,位于安徽省黄山市徽州区,始建于东汉三国时期,距今已有1800多年的历史。古村落内的房屋属于典型的徽派建筑风格,黑瓦白墙的民居高低错落地分布在村里的每一个角落,紧凑的格大量实体店关门倒闭,只有这几类商铺越开越多,啥情况?这些年随着电商网购的突飞猛进,实体店受到了非常大的冲击,街上商场冷冷清清,商铺关门转让招租随处可见,曾经人来人往熙熙攘攘的街道也变成了美好的回忆。据悉在2021年,有超过1000万中国GDP,只有美国的七成多,为啥在贸易战中可以扛住美国的压力?记得特朗普刚开始和中国打贸易战时,网上一片哀嚎,以为中国要完了,甚至建议中国应该主动示弱,继续进行韬光养晦。但贸易战打着打着,中国反而觉得没必要让步了,不仅扛住了贸易战的硝烟,还取王长贵直播向粉丝哭穷,每月花销50万压力大,抱怨赵本山无情无义前段时间,前本山传媒旗下演员娇娇跟赵本山宣战的事,一直在持续发酵。娇娇多次控诉赵本山肆意操控他人命运,限制自己上台,称早已看透赵本山这个所谓艺术家的真实面目。大部分网友都认为娇娇这茶叶中的7种主要营养物质,你知道是什么吗?茶虽不是药,但茶对于我们而言依旧是养生路上必不可少的饮品,绿色且健康,那么茶叶中都有哪些对人体有益的营养物质呢?下面就来好好给大家解析一下茶叶中的那些营养成分。蛋白质茶叶中的蛋白质嗓子总有痰,咳也咳不完是怎么回事?1。体质虚弱由于脾虚不运脾气不足,运化无力导致水湿停留,凝聚成痰2。脾胃薄弱宿食逗留,晚上吃的饭不消化,伤到了脾胃3。外感失治因为外在因素引起的,比如外感风寒等,造成肺气损伤,不能失眠多梦易惊醒?一剂小药方,清热安神,一觉到天亮每当工作了一天,疲惫的身体躺在床上,舒舒服服地睡一觉,那是最美好的事了,但偏偏有的人,躺在床上辗转反侧就是无法入睡,心烦意乱,搞得自己身心俱疲,这到底是怎么了?在我们中医上讲,失眠辟谣别再信睡前的老话说,会睡得好我们在说到所有健康养身的时候,都一定要打响其中必要的一枪,那就是保持好睡眠,别熬夜。那么你知道我们平时听多了很多睡前的老话,说是可以让睡眠更好!可其实它们只会毁了睡眠吞了健康,这个当下流行的半身裙有哪些?10套搭配,出门拍照太美了煎熬的冬季,已经慢慢离我们而去,春季正在来临,一切又开始恢复了生机。春季,是一个穿半身裙的好时节,怎样搭配才算好看,这是考验大家技术的时候了,下面糖果就为大家介绍几个比较常见的搭配年纪偏大的女性,如何穿粉色才不庸俗还减龄?嗨,各位大小仙女们,大家好呀!是不是在很多人的眼中,都觉得四十岁以后的女人,穿粉色系的单品,就会给人呈现出一种装嫩的嫌疑?其实,真的不是这样,最重要的一个原因还是因为你自己不会穿而
济南又下雪了!春节假期,现实版冰雪奇缘火出圈记者程凌润通讯员王红娟山野茫茫,雾凇层峦。济南九如山又下雪了!1月23日,九如山迎来了2023年的第二场雪,轻柔的雪花点缀在山涧屋脊栈道白雪皑皑,银装素裹,尽情描摹着山川之美。春节53岁TVB女星路边买年货,搀扶年迈母亲太温馨,打扮低调毫无架子近日,香港TVB知名女星郭少芸在社交平台上晒出自己在香港街头买年货的视频,从视频中能看出当天的集市上人来人往,大家都在为过年做准备,一幅喜气洋洋的景象,能看的出祖国也越加繁荣,而且90后00后成年货消费主力军!超7成海鲜礼盒,销往中西部城市!跨区域采购成新时尚视频加载中(央视财经第一时间)春节渐近,消费市场活力满满。今年年货消费有哪些新特点?各地餐厅年夜饭预订情况如何?哪些城市是旅游过年热门目的地?来自电商平台的销售数据和用户调查显示,P。J。Tucker穿着两款AirJordanOregonDucks配色费城76人队前锋P。J。Tucker穿着两双俄勒冈鸭队球员专属的AirJordan球鞋。与其他任何职业运动相比,NBA是一个更注重球场内外鞋类的联盟。在该协会的所有400多名球员中碧昂丝穿着深v金色镂空礼服,举行四年来的首场演唱会,性感十足周六,碧昂丝在迪拜举行了一场盛大的演出,这是她四年来的首次演唱会。这位41岁的音乐偶像登台庆祝新豪华酒店亚特兰蒂斯皇家酒店的开业,吸引了一群超一线大明星。这位女歌手在红毯上的表现更春晚公开百鸟归巢舞台细节谭维维发长文揭震撼开唱幕后天空为羽,大地为翼,日月为眸,余光是星。除夕之夜,春晚现场,谭维维一曲百鸟归巢唱响千年传承的非遗南音元素,不少观众称惊艳震撼。百鸟归巢是一部打磨了近2年的创新作品。据词作者喻江回忆Perkz有机会复仇G2?Bo拿下Jankos开局两连胜,比上一局好太多在不久前结束的LEC春季赛第二个比赛日当中,VIT战队战胜了TH战队,拿下了自己的二连胜。这场比赛当中有很多大家的熟面孔,VIT这边有Bo和Perkz,TH则是有前DFM上单Evi基金放大镜23年老将依旧香,朱少醒管理的富国天惠去年四季度还被净申购了记者杜萌19日晚间,23年老将朱少醒管理的富国天惠成长披露去年四季报。从收益来看,富国天惠混合去年全年浮亏23。9,如果将统计维度放宽到过去五年,那么收益率为43。87。从规模来看国际油价创1991年来最糟糕开局!上蹿下跳或延续我们刚刚经历了世界能源市场历史上最为动荡的一年。不仅是原油价格剧烈波动,天然气汽油和柴油等价格都发生了大幅波动,能源危机阴影笼罩整个2022年。随着全球地缘政治争端加剧,围绕石油天内维尔现在的枪手就像20年前一样,但他们离获得冠军还差很远直播吧1月23日讯今天凌晨,阿森纳在主场32逆转曼联继续在联赛中领跑,赛后加里内维尔认为现在的阿森纳感觉就像20年前一样如此优秀。加里内维尔在他最新的天空体育播客中说我真的很喜欢,GitHub总是突然无法访问?教你这样轻松解决!关注头条号天域云网络浏览更多内容前言GitHub是全球最大的代码开源平台,小伙伴们平时都喜欢在那里找一些优质的开源项目来学习,以提升自己的编程技能。但最近,由于某些特殊原因,越来越