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

java内存模型详细解析

  一. java结构体系
  Description of Java Conceptual Diagram(java结构)
  我们经常说到JVM调优,JVM和JDK到底什么关系,大家知道么?这是java基础。
  这幅图很重要,一定要了解其结构。这是jdk的结构图。从结构上可以看出java结构体系, JDK主要包含两部分:
  第一部分:是java 工具(Tools&Tool APIs)
  比如java, javac, javap等命令. 我们常用的命令都在这里
  第二部分: JRE(全称:Java Runtime Enveriment), jre是Java的核心,。
  ​ jre里面定义了java运行时需要的核心类库, 比如:我们常用的lang包, util包, Math包, Collection包等等.这里还有一个很重要的部分JVM(最后一部分青色的) java 虚拟机, 这部分也是属于jre, 是java运行时环境的一部分. 下面来详细看看:
  最底层的是Java Virtual Machine: java虚拟机
  常用的基础类库:lang and util。在这里定义了我们常用的Math、Collections、Regular Expressions(正则表达式),Logging日志,Reflection反射等等。
  其他的扩展类库:Beans,Security,Serialization序列化,Networking网络,JNI,Date and Time,Input/Output等。
  集成一体化类库:JDBC数据库连接,jndi,scripting等。
  用户接口工具:User Interface Toolkits。
  部署工具:Deployment等。
  从上面就可看出,jvm是整个jdk的最底层。jvm是jdk的一部分。
  二. java语言的跨平台特性
  1. java语言是如何实现跨平台的?
  跨平台指的是, 程序员开发出的一套代码, 在windows平台上能运行, 在linux上也能运行, 在mac上也能运行. 我们知道, 机器最终运行的指令都是二进制指令. 同样的代码, 在windows上生成的二进制指令可能是0101, 但是在linux上是1001, 而在mac上是1011。这样同样的代码, 如果要想在不同的平台运行, 放到相应的平台, 就要修改代码, 而java却不用, 那么java这种跨平台特性是怎么做到的呢?
  原因在于jdk, 我们最终是将程序编译成二进制码,把他丢在jvm上运行的, 而jvm是jre的一部分. 我们在不同的平台下载的jdk是不同的. windows平台要选择下载适用于windows的jdk, linux要选择适用于linux的jdk, mac要选择适用于mac的jdk. 不同平台的jvm针对该平台有一个特定的实现, 正是这种特点的实现, 让java实现了跨平台。
  2. 延伸思考
  通过上面的分析,我们知道能够实现跨平台是因为jvm封装了变化。我们经常说进行jvm调优,那么在不同平台的调优参数可以通用么?显然是不可以的。不同平台的jvm尤其个性化差异。
  封装变化的部分是JDK中的jvm,JVM的整体结构是怎样的呢?来看下面一个部分。
  三. JVM整体结构和内存模型
  1.JVM由三部分组成:
  类装载子系统
  运行时数据区(内存模型)
  字节码执行引擎
  其中类装载子系统是C++实现的, 他把类加载进来, 放入到虚拟机中. 这一块就是之前分析过的类加载器加载类,采用双亲委派机制,把类加载进来放入到jvm虚拟机中。
  然后, 字节码执行引擎去虚拟机中读取数据. 字节码执行引擎也是c++实现的. 我们重点研究运行时数据区。
  2.运行时数据区的构成
  运行时数据区主要由5个部分构成: 堆,栈,本地方法栈,方法区,程序计数器。
  3.JVM三部分密切配合工作
  下面我们来看看一个程序运行的时候, 类装载子系统, 运行时数据区, 字节码执行引擎是如何密切配合工作的?
  我们举个例子来说一下:package com.lxl.jvm;  public class Math {     public static int initData = 666;     public static User user = new User();      public int compute() {         int a = 1;         int b = 2;         int c = (a + b) * 10;         return c;     }      public static void main(String[] args) {         Math math = new Math();         math.compute();     } }
  当我们在执行main方法的时候, 都做了什么事情呢?
  第一步: 类加载子系统加载Math.class类, 然后将其丢到内存区域, 这个就是前面博客研究的部分,类加载的过程, 我们看源码也发现,里面好多代码都是native本地的, 是c++实现的
  第二步: 在内存中处理字节码文件, 这一部分内容较多, 也是我们研究的重点, 后面会对每一个部分详细说
  第三步: 由字节码执行引擎执行java虚拟机中的内存代码, 而字节码执行引擎也是由c++实现的
  这里最核心的部分是第二部分运行时数据区(内存模型), 我们后面的调优, 都是针对这个区域来进行的.
  下面详细来说内存区域
  这是java的内存区域, 内存区域干什么呢?内存区域其实就是放数据的,各种各样的数据j放在不同的内存区域
  四. 栈
  栈是用来存放变量的
  4.1. 栈空间
  还是用Math的例子来说,当程序运行的时候, 会创建一个线程, 创建线程的时候, 就会在大块的栈空间中分配一块小空间, 用来存放当前要运行的线程的变量public static void main(String[] args) {         Math math = new Math();         math.compute();     }
  比如,这段代码要运行,首先会在大块的栈空间中给他分配一块小空间. 这里的math这个局部变量就会被保存在分配的小空间里面.
  在这里面我们运行了math.compute()方法, 我们看看compute方法内部实现public int compute() {         int a = 1;         int b = 2;         int c = (a + b) * 10;         return c;     }
  这里面有a, b, c这样的局部变量, 这些局部变量放在那里呢? 也放在上面分配的栈小空间里面.
  效果如上图, 在栈空间中, 分配一块小的区域, 用来存放Math类中的局部变量
  如果再有一个线程呢? 我们就会再次在栈空间中分配一块小的空间, 用来存放新的线程内部的变量
  同样是变量, main方法中的变量和compute()方法中的变量放在一起么?他们是怎么放得呢?这就涉及到栈帧的概念。
  4.2. 栈帧
  1.什么是栈帧呢?package com.lxl.jvm;  public class Math {     public static int initData = 666;     public static User user = new User();      public int compute() {         int a = 1;         int b = 2;         int c = (a + b) * 10;         return c;     }      public static void main(String[] args) {         Math math = new Math();         math.compute();     } }
  还是这段代码, 我们来看一下, 当我们启动一个线程运行main方法的时候, 一个新的线程启动,会现在栈空间中分配一块小的栈空间。然后在栈空间中分配一块区域给main方法,这块区域就叫做栈帧空间.
  当程序运行到compute()计算方法的时候, 会要去调用compute()方法, 这时候会再分配一个栈帧空间, 给compute()方法使用.
  2.为什么要将一个线程中的不同方法放在不同的栈帧空间里面呢?
  一方面: 我们不同方法里的局部变量是不能相互访问的. 比如compute的a,b,c在main里不能被访问到。使用栈帧做了很好的隔离作用。
  另一方面: 方便垃圾回收, 一个方法用完了, 值也返回了, 那他里面的变量就是垃圾了, 后面直接回收这个栈帧就好了.
  如下图, 在Math中两个方法, 当运行到main方法的时候, 会将main方法放到一块栈帧空间, 这里面仅仅是保存main方法中的局部变量, 当执行到compute方法的时候, 这时会开辟一块compute栈帧空间, 这部分空间仅存放compute()方法的局部变量.
  不同的方法开辟出不同的内存空间, 这样方便我们各个方法的局部变量进行管理, 同时也方便垃圾回收.
  3.java内存模型中的栈算法
  我们学过栈算法, 栈算法是先进后出的. 那么我们的内存模型里的栈和算法里的栈一样么?有关联么?
  我们java内存模型中的栈使用的就是栈算法, 先进后出.举个例子, 还是这段代码package com.lxl.jvm;  public class Math {     public static int initData = 666;     public static User user = new User();      public int compute() {         int a = 1;         int b = 2;         int c = (a + b) * 10;         return c;     }      public int add() {         int a = 1;         int b = 2;         int c = a + b;         return c;     }      public static void main(String[] args) {         Math math = new Math();         math.compute();         math.add();   // 注意这里调用了两次compute()方法     } }
  这时候加载的内存模型是什么样呢?
  最先进入栈的是main方法, 会首先在线程栈中分配一块栈帧空间给main方法。
  main方法里面调用了compute方法, 然后会在创建一个compute方法的栈帧空间, 我们知道compute方法后加载,但是他却会先执行, 执行完以后, compute中的局部变量就会被回收, 那么也就是出栈.
  然后在执行add方法,给add方法分配一块栈帧空间。add执行完以后出栈。
  最后执行完main方法, main方法最后出栈. 这个算法刚好验证了先进后出. 后加载的方法会被先执行. 也符合程序执行的逻辑。
  4.3 栈帧的内部构成
  我们上面说了, 每个方法在运行的时候都会有一块对应的栈帧空间, 那么栈帧空间内部的结构是怎样的呢?
  栈帧内部有很多部分, 我们主要关注下面这四个部分:局部变量表 操作数栈 动态链接 方法出口
  4.2.1 局部变量表: 存放局部变量
  局部变量表,顾名思义,用来存放局部变量的。
  4.2.2 操作数栈
  那么操作数栈,动态链接, 方法出口他们是干什么的呢? 我们用例子来说明操作数栈
  那么这四个部分是如何工作的呢?
  我们用代码的执行过程来对照分析.
  我们要看的是jvm反编译后的字节码文件, 使用javap命令生成反编译字节码文件.
  javap命令是干什么用的呢? 我们可以查看javap的帮助文档
  主要使用javap -c和javap -vjavap -c: 对代码进行反编译 javap -v: 输出附加信息, 他比javap -c会输出更多的内容
  下面使用命令生成一个Math.class的字节码文件. 我们将其生成到文件javap -c Math.class > Math.txt
  打开Math.txt文件, 如下. 这就是对java字节码反编译成jvm汇编语言.Compiled from "Math.java" public class com.lxl.jvm.Math {   public static int initData;    public static com.lxl.jvm.User user;    public com.lxl.jvm.Math();     Code:        0: aload_0        1: invokespecial #1                  // Method java/lang/Object."":()V        4: return    public int compute();     Code:        0: iconst_1        1: istore_1        2: iconst_2        3: istore_2        4: iload_1        5: iload_2        6: iadd        7: bipush        10        9: imul       10: istore_3       11: iload_3       12: ireturn    public static void main(java.lang.String[]);     Code:        0: new           #2                  // class com/lxl/jvm/Math        3: dup        4: invokespecial #3                  // Method "":()V        7: astore_1        8: aload_1        9: invokevirtual #4                  // Method compute:()I       12: pop       13: return    static {};     Code:        0: sipush        666        3: putstatic     #5                  // Field initData:I        6: new           #6                  // class com/lxl/jvm/User        9: dup       10: invokespecial #7                  // Method com/lxl/jvm/User."":()V       13: putstatic     #8                  // Field user:Lcom/lxl/jvm/User;       16: return }
  这就是jvm生成的反编译字节码文件.
  要想看懂这里面的内容, 我们需要知道jvm文档手册. 现在我们不会没关系, 参考文章(javascript:void(0))最后面的内容, 遇到了就去后面查就行了
  我们以compute()方法为例来说说这个方法是如何在在栈中处理的源代码 public int compute() {   int a = 1;   int b = 2;   int c = (a + b) * 10;   return c; }   反编译后的jvm指令 public int compute();     Code:        0: iconst_1        1: istore_1        2: iconst_2        3: istore_2        4: iload_1        5: iload_2        6: iadd        7: bipush        10        9: imul       10: istore_3       11: iload_3       12: ireturn
  jvm的反编译代码是什么意思呢? 我们对照着查询手册
  0: iconst_1 将int类型常量1压入操作数栈, 这句话的意思就是先把int a=1;中的1先压入操作数栈
  1: istore_1 将int类型值存入局部变量1-->意思是将int a=1; 中的a变量存入局部变量表
  注意: 这里的1不是变量的值, 他指的是局部变量的一个下标. 我们看手册上有局部变量0,1,2,3
  0表示的是this, 1表示将变量放入局部变量的第二个位置, 2表示放入第三个位置.
  对应到compute()方法,0表示的是this, 1表示的局部变量a, 2表示局部变量b,3表示局部变量c
  1: istore_1 将int类型值存入局部变量1-->意思是将int a=1; 中的a放入局部变量表的第二个位置, 然后让操作数栈中的1出栈, 赋值给a
  2: iconst_2 将int类型常量2压入栈-->意思是将int b=2;中的常量2 压入操作数栈
  3: istore_2 将int类型值存入局部变量2 -->意思是将int b=2;中的变量b存入局部变量表中第三个位置, 然后让操作数栈中的数字2出栈, 给局部变量表中的b赋值为2
  4: iload_1 从局部变量1中装载int类型值--->这句话的意思是, 将操作数1从操作数栈取出, 转入局部变量表中的a, 现在局部变量表中a=1
  要想更好的理解iload_1,我们要先来研究程序计数器。
  程序计数器
  在JVM虚拟机中,程序计数器是其中的一个组成部分。
  程序计数器是每一个线程独有的, 他用来存放马上要执行的那行代码的内存位置, 也可以叫行号. 我们看到jvm反编译代码里,都会有0 1 2 3这样的位置(如下图), 我们可以将其认为是一个标识.而程序计数器可以简单理解为是记录这些数字的. 而实际上这些数字对应的是内存里的地址
  当字节码执行引擎执行到第4行的时候,将执行到4: iload_1, 我们可以简单理解为程序计数器记录的代码位置是4. 我们的方法Math.class是放在方法区的, 由字节码执行引擎执行, 每次执行完一行代码, 字节码执行引擎都会修改程序计数器的位置, 让其向下移动一位
  java虚拟机为什么要设计程序计数器呢?
  因为多线程。当一个线程正在执行, 被另一个线程抢占了cpu, 这时之前的线程就要挂起, 当线程2执行完以后, 再执行线程1. 那么线程1之前执行到哪里了呢? 程序计数器帮我们记录了.
  下面执行这句话
  4: iload_1 从局部变量1中装载int类型值--> 意思是从局部变量表的第二个位置取出int类型的变量值, 将其放入到操作数栈中.此时程序计数器指向的是4
  5: iload_2 从局部变量2中装载int类型值-->意思是将局部变量中的第三个int类型的元素b的值取出来, 放到操作数栈, 此时程序计数器指向的是5
  6: iadd 执行int类型的加法 ---> 将两个局部变量表中的数取出, 进行加法操作, 此操作是在cpu中完成的, 将执行后的结果3在放入到操作数栈 ,此时程序计数器指向的是6
  7: bipush 10 :将一个8位带符号整数压入栈 --> 这句话的意思是将10压入操作数栈
  我们发现这里的位置是7, 但是下一个就变成了9, 那8哪里去了呢? 其实这里的0 1 2 3 ...都是对应的内存地址, 我们的乘数10也会占用内存空间, 所以, 8的位置存的是乘数10
  9: imul 执行int类型的乘法 --> 这个和iadd加法一样, 首先将操作数栈中的3和10取出来, 在cpu里面进行计算, 将计算的结果30在放回操作数栈
  乘法操作是在cpu的寄存器中进行计算的. 我们这里说的都是保存在内存中.
  10: istore_3 将int类型值存入局部变量表中 ---> 意思是是将c这个变量放入局部变量表, 然后让操作数栈中的30出栈, 赋值给变量c
  11: iload_3 从局部变量3中装载int类型值 --> 将局部变量表中取出第4个位置的值30, 装进局部变量表
  12: ireturn 从方法中返回int类型的数据 --> 最后将得到的结果c返回.
  这个方法中的变量是如何在操作数栈和局部变量表中转换的, 我们就知道了. 现在应该可以理解操作数栈和局部变量表了吧~~~
  总结:什么是操作数栈?**
  在运算的过程中, 常数1, 2, 10, 也需要有内存空间存放, 那么它存在哪里呢? 就保存在操作数栈里面
  操作数栈就是在运行的过程中, 一块临时的内存中转空间
  4.3.3 动态链接
  在之前说过什么是动态链接: 参考文章: javascript:void(0) 搜索:动态链接
  静态链接是在程序加载的时候一同被加载进来的. 通常用静态常量, 静态方法等, 因为他们在内存地址中只有一份, 所以, 为了性能, 就直接被加载进来了
  而动态链接, 是使用的时候才会被加载进来的链接, 比如compute方法. 只要在执行到math.compute()方法的时候才会真的进行加载.
  4.3.4 方法出口
  当我们运行完compute()方法以后, 还要返回到main方法的math.comput()方法的位置, 那么他怎么返回回来呢?返回回来以后该执行哪一句代码了呢?在进入compute()方法之前,就在方法出口里记录好了, 我应该如何返回,返回到哪里. 方法出口就是记录一些方法的信息的.
  五. 堆和栈的关系
  上面研究了compute()方法的栈帧空间,再来看一下main方法的栈帧空间。整体来说,都是一样的,但有一块需要说明一下,那就是局部变量表。来看看下面的代码public static void main(String[] args) {   Math math = new Math();   math.compute(); }
  ain方法的局部变量和compute()有什么区别呢? main方法中的math是一个对象. 我们知道通常对象是被创建在堆里面的. 而math是在局部变量表中, 记录的是堆中new Math对象的地址。
  说的明白一些,math里存放的不是具体的内容,而是实例对象的地址。
  那么栈和堆的关系就出来了, 如果栈中有很多new对象, 这些对象是创建在堆里面的. 栈里面存的是这些堆中创建的对象的内存地址。
  六. 方法区
  我们可以通过javap -v Math.class > Math.txt命令, 打印更详细的jvm反编译后的代码
  这次生成的代码,和使用javap -c生成的代码的区别是多了Constant pool常量池。这些常量池是放在哪里的呢?放在方法区。这里看到的常量池叫做运行时常量池。还有很多其他的常量池,比如:八大数据类型的对象常量池,字符串常量池等。
  这里主要理解运行时常量池。运行时常量池放在方法区里。
  方法区主要有哪些元素呢?
  常量 + 静态变量 + 类元信息(就是类的代码信息)1.在Math.class类中, 就有常量和静态常量
  在Math.class类中, 就有常量和静态常量public static int initData = 666; public static User user = new User();
  他们就放在方法区里面. 这里面 new User()是放在堆里面的, 在堆中分配了一个内存地址,而user对象是放在方法区里面的. 方法区中user对象指向了在堆中分配的内存空间。
  堆和方法区的关系是: 方法区中对象引用的是堆中new出来的对象的地址
  类元信息: Math.class整个类中定义的内容就是类元信息, 也放在方法区。
  七. 本地方法栈
  本地方法栈是有c++代码实现的方法. 方法名带有native的代码.
  比如:new Thread().start();
  这里的start()调用的就是本地方法
  这就是本地方法
  本地方法栈: 运行的时候也需要有内存空间去储存, 这些内存空间就是本地方法栈提供的
  每一个线程都会分配一个栈空间,本地方法栈和程序计数器。如上图main线程:包含线程栈,本地方法栈,程序计数器。
  记得关注,,转发+收藏!!!

为什么很多私立学校的升学率非常高,家长挤破头也要让孩子读私立名校,这样真的好吗?为什么私立学校升学率非常高,家长挤破头也要让儿子读私立学校,一,私立学校管理非常严格,在各个方面,文化,交通,文明礼貌,安全,我说一个简单的事,我到学校接孙子,小学生都这样称呼,爷如何让孩子的语文成绩快速提到98分以上?语文得分点有五大部分,只要掌握这五大部分的做题方法,拿98分以上一点也不难。一。基础基础题型包括看拼音写汉字,句型转换,多音字题型,形近字题型,词语积累题型等想要得高分,每一种题型三年级孩子学习不认真该不该打?以前大多数家长遵循老一辈不打不成器,棍棒底下出孝子孩子不听话,打学习成绩差,打做错事,打懒,打。实话实说打虽然能解决些问题,但有的孩子,越打反而越犟,越打反而越顽皮甚至对着干,气死如何才能做一个不吼叫的妈妈?首先,父母们会发现,在吼叫的道路上,你永远不孤单,不是你一个人是这样,但是你意识到了,祝贺你,你有了觉悟,对你和孩子来说都是一个好的开端。其次,为什么会吼叫呢?也许的因为你的父母曾老人不给看孩子,等老人不能动时,我们应该怎么对待他们?公公婆婆不给看孩子,人家没有义务。帮你是情分,不帮你也没有错,至于以后如何对待他,说白了,一没有养育之恩,二没有帮扶之情,如何对待他们,在他们没有帮你的时候他们自己就清楚了吧老人不四川省有哪些百强高中,你觉得哪个最好?谢谢邀请!四川省有哪些不错的高中,相信很多四川的朋友心目中都有自己的想法,现在就让我们一起来看看第二届中国名校校长论坛发布的2018年全国百强高中名单中,四川省有哪些高中入选?中国银行从业第10年,目前是二级支行负责人。不想继续从事银行业,想考教师资格证当老师,可行吗?您好,很高兴回答你提出的问题,针对于你提出的问题,我将用自己的见解帮你解答。银行工作10年,想要转行进入教师行业,其实从各个角度来看,并不是很赞成你的决定。因为不仅因为跨度比较大,财政部称要做好房地产税试点准备工作,你准备好了吗?你好,房产税是一定会落地的,而且是快速落地,因为地方等不起了。我翻看了2019鹤岗的预算执行情况,和2020的预算(草案)有兴趣的朋友可以去搜来看看(21年和20年的都找不到)。第全职宝妈在家带娃无任何收入,有什么合适的兼职推荐吗?有啊,我也是全职宝妈,现在正在努力挣副业的路上,西瓜视频,今日头条,抖音,现在这3个平台给我们这样没有任何收入的全职宝妈,带来了福利,每天发视频,搞创作,就可以挣工资。虽然我是新手什么叫教育?教育一般地有家庭教育学校教育社会教育还有自我教育其核心是关于生命意义猪猪真探社为你探索最真实的生活之理。在生活中,干什么都离不开教育。有人的地方,就有教育。总的来说我觉得,教育者的张庭事件对那些做微商的宝妈有多大影响?诚信。不论对人或对事,都以诚信为本,诚实办事,不要想歪门邪倒,以假充真,欺骗广大的顾客,来骗去钱财,现场是很惨的。喜欢她在穿越时空的爱恋中饰演的小玩子,演技没得说,但自从知道她做了
鉴证仁心之后来的我们中篇后来的我们(16)有些失去是注定的,有些缘份是不会有结果的。爱一个人,不一定要拥有。拥有一个人,就一定要好好去爱夜,兰桂坊介不介意,跟我说说你的心事呢?程志美终于夺下江新月手里的酒原创中华文明探源研究中的疆域与族群问题作者王坤鹏吉林大学历史系副教授习近平总书记在中共中央政治局第三十九次集体学习时指出,要把中华文明起源研究同中华文明特质和形态等重大问题研究紧密结合起来,深入研究阐释中华文明起源所昭军事战能源战,普京同时操控两场战争两场战争关键在冬季冬天是俄罗斯的朋友,也是欧洲的敌人。这句话曾有两个欧洲名人验证过,不过现在还有人想进一步加以验证。这已经不是什么秘密,尤其在能源方面,普京正在利用冬季来临反击欧洲。军事方面也有许多印度54岁富翁丧命为什么有钱人,总是死于车祸近日,印度54岁的亿万富翁去世了,作为印度最大企业之一的塔塔集团前董事长,他的死因为车祸。不是天灾人祸,也不是病魔缠身,而是车祸。车祸又把有钱人送走了。为什么会在此强调这是车祸呢?摆个小摊,胜过县官年轻人流行各种摆摊真能赚到钱?(ICphoto图)掀开汽车尾门,露出内里乾坤通过简约的装饰,带上自制的食物饮品,一个后备箱小摊就这样落地了。一段时间以来,后备箱摆摊在年轻人群体中流行起来,并蔓延到全国几乎所有城抗美援朝第一位用枪把飞机击落的一级战斗英雄您知道在抗美援朝时第一位用枪把飞机打下来的战士是谁吗?这不是神剧,这是历史,他就是有着枪神和抗美援朝最伟大战士美称的一级战斗英雄关崇贵。关崇贵出生于吉林梅河口市,长大后参军入伍,随俄气北溪1设备故障有爆炸风险点蓝字关注,不迷路来源央视财经俄罗斯天然气工业股份公司(俄气)9月5日说,北溪1天然气管道设备故障有起火和爆炸的风险。俄气当天在社交媒体发布消息说,北溪1管道波尔托瓦亚压气站现在是桃花蛇一个不规则的三角形犹如鳐鱼一般的体型,三分之一的地方是类似于头发一样一根一根的存在,粉粉嫩嫩的垂在一起,另外三分之二的身体像一片多肉,因为害怕它会咬我,只是远远的看了一眼。卖蛇的小为啥老是感觉耳朵痒痒的,用挖耳勺反而什么都掏不出来?头条创作挑战赛俗话说眼观六路,耳听八方,这不仅形容一个人的细心,也从侧面反映出耳朵的重要性,耳朵不仅可以帮助人聆听世间万物,还是重要的交流器官。不过很多人都遇到过这种情况,正在做一悼念安倍晋三遭网暴后,余文乐愈发憔悴不堪,在美国工作不再发博近日,中国香港知名男演员余文乐通过海外社交账号,晒出了一组照片记录了自己在美国生活和工作时的状态,同时还发文感谢团队嘉宾观众和赞助商,能够配合到位顺利完成拍摄任务,让他感到特别的开燃料再次泄漏美国登月火箭发射推迟因发动机故障推迟至北京时间9月4日凌晨发射的美国太空发射系统(SLS)重型火箭,再度因为燃料泄漏问题临阵取消首飞任务。美国国家航空航天局(NASA)的项目团队表示,由于液氢泄漏问题