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

Java并发编程的艺术线程间交换数据的Exchanger

  Exchanger是用于线程协作的工具类, 主要用于两个线程之间的数据交换。带着BAT大厂的面试问题去理解Exchanger
  请带着这些问题继续后文,会很大程度上帮助你更好的理解Exchanger。Exchanger主要解决什么问题?对比SynchronousQueue,为什么说Exchanger可被视为 SynchronousQueue 的双向形式?Exchanger在不同的JDK版本中实现有什么差别?Exchanger实现机制?Exchanger已经有了slot单节点,为什么会加入arena node数组? 什么时候会用到数组?arena可以确保不同的slot在arena中是不会相冲突的,那么是怎么保证的呢?什么是伪共享,Exchanger中如何体现的?Exchanger实现举例Exchanger简介
  Exchanger用于进行两个线程之间的数据交换。它提供一个同步点,在这个同步点,两个线程可以交换彼此的数据。这两个线程通过exchange()方法交换数据,当一个线程先执行exchange()方法后,它会一直等待第二个线程也执行exchange()方法,当这两个线程到达同步点时,这两个线程就可以交换数据了。Exchanger实现机制for (;;) {     if (slot is empty) { // offer         // slot为空时,将item 设置到Node 中                 place item in a Node;         if (can CAS slot from empty to node) {             // 当将node通过CAS交换到slot中时,挂起线程等待被唤醒             wait for release;             // 被唤醒后返回node中匹配到的item             return matching item in node;         }     } else if (can CAS slot from node to empty) { // release          // 将slot设置为空         // 获取node中的item,将需要交换的数据设置到匹配的item         get the item in node;         set matching item in node;         // 唤醒等待的线程         release waiting thread;     }     // else retry on CAS failure }
  比如有2条线程A和B,A线程交换数据时,发现slot为空,则将需要交换的数据放在slot中等待其它线程进来交换数据,等线程B进来,读取A设置的数据,然后设置线程B需要交换的数据,然后唤醒A线程,原理就是这么简单。但是当多个线程之间进行交换数据时就会出现问题,所以Exchanger加入了slot数组。Exchanger源码解析内部类 - Participantstatic final class Participant extends ThreadLocal {     public Node initialValue() { return new Node(); } }
  Participant的作用是为每个线程保留唯一的一个Node节点, 它继承ThreadLocal,说明每个线程具有不同的状态。内部类 - Node@sun.misc.Contended static final class Node {      // arena的下标,多个槽位的时候利用     int index;      // 上一次记录的Exchanger.bound     int bound;      // 在当前bound下CAS失败的次数;     int collides;     // 用于自旋;     int hash;      // 这个线程的当前项,也就是需要交换的数据;     Object item;      //做releasing操作的线程传递的项;     volatile Object match;      //挂起时设置线程值,其他情况下为null;     volatile Thread parked; }
  在Node定义中有两个变量值得思考:bound以及collides。前面提到了数组area是为了避免竞争而产生的,如果系统不存在竞争问题,那么完全没有必要开辟一个高效的arena来徒增系统的复杂性。首先通过单个slot的exchanger来交换数据,当探测到竞争时将安排不同的位置的slot来保存线程Node,并且可以确保没有slot会在同一个缓存行上。如何来判断会有竞争呢? CAS替换slot失败,如果失败,则通过记录冲突次数来扩展arena的尺寸,我们在记录冲突的过程中会跟踪"bound"的值,以及会重新计算冲突次数在bound的值被改变时。核心属性private final Participant participant; private volatile Node[] arena; private volatile Node slot;为什么会有 arena数组槽?
  slot为单个槽,arena为数组槽, 他们都是Node类型。在这里可能会感觉到疑惑,slot作为Exchanger交换数据的场景,应该只需要一个就可以了啊? 为何还多了一个Participant 和数组类型的arena呢? 一个slot交换场所原则上来说应该是可以的,但实际情况却不是如此,多个参与者使用同一个交换场所时,会存在严重伸缩性问题。既然单个交换场所存在问题,那么我们就安排多个,也就是数组arena。通过数组arena来安排不同的线程使用不同的slot来降低竞争问题,并且可以保证最终一定会成对交换数据。但是Exchanger不是一来就会生成arena数组来降低竞争,只有当产生竞争是才会生成arena数组。那么怎么将Node与当前线程绑定呢?
  Participant,Participant 的作用就是为每个线程保留唯一的一个Node节点,它继承ThreadLocal,同时在Node节点中记录在arena中的下标index。构造函数/** * Creates a new Exchanger. */ public Exchanger() {     participant = new Participant(); }
  初始化participant对象。核心方法 - exchange(V x)
  等待另一个线程到达此交换点(除非当前线程被中断),然后将给定的对象传送给该线程,并接收该线程的对象。public V exchange(V x) throws InterruptedException {     Object v;     // 当参数为null时需要将item设置为空的对象     Object item = (x == null) ? NULL_ITEM : x; // translate null args     // 注意到这里的这个表达式是整个方法的核心     if ((arena != null ||             (v = slotExchange(item, false, 0 L)) == null) &&         ((Thread.interrupted() || // disambiguates null return             (v = arenaExchange(item, false, 0 L)) == null)))         throw new InterruptedException();     return (v == NULL_ITEM) ? null : (V) v; }
  这个方法比较好理解:arena为数组槽,如果为null,则执行slotExchange()方法,否则判断线程是否中断,如果中断值抛出InterruptedException异常,没有中断则执行arenaExchange()方法。整套逻辑就是:如果slotExchange(Object item, boolean timed, long ns)方法执行失败了就执行arenaExchange(Object item, boolean timed, long ns)方法,最后返回结果V。
  NULL_ITEM 为一个空节点,其实就是一个Object对象而已,slotExchange()为单个slot交换。slotExchange(Object item, boolean timed, long ns)private final Object slotExchange(Object item, boolean timed, long ns) {     // 获取当前线程node对象     Node p = participant.get();     // 当前线程     Thread t = Thread.currentThread();     // 若果线程被中断,就直接返回null     if (t.isInterrupted()) // preserve interrupt status so caller can recheck         return null; 	// 自旋     for (Node q;;) {         // 将slot值赋给q         if ((q = slot) != null) {              // slot 不为null,即表示已有线程已经把需要交换的数据设置在slot中了 			// 通过CAS将slot设置成null             if (U.compareAndSwapObject(this, SLOT, q, null)) {                 // CAS操作成功后,将slot中的item赋值给对象v,以便返回。                 // 这里也是就读取之前线程要交换的数据                 Object v = q.item;                 // 将当前线程需要交给的数据设置在q中的match                 q.match = item;                  // 获取被挂起的线程                 Thread w = q.parked;                 if (w != null)                     // 如果线程不为null,唤醒它                     U.unpark(w);                 // 返回其他线程给的V                 return v;             }             // create arena on contention, but continue until slot null             // CAS 操作失败,表示有其它线程竞争,在此线程之前将数据已取走             // NCPU:CPU的核数             // bound == 0 表示arena数组未初始化过,CAS操作bound将其增加SEQ             if (NCPU > 1 && bound == 0 &&                 U.compareAndSwapInt(this, BOUND, 0, SEQ))                 // 初始化arena数组                 arena = new Node[(FULL + 2) << ASHIFT];         }         // 上面分析过,只有当arena不为空才会执行slotExchange方法的 		// 所以表示刚好已有其它线程加入进来将arena初始化         else if (arena != null)             // 这里就需要去执行arenaExchange             return null; // caller must reroute to arenaExchange         else {             // 这里表示当前线程是以第一个线程进来交换数据             // 或者表示之前的数据交换已进行完毕,这里可以看作是第一个线程             // 将需要交换的数据先存放在当前线程变量p中             p.item = item;             // 将需要交换的数据通过CAS设置到交换区slot             if (U.compareAndSwapObject(this, SLOT, null, p))                 // 交换成功后跳出自旋                 break;             // CAS操作失败,表示有其它线程刚好先于当前线程将数据设置到交换区slot             // 将当前线程变量中的item设置为null,然后自旋获取其它线程存放在交换区slot的数据             p.item = null;         }     }      // await release     // 执行到这里表示当前线程已将需要的交换的数据放置于交换区slot中了,     // 等待其它线程交换数据然后唤醒当前线程     int h = p.hash;     long end = timed ? System.nanoTime() + ns : 0 L;     // 自旋次数     int spins = (NCPU > 1) ? SPINS : 1;     Object v;     // 自旋等待直到p.match不为null,也就是说等待其它线程将需要交换的数据放置于交换区slot     while ((v = p.match) == null) {         // 下面的逻辑主要是自旋等待,直到spins递减到0为止         if (spins > 0) {             h ^= h << 1;             h ^= h >>> 3;             h ^= h << 10;             if (h == 0)                 h = SPINS | (int) t.getId();             else if (h < 0 && (--spins & ((SPINS >>> 1) - 1)) == 0)                 Thread.yield();         } else if (slot != p)             spins = SPINS;         // 此处表示未设置超时或者时间未超时         else if (!t.isInterrupted() && arena == null &&             (!timed || (ns = end - System.nanoTime()) > 0 L)) {             // 设置线程t被当前对象阻塞             U.putObject(t, BLOCKER, this);             // 给p挂机线程的值赋值             p.parked = t;             if (slot == p)                 // 如果slot还没有被置为null,也就表示暂未有线程过来交换数据,需要将当前线程挂起                 U.park(false, ns);             // 线程被唤醒,将被挂起的线程设置为null             p.parked = null;             // 设置线程t未被任何对象阻塞             U.putObject(t, BLOCKER, null);         // 不是以上条件时(可能是arena已不为null或者超时)             } else if (U.compareAndSwapObject(this, SLOT, p, null)) {              // arena不为null则v为null,其它为超时则v为超市对象TIMED_OUT,并且跳出循环             v = timed && ns <= 0 L && !t.isInterrupted() ? TIMED_OUT : null;             break;         }     }     // 取走match值,并将p中的match置为null     U.putOrderedObject(p, MATCH, null);     // 设置item为null     p.item = null;     p.hash = h;     // 返回交换值     return v; }
  程序首先通过participant获取当前线程节点Node。检测是否中断,如果中断return null,等待后续抛出InterruptedException异常。如果slot不为null,则进行slot消除,成功直接返回数据V,否则失败,则创建arena消除数组。如果slot为null,但arena不为null,则返回null,进入arenaExchange逻辑。如果slot为null,且arena也为null,则尝试占领该slot,失败重试,成功则跳出循环进入spin+block(自旋+阻塞)模式。
  在自旋+阻塞模式中,首先取得结束时间和自旋次数。如果match(做releasing操作的线程传递的项)为null,其首先尝试spins+随机次自旋(改自旋使用当前节点中的hash,并改变之)和退让。当自旋数为0后,假如slot发生了改变(slot != p)则重置自旋数并重试。否则假如:当前未中断&arena为null&(当前不是限时版本或者限时版本+当前时间未结束):阻塞或者限时阻塞。假如:当前中断或者arena不为null或者当前为限时版本+时间已经结束:不限时版本:置v为null;限时版本:如果时间结束以及未中断则TIMED_OUT;否则给出null(原因是探测到arena非空或者当前线程中断)。
  match不为空时跳出循环。arenaExchange(Object item, boolean timed, long ns)
  此方法被执行时表示多个线程进入交换区交换数据,arena数组已被初始化,此方法中的一些处理方式和slotExchange比较类似,它是通过遍历arena数组找到需要交换的数据。// timed 为true表示设置了超时时间,ns为>0的值,反之没有设置超时时间 private final Object arenaExchange(Object item, boolean timed, long ns) {     Node[] a = arena;     // 获取当前线程中的存放的node     Node p = participant.get();     //index初始值0     for (int i = p.index;;) { // access slot at i         // 遍历,如果在数组中找到数据则直接交换并唤醒线程,如未找到则将需要交换给其它线程的数据放置于数组中         int b, m, c;         long j; // j is raw array offset         // 其实这里就是向右遍历数组,只是用到了元素在内存偏移的偏移量         // q实际为arena数组偏移(i + 1) *  128个地址位上的node         Node q = (Node) U.getObjectVolatile(a, j = (i << ASHIFT) + ABASE);         // 如果q不为null,并且CAS操作成功,将下标j的元素置为null         if (q != null && U.compareAndSwapObject(a, j, q, null)) {             // 表示当前线程已发现有交换的数据,然后获取数据,唤醒等待的线程             Object v = q.item; // release             q.match = item;             Thread w = q.parked;             if (w != null)                 U.unpark(w);             return v;         // q 为null 并且 i 未超过数组边界             } else if (i <= (m = (b = bound) & MMASK) && q == null) {              // 将需要给其它线程的item赋予给p中的item             p.item = item; // offer             if (U.compareAndSwapObject(a, j, null, p)) {                 // 交换成功                 long end = (timed && m == 0) ? System.nanoTime() + ns : 0 L;                 Thread t = Thread.currentThread(); // wait                 // 自旋直到有其它线程进入,遍历到该元素并与其交换,同时当前线程被唤醒                 for (int h = p.hash, spins = SPINS;;) {                     Object v = p.match;                     if (v != null) {                         // 其它线程设置的需要交换的数据match不为null                         // 将match设置null,item设置为null                         U.putOrderedObject(p, MATCH, null);                         p.item = null; // clear for next use                         p.hash = h;                         return v;                     } else if (spins > 0) {                         h ^= h << 1;                         h ^= h >>> 3;                         h ^= h << 10; // xorshift                         if (h == 0) // initialize hash                             h = SPINS | (int) t.getId();                         else if (h < 0 && // approx 50% true                             (--spins & ((SPINS >>> 1) - 1)) == 0)                             Thread.yield(); // two yields per wait                     } else if (U.getObjectVolatile(a, j) != p)                         // 和slotExchange方法中的类似,arena数组中的数据已被CAS设置                        // match值还未设置,让其再自旋等待match被设置                         spins = SPINS; // releaser hasn"t set match yet                     else if (!t.isInterrupted() && m == 0 &&                         (!timed ||                             (ns = end - System.nanoTime()) > 0 L)) {                         // 设置线程t被当前对象阻塞                         U.putObject(t, BLOCKER, this); // emulate LockSupport                          // 线程t赋值                         p.parked = t; // minimize window                         if (U.getObjectVolatile(a, j) == p)                             // 数组中对象还相等,表示线程还未被唤醒,唤醒线程                             U.park(false, ns);                         p.parked = null;                          // 设置线程t未被任何对象阻塞                         U.putObject(t, BLOCKER, null);                     } else if (U.getObjectVolatile(a, j) == p &&                         U.compareAndSwapObject(a, j, p, null)) {                         // 这里给bound增加加一个SEQ                         if (m != 0) // try to shrink                             U.compareAndSwapInt(this, BOUND, b, b + SEQ - 1);                         p.item = null;                         p.hash = h;                         i = p.index >>>= 1; // descend                         if (Thread.interrupted())                             return null;                         if (timed && m == 0 && ns <= 0 L)                             return TIMED_OUT;                         break; // expired; restart                     }                 }             } else                 // 交换失败,表示有其它线程更改了arena数组中下标i的元素                 p.item = null; // clear offer         } else {             // 此时表示下标不在bound & MMASK或q不为null但CAS操作失败            // 需要更新bound变化后的值             if (p.bound != b) { // stale; reset                 p.bound = b;                 p.collides = 0;                 // 反向遍历                 i = (i != m || m == 0) ? m : m - 1;             } else if ((c = p.collides) < m || m == FULL ||                 !U.compareAndSwapInt(this, BOUND, b, b + SEQ + 1)) {                  // 记录CAS失败的次数                 p.collides = c + 1;                 // 循环遍历                 i = (i == 0) ? m : i - 1; // cyclically traverse             } else                 // 此时表示bound值增加了SEQ+1                 i = m + 1; // grow             // 设置下标             p.index = i;         }     } }
  首先通过participant取得当前节点Node,然后根据当前节点Node的index去取arena中相对应的节点node。前面提到过arena可以确保不同的slot在arena中是不会相冲突的,那么是怎么保证的呢?arena = new Node[(FULL + 2) << ASHIFT]; // 这个arena到底有多大呢? 我们先看FULL 和ASHIFT的定义: static final int FULL = (NCPU >= (MMASK << 1)) ? MMASK : NCPU >>> 1; private static final int ASHIFT = 7;  private static final int NCPU = Runtime.getRuntime().availableProcessors(); private static final int MMASK = 0xff;        // 255 // 假如我的机器NCPU = 8 ,则得到的是768大小的arena数组。然后通过以下代码取得在arena中的节点:  Node q = (Node)U.getObjectVolatile(a, j = (i << ASHIFT) + ABASE); // 它仍然是通过右移ASHIFT位来取得Node的,ABASE定义如下:  Class<?> ak = Node[].class; ABASE = U.arrayBaseOffset(ak) + (1 << ASHIFT); // U.arrayBaseOffset获取对象头长度,数组元素的大小可以通过unsafe.arrayIndexScale(T[].class) 方法获取到。这也就是说要访问类型为T的第N个元素的话,你的偏移量offset应该是arrayOffset+N*arrayScale。也就是说BASE = arrayOffset+ 128 。用@sun.misc.Contended来规避伪共享?
  伪共享说明:假设一个类的两个相互独立的属性a和b在内存地址上是连续的(比如FIFO队列的头尾指针),那么它们通常会被加载到相同的cpu cache line里面。并发情况下,如果一个线程修改了a,会导致整个cache line失效(包括b),这时另一个线程来读b,就需要从内存里再次加载了,这种多线程频繁修改ab的情况下,虽然a和b看似独立,但它们会互相干扰,非常影响性能。
  我们再看Node节点的定义, 在Java 8 中我们是可以利用sun.misc.Contended来规避伪共享的。所以说通过 << ASHIFT方式加上sun.misc.Contended,所以使得任意两个可用Node不会再同一个缓存行中。@sun.misc.Contended static final class Node{ .... }
  我们再次回到arenaExchange()。取得arena中的node节点后,如果定位的节点q 不为空,且CAS操作成功,则交换数据,返回交换的数据,唤醒等待的线程。如果q等于null且下标在bound & MMASK范围之内,则尝试占领该位置,如果成功,则采用自旋 + 阻塞的方式进行等待交换数据。如果下标不在bound & MMASK范围之内获取由于q不为null但是竞争失败的时候:消除p。加入bound 不等于当前节点的bond(b != p.bound),则更新p.bound = b,collides = 0 ,i = m或者m - 1。如果冲突的次数不到m 获取m 已经为最大值或者修改当前bound的值失败,则通过增加一次collides以及循环递减下标i的值;否则更新当前bound的值成功:我们令i为m+1即为此时最大的下标。最后更新当前index的值。更深入理解SynchronousQueue对比?
  Exchanger是一种线程间安全交换数据的机制。可以和之前分析过的SynchronousQueue对比一下:线程A通过SynchronousQueue将数据a交给线程B;线程A通过Exchanger和线程B交换数据,线程A把数据a交给线程B,同时线程B把数据b交给线程A。可见,SynchronousQueue是交给一个数据,Exchanger是交换两个数据。不同JDK实现有何差别?在JDK5中Exchanger被设计成一个容量为1的容器,存放一个等待线程,直到有另外线程到来就会发生数据交换,然后清空容器,等到下一个到来的线程。从JDK6开始,Exchanger用了类似ConcurrentMap的分段思想,提供了多个slot,增加了并发执行时的吞吐量。
  JDK1.6实现可以参考 这里 (opens new window)Exchanger示例
  来一个非常经典的并发问题:你有相同的数据buffer,一个或多个数据生产者,和一个或多个数据消费者。只是Exchange类只能同步2个线程,所以你只能在你的生产者和消费者问题中只有一个生产者和一个消费者时使用这个类。public class Test {     static class Producer extends Thread {         private Exchanger exchanger;         private static int data = 0;         Producer(String name, Exchanger exchanger) {             super("Producer-" + name);             this.exchanger = exchanger;         }          @Override         public void run() {             for (int i=1; i<5; i++) {                 try {                     TimeUnit.SECONDS.sleep(1);                     data = i;                     System.out.println(getName()+" 交换前:" + data);                     data = exchanger.exchange(data);                     System.out.println(getName()+" 交换后:" + data);                 } catch (InterruptedException e) {                     e.printStackTrace();                 }             }         }     }      static class Consumer extends Thread {         private Exchanger exchanger;         private static int data = 0;         Consumer(String name, Exchanger exchanger) {             super("Consumer-" + name);             this.exchanger = exchanger;         }          @Override         public void run() {             while (true) {                 data = 0;                 System.out.println(getName()+" 交换前:" + data);                 try {                     TimeUnit.SECONDS.sleep(1);                     data = exchanger.exchange(data);                 } catch (InterruptedException e) {                     e.printStackTrace();                 }                 System.out.println(getName()+" 交换后:" + data);             }         }     }      public static void main(String[] args) throws InterruptedException {         Exchanger exchanger = new Exchanger();         new Producer("", exchanger).start();         new Consumer("", exchanger).start();         TimeUnit.SECONDS.sleep(7);         System.exit(-1);     } }            @pdai: 代码已经复制到剪贴板
  可以看到,其结果可能如下:Consumer- 交换前:0 Producer- 交换前:1 Consumer- 交换后:1 Consumer- 交换前:0 Producer- 交换后:0 Producer- 交换前:2 Producer- 交换后:0 Consumer- 交换后:2 Consumer- 交换前:0 Producer- 交换前:3 Producer- 交换后:0 Consumer- 交换后:3 Consumer- 交换前:0 Producer- 交换前:4 Producer- 交换后:0 Consumer- 交换后:4 Consumer- 交换前:0

比空间站的微重力水平高1000倍!天空课堂解锁神秘实验台,可检验爱因斯坦相对论外层喷气悬浮,内层磁悬浮,在今天(23日)下午举行的天空课堂中出现了一个重量级的家伙高微重力柜中的悬浮实验台,它是实现高微重力环境的关键设备,也是太空黑科技之一。随着王亚平打开黑色各种山寨币骗局被包装成高大上的山寨币,不少朋友在生活中都见过,它们仿照世界知名的比特币和以太坊去发行,统称为山寨币。很多人会极力推销各种山寨币给大家,告诉你买了之后,你持有一段时间,你会暴富的。考特尼斯托登出街气场强大近日考特尼斯托登现身街头,这次考特尼斯托登是外出散步的,考特尼斯托登是一个很漂亮的模特,这次现身留着一头漂亮的银色长发,整个人看上去很有健美风情,考特尼斯托登这次现身穿了一款白色的孙俪终于收获高奢代言,靠的是什么?前几天,孙俪晒出两张相隔18年的合照出圈了,另一位主角关晓彤形容为奇妙的相遇。04年关晓彤还是个6岁的小女孩,孙俪也刚凭借玉观音爆红。如今多年后重逢,她们早已褪去当年的稚嫩与青涩。血液疾病对怀孕的风险和影响常见疾病风险评估1。贫血贫血是指外周血中单位容积内血红蛋白浓度红细胞计数和(或)血细胞比容低于相同年龄性别和地区的正常标准。常伴有其他非特异性的症状体征,如疲劳注意力不集中结膜指甲京东排行榜第一空净测评!净化异味高效杀菌消毒,深藏不露随着现在人们越来越重视空气质量后,雾霾的情况逐渐减少,但是出现一个奇怪的现象,空气质量逐渐变好,购买空气净化器的人数却逐渐激增。我想很大部分的原因在于现在全球yq情况一直持续着,在她和张晨光低调隐婚20年,不求名利,如今被张晨光宠成公主张晨光的许多好友虽然已年过六旬,但身体仍十分硬实,是一位非常有经验的艺人,也是一位十分英俊的男孩子。由他现在的样子可以看出,以前的张晨光一定很英俊,年轻时一定是非常漂亮动人的。这个冯绍峰与赵丽颖是模范前夫前妻,打了一些离异夫妻的脸点击关注,每天都有名人故事感动您!冯绍峰2022年3月17日,35集都市情感剧心居在东方浙江卫视及爱奇艺同步热播,引发观众追剧和热议。当红男星冯绍峰在剧中饰演男主角施源,聚焦了观众顶流小生邓伦,守法公民不做非要作死,如今狼狈不堪近日,邓伦因偷税漏税被追缴并罚款1。06亿的新闻可以说是刷爆了各大媒体平台。继范冰冰郑爽薇娅等明星因逃税漏税被公布封杀后,邓伦成为了又一个被金钱与欲望支配的牺牲品,好好的守法公民不无糖革命到来!哪些天然甜味剂将引爆市场?糖跟每一个人息息相关,从早期的蜂蜜到工业化时代糖产品再到目前的代糖原料,每一次的变化都代表着市场消费趋势和饮食结构的变革。新时代消费趋势下,消费者不想背起甜蜜的负担,也想要保持身体A股今年,可以期待创新高了今年,上证指数创新高,之前只是憧憬,慢慢变成客观事实。下跌并不可怕,每次下跌,急跌都是为了创新高,只是炒股的人都太聪明,等不到创新高的那一天。原本预期主力资金会选择在3400370
最新最响亮好记的餐馆店铺名字大全,送给餐馆店铺起名取名文案社来了,废话不多说,直接上干货。店铺名字大全,公司取名注册公司。店铺名字大全,结合名字,有内涵寓意好。徽源食宴徽可指安徽省,也有美好的意思源五行属水,意为源远流长,源源不断酸奶的好处被低估了!新研究证实可减轻脂肪肝口感酸甜营养丰富的酸奶,是老少皆宜的健康食品。喝牛奶会腹胀腹泻的人,喝酸奶就能完美避免乳糖不耐受的问题。近日,权威杂志自然通讯上一项研究发现酸奶的新功效减轻脂肪肝。结合这项研究,采油焖新鲜笋材料春笋57根,糖1勺,生抽2勺,老抽1勺,葱姜适量,盐少许步骤1春笋剥皮,根部切掉,清洗干净,切成小段步骤2锅中加水煮开,倒入笋段,煮开5分钟,除去涩味,然后多清洗几遍步骤3热锅最健康的5种食物!多吃这几种!让你少往医院跑健康饮食对身体和大脑都有好处。当我们吃得健康的时候,我们感觉好,当我们感觉好的时候,我们更快乐,当我们更快乐的时候,我们的活力更高,这是一个循环。这份世界上5种最健康食物的榜单,一311610,威少替补今天爆发!湖人成功逆转掘金,掘金首轮遇勇士北京时间4月11日,湖人队在客场以146141通过加时赛逆转击败掘金队,取得收官战胜利。本场比赛湖人队最多时曾落后掘金队17分,差距非常大,不过好在,里夫斯在末节和加时赛挺身而出,里弗斯311610蒙克41分湖人加时擒掘金33胜收官NBA常规赛4月11日继续进行,本场比赛湖人这边詹姆斯戴维斯威少安东尼和霍华德缺阵。掘金方面约基奇戈登巴顿和莫里斯不打。最终,湖人以146141战胜掘金。掘金最终以48胜34负排名沪深战裁判报告漏判王哲林对顾全犯规直接影响比赛结果北京时间4月11日,CBA季后赛14决赛次回合,上海和深圳的裁判报告出炉。在最后的2分钟里的4次判罚里,3次正确1次错误,唯一的错判上海男篮受益。在昨晚展开的14决赛次回合较量,上解说杨毅给出预测!广东和辽宁胜负,或3比0横扫对手解说杨毅给出预测!广东和辽宁胜负,或3比0横扫对手亲情提示亲爱的读者,如何能每天能读到这样的体育资讯呢?点击右上角的关注按钮即可,您的关注将是我创作的最大动力!CBA季后赛继续进行退休老教授带娃,孩子轻松玩出超强数学思维,这三点经验值得学习闺蜜的婆婆是一位退休的大学教授,本以为高知家庭的老太太能够跟得上年轻人的育儿观念,可是谈起两个人关于育儿的分歧,闺蜜也是有苦说不出。闺蜜是一位职场女性,她养孩子完全是照书养,怀孕就孩子沉迷手机?这本经典图画书,让孩子瞬间爱上阅读很多宝妈都愁孩子太过于对手机着迷,让孩子读书根本读不进去,或者一让孩子读书,孩子就昏昏欲睡,担心孩子因为阅读量上不去而影响语文成绩。如果你家孩子也不喜欢读书,主要原因可能是书籍没选黄褐斑怎么样才能彻底的去掉?黄褐斑怎么样才能彻底的去掉该怎么样才能彻底去掉黄褐斑最有效的祛斑方式应该是分析自身色斑形成的原因进行去除。也就是科学的祛斑方式。祛斑单单只依靠一种祛斑产品是不能够把色斑去除的,首先