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

(新手勿进,全是代码)ComponentScanComponentScans详解

  灵魂拷问@ComponentScan注解是做什么的? basePackages的方式和basePackageClasses的方式有什么区别?你建议用哪个?为什么? useDefaultFilters有什么用? 常见的过滤器有哪些类型?说说你知道的几个 @ComponentScan是在哪个类中处理的?说一下大概的解析过程?
  这些问题如果都ok,恭喜你,太优秀了,不知道没关系,一起来看看。 背景介绍
  到目前为止,介绍了2种注册bean的方式: xml中bean元素的方式 @Bean注解标注方法的方式
  通常情况下,项目中大部分类都需要交给spring去管理,按照上面这2种方式,代码量还是挺大的。
  为了更方便bean的注册,Spring提供了批量的方式注册bean,方便大量bean批量注册,spring中的@ComponentScan就是干这个事情的。 @ComponentScan
  @ComponentScan用于批量注册bean。
  这个注解会让spring去扫描某些包及其子包中所有的类,然后将满足一定条件的类作为bean注册到spring容器容器中。
  具体需要扫描哪些包?以及这些包中的类满足什么条件时被注册到容器中,这些都可以通过这个注解中的参数动态配置。
  先来看一下这个注解的定义: @Retention(RetentionPolicy.RUNTIME) @Target(ElementType.TYPE) @Documented @Repeatable(ComponentScans.class) //@1 public @interface ComponentScan {      @AliasFor("basePackages")     String[] value() default {};      @AliasFor("value")     String[] basePackages() default {};      Class<?>[] basePackageClasses() default {};      Class<? extends BeanNameGenerator> nameGenerator() default BeanNameGenerator.class;      Class<? extends ScopeMetadataResolver> scopeResolver() default AnnotationScopeMetadataResolver.class;      ScopedProxyMode scopedProxy() default ScopedProxyMode.DEFAULT;      String resourcePattern() default "**/*.class";      boolean useDefaultFilters() default true;      Filter[] includeFilters() default {};      Filter[] excludeFilters() default {};      boolean lazyInit() default false; } 定义上可以看出此注解可以用在任何类型上面,不过我们通常将其用在类上面。
  常用参数:
  value:指定需要扫描的包,如:com.javacode2018
  basePackages:作用同value;value和basePackages不能同时存在设置,可二选一
  basePackageClasses:指定一些类,spring容器会扫描这些类所在的包及其子包中的类
  nameGenerator:自定义bean名称生成器
  resourcePattern:需要扫描包中的那些资源,默认是:**/*.class,即会扫描指定包中所有的class文件
  useDefaultFilters:对扫描的类是否启用默认过滤器,默认为true
  includeFilters:过滤器:用来配置被扫描出来的那些类会被作为组件注册到容器中
  excludeFilters:过滤器,和includeFilters作用刚好相反,用来对扫描的类进行排除的,被排除的类不会被注册到容器中
  lazyInit:是否延迟初始化被注册的bean
  @1:@Repeatable(ComponentScans.class),这个注解可以同时使用多个。
  @ComponentScan工作的过程:
  1. Spring会扫描指定的包,且会递归下面子包,得到一批类的数组
  2. 然后这些类会经过上面的各种过滤器,最后剩下的类会被注册到容器中
  所以玩这个注解,主要关注2个问题:
  第一个:需要扫描哪些包?通过 value、backPackages、basePackageClasses  这3个参数来控制
  第二:过滤器有哪些?通过 useDefaultFilters、includeFilters、excludeFilters  这3个参数来控制过滤器
  这两个问题搞清楚了,就可以确定哪些类会被注册到容器中。
  默认情况下,任何参数都不设置的情况下,此时,会将@ComponentScan修饰的类所在的包作为扫描包;默认情况下useDefaultFilters为true,这个为true的时候,spring容器内部会使用默认过滤器,规则是:凡是类上有 @Repository、@Service、@Controller、@Component  这几个注解中的任何一个的,那么这个类就会被作为bean注册到spring容器中,所以默认情况下,只需在类上加上这几个注解中的任何一个,这些类就会自动交给spring容器来管理了。 @Component、@Repository、@Service、@Controller
  这几个注解都是spring提供的。
  先说一下@Component  这个注解,看一下其定义: @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) @Documented @Indexed public @interface Component {     String value() default ""; } 从定义中可以看出,这个注解可以用在任何类型上面。
  通常情况下将这个注解用在类上面,标注这个类为一个组件,默认情况下,被扫描的时候会被作为bean注册到容器中。
  value参数:被注册为bean的时候,用来指定bean的名称,如果不指定,默认为类名首字母小写。如:类UserService对应的beanname为userService
  再来看看@Repository  源码如下: @Target({ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Documented @Component public @interface Repository {      @AliasFor(annotation = Component.class)     String value() default "";  } Repository上面有@Component注解。
  value参数上面有  @AliasFor(annotation = Component.class)  ,设置value参数的时候,也相当于给@Component  注解中的value设置值。
  其他两个注解@Service、@Controller  源码和@Repository  源码类似。
  这4个注解本质上是没有任何差别,都可以用在类上面,表示这个类被spring容器扫描的时候,可以作为一个bean组件注册到spring容器中。
  spring容器中对这4个注解的解析并没有进行区分,统一采用@Component  注解的方式进行解析,所以这几个注解之间可以相互替换。
  spring提供这4个注解,是为了让系统更清晰,通常情况下,系统是分层结构的,多数系统一般分为controller层、service层、dao层。
  @controller通常用来标注controller层组件,@service注解标注service层的组件,@Repository标注dao层的组件,这样可以让整个系统的结构更清晰,当看到这些注解的时候,会和清晰的知道属于哪个层,对于spring来说,将这3个注解替换成@Component注解,对系统没有任何影响,产生的效果是一样的。
  下面通过案例来感受@ComponentScan各种用法。 案例1:任何参数未设置UserControllerpackage com.javacode2018.lesson001.demo22.test1.controller;  import org.springframework.stereotype.Controller;  @Controller public class UserController { } UserServicepackage com.javacode2018.lesson001.demo22.test1.service;  import org.springframework.stereotype.Service;  @Service public class UserService { } UserDaopackage com.javacode2018.lesson001.demo22.test1.dao;  import org.springframework.stereotype.Repository;  @Repository public class UserDao { } UserModelpackage com.javacode2018.lesson001.demo22.test1;  import org.springframework.stereotype.Component;  @Component public class UserModel { }
  上面几个类中,分别使用了4种注解。 @CompontentScan修饰的类package com.javacode2018.lesson001.demo22.test1;  import org.springframework.context.annotation.ComponentScan;  @ComponentScan public class ScanBean1 { } 上面几个类的结构图
  测试用例package com.javacode2018.lesson001.demo22;  import com.javacode2018.lesson001.demo22.test1.ScanBean1; import org.junit.Test; import org.springframework.context.annotation.AnnotationConfigApplicationContext;  public class ComponentScanTest {      @Test     public void test1() {         AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ScanBean1.class);         for (String beanName : context.getBeanDefinitionNames()) {             System.out.println(beanName + "->" + context.getBean(beanName));         }     } } @1:使用AnnotationConfigApplicationContext作为ioc容器,将  ScanBean  作为参数传入。
  默认会扫描  ScanBean  类所在的包中的所有类,类上有@Component、@Repository、@Service、@Controller任何一个注解的都会被注册到容器中 运行输出
  部分输出如下: userModel->com.javacode2018.lesson001.demo22.test1.UserModel@595b007d userController->com.javacode2018.lesson001.demo22.test1.controller.UserController@72d1ad2e userDao->com.javacode2018.lesson001.demo22.test1.dao.UserDao@2d7275fc userService->com.javacode2018.lesson001.demo22.test1.service.UserService@399f45b1
  注意最后4行这几个bean,都被注册成功了。   案例2:指定需要扫描的包
  指定需要扫毛哪些包,可以通过value或者basePackage来配置,二者选其一,都配置运行会报错,下面我们通过value来配置。 ScanBean2package com.javacode2018.lesson001.demo22.test2;  import org.springframework.context.annotation.ComponentScan;  @ComponentScan({         "com.javacode2018.lesson001.demo22.test1.controller",         "com.javacode2018.lesson001.demo22.test1.service" }) public class ScanBean2 { }
  上面指定了2需要扫描的包,这两个包中有2个类。   测试用例
  ComponentScanTest中新增个方法 @Test public void test2() {     AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ScanBean2.class);     for (String beanName : context.getBeanDefinitionNames()) {         System.out.println(beanName + "->" + context.getBean(beanName));     } } 运行输出
  截取了关键几行如下: userController->com.javacode2018.lesson001.demo22.test1.controller.UserController@dd8ba08 userService->com.javacode2018.lesson001.demo22.test1.service.UserService@245b4bdc
  可以看出只有controller包和service包中的2个类被注册为bean了。 注意
  指定包名的方式扫描存在的一个隐患,若包被重名了,会导致扫描会失效,一般情况下面我们使用basePackageClasses的方式来指定需要扫描的包,这个参数可以指定一些类型,默认会扫描这些类所在的包及其子包中所有的类,这种方式可以有效避免这种问题。
  下面来看一下basePackageClasses的方式。 案例:basePackageClasses指定扫描范围
  我们可以在需要扫描的包中定义一个标记的接口或者类,他们的唯一的作用是作为basePackageClasses的值,其他没有任何用途。 下面我们定义这样一个接口package com.javacode2018.lesson001.demo22.test6.beans;  public interface ScanClass { } 再来定义2个类,用@Component注解标记package com.javacode2018.lesson001.demo22.test6.beans;  import org.springframework.stereotype.Component;  @Component public class Service1 { } package com.javacode2018.lesson001.demo22.test6.beans;  import org.springframework.stereotype.Component;  @Component public class Service2 { } 来一个@CompontentScan标记的类package com.javacode2018.lesson001.demo22.test6;  import com.javacode2018.lesson001.demo22.test6.beans.ScanClass; import org.springframework.context.annotation.ComponentScan;  @ComponentScan(basePackageClasses = ScanClass.class) public class ScanBean6 { } 测试用例
  ComponentScanTest中新增个方法 @Test public void test6() {     AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ScanBean6.class);     for (String beanName : context.getBeanDefinitionNames()) {         System.out.println(beanName + "->" + context.getBean(beanName));     } } 运行输出service1->com.javacode2018.lesson001.demo22.test6.beans.Service1@79924b service2->com.javacode2018.lesson001.demo22.test6.beans.Service2@7b9a4292 includeFilters的使用用法
  再来看一下includeFilters这个参数的定义: Filter[] includeFilters() default {};
  是一个Filter  类型的数组,多个Filter之间为或者关系,即满足任意一个就可以了 ,看一下Filter  的代码: @Retention(RetentionPolicy.RUNTIME) @Target({}) @interface Filter {      FilterType type() default FilterType.ANNOTATION;      @AliasFor("classes")     Class<?>[] value() default {};      @AliasFor("value")     Class<?>[] classes() default {};      String[] pattern() default {};  } 可以看出Filter也是一个注解,参数:
  type:过滤器的类型,是个枚举类型,5种类型
  ANNOTATION:通过注解的方式来筛选候选者,即判断候选者是否有指定的注解
  ASSIGNABLE_TYPE:通过指定的类型来筛选候选者,即判断候选者是否是指定的类型
  ASPECTJ:ASPECTJ表达式方式,即判断候选者是否匹配ASPECTJ表达式
  REGEX:正则表达式方式,即判断候选者的完整名称是否和正则表达式匹配
  CUSTOM:用户自定义过滤器来筛选候选者,对候选者的筛选交给用户自己来判断
  value:和参数classes效果一样,二选一
  classes:3种情况如下
  当type=FilterType.ANNOTATION时,通过classes参数可以指定一些注解,用来判断被扫描的类上是否有classes参数指定的注解
  当type=FilterType.ASSIGNABLE_TYPE时,通过classes参数可以指定一些类型,用来判断被扫描的类是否是classes参数指定的类型
  当type=FilterType.CUSTOM时,表示这个过滤器是用户自定义的,classes参数就是用来指定用户自定义的过滤器,自定义的过滤器需要实现org.springframework.core.type.filter.TypeFilter接口
  pattern:2种情况如下
  当type=FilterType.ASPECTJ时,通过pattern来指定需要匹配的ASPECTJ表达式的值
  当type=FilterType.REGEX时,通过pattern来自正则表达式的值   案例:扫描包含注解的类需求
  我们自定义一个注解,让标注有这些注解的类自动注册到容器中 代码实现
  下面的代码都在com.javacode2018.lesson001.demo22.test3  包中。 定义一个注解package com.javacode2018.lesson001.demo22.test3;  import java.lang.annotation.*;  @Documented @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) public @interface MyBean { } 创建一个类,使用这个注解标注package com.javacode2018.lesson001.demo22.test3;  @MyBean public class Service1 { } 再来一个类,使用spring中的`@Compontent`标注package com.javacode2018.lesson001.demo22.test3;  import org.springframework.stereotype.Component;  @Component public class Service2 { } 再来一个类,使用@CompontentScan标注package com.javacode2018.lesson001.demo22.test3;  import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.FilterType;  @ComponentScan(includeFilters = {         @ComponentScan.Filter(type = FilterType.ANNOTATION, classes = MyBean.class) }) public class ScanBean3 { }
  上面指定了Filter的type为注解的类型,只要类上面有  @MyBean  注解的,都会被作为bean注册到容器中。 测试用例
  ComponentScanTest中新增个测试用例 @Test public void test3() {     AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ScanBean3.class);     for (String beanName : context.getBeanDefinitionNames()) {         System.out.println(beanName + "->" + context.getBean(beanName));     } } 运行输出,截取了主要的几行service1->com.javacode2018.lesson001.demo22.test3.Service1@6b81ce95 service2->com.javacode2018.lesson001.demo22.test3.Service2@2a798d51
  Service1上标注了 @MyBean  注解,被注册到容器了,但是Service2  上没有标注@MyBean  啊,怎么也被注册到容器了?
  原因:Service2上标注了 @Compontent  注解,而@CompontentScan注解中的useDefaultFilters  默认是true  ,表示也会启用默认的过滤器,而默认的过滤器会将标注有@Component、@Repository、@Service、@Controller  这几个注解的类也注册到容器中
  如果我们只想将标注有@MyBean  注解的bean注册到容器,需要将默认过滤器关闭,即:useDefaultFilters=false,我们修改一下ScanBean3的代码如下: @ComponentScan(         useDefaultFilters = false, //不启用默认过滤器         includeFilters = {                 @ComponentScan.Filter(type = FilterType.ANNOTATION, classes = MyBean.class)         }) public class ScanBean3 { }
  再次运行test3  输出: service1->com.javacode2018.lesson001.demo22.test3.Service1@294425a7 扩展:自定义注解支持定义bean名称
  上面的自定义的@MyBean注解,是无法指定bean的名称的,可以对这个注解做一下改造,加个value参数来指定bean的名称,如下: @Documented @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) @Component //@1 public @interface MyBean {     @AliasFor(annotation = Component.class) //@2     String value() default ""; //@3 } 重点在于@1和@2这2个地方的代码,通过上面的参数可以间接给@Component注解中的value设置值。
  这块用到了@AliasFor注解,对这块不了解的,可以去看一下:java注解详解及spring对注解的增强
  修改一下Service1的代码: @MyBean("service1Bean") public class Service1 { }
  运行test3用例输出: service1Bean->com.javacode2018.lesson001.demo22.test3.Service1@222545dc
  此时bean名称就变成了service1Bean  。 案例:包含指定类型的类
  下面的代码都位于com.javacode2018.lesson001.demo22.test4  包中。 来个接口package com.javacode2018.lesson001.demo22.test4;  public interface IService { }
  让spring来进行扫描,类型满足IService的都将其注册到容器中。 来2个实现类package com.javacode2018.lesson001.demo22.test4;  public class Service1 implements IService { } package com.javacode2018.lesson001.demo22.test4;  public class Service2 implements IService { } 来一个@CompontentScan标注的类package com.javacode2018.lesson001.demo22.test4;  import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.FilterType;  @ComponentScan(         useDefaultFilters = false, //不启用默认过滤器         includeFilters = {                 @ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE, classes = IService.class) //@1         }) public class ScanBean4 { }
  @1:被扫描的类满足  IService.class.isAssignableFrom(被扫描的类)  条件的都会被注册到spring容器中 来个测试用例
  ComponentScanTest中新增个测试用例 @Test public void test4() {     AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ScanBean4.class);     for (String beanName : context.getBeanDefinitionNames()) {         System.out.println(beanName + "->" + context.getBean(beanName));     } } 运行输出service1->com.javacode2018.lesson001.demo22.test4.Service1@6379eb service2->com.javacode2018.lesson001.demo22.test4.Service2@294425a7 自定义Filter用法
  有时候我们需要用到自定义的过滤器,使用自定义过滤器的步骤: 1.设置@Filter中type的类型为:FilterType.CUSTOM 2.自定义过滤器类,需要实现接口:org.springframework.core.type.filter.TypeFilter 3.设置@Filter中的classses为自定义的过滤器类型
  来看一下TypeFilter  这个接口的定义: @FunctionalInterface public interface TypeFilter {     boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory)             throws IOException; }
  是一个函数式接口,包含一个match方法,方法返回boolean类型,有2个参数,都是接口类型的,下面介绍一下这2个接口。 MetadataReader接口
  类元数据读取器,可以读取一个类上的任意信息,如类上面的注解信息、类的磁盘路径信息、类的class对象的各种信息,spring进行了封装,提供了各种方便使用的方法。
  看一下这个接口的定义: public interface MetadataReader {      /**      * 返回类文件的资源引用      */     Resource getResource();      /**      * 返回一个ClassMetadata对象,可以通过这个读想获取类的一些元数据信息,如类的class对象、是否是接口、是否有注解、是否是抽象类、父类名称、接口名称、内部包含的之类列表等等,可以去看一下源码      */     ClassMetadata getClassMetadata();      /**      * 获取类上所有的注解信息      */     AnnotationMetadata getAnnotationMetadata();  } MetadataReaderFactory接口
  类元数据读取器工厂,可以通过这个类获取任意一个类的MetadataReader对象。
  源码: public interface MetadataReaderFactory {      /**      * 返回给定类名的MetadataReader对象      */     MetadataReader getMetadataReader(String className) throws IOException;      /**      * 返回指定资源的MetadataReader对象      */     MetadataReader getMetadataReader(Resource resource) throws IOException;  } 自定义Filter案例需求
  我们来个自定义的Filter,判断被扫描的类如果是IService  接口类型的,就让其注册到容器中。 代码实现
  来个自定义的TypeFilter类: package com.javacode2018.lesson001.demo22.test5;  import com.javacode2018.lesson001.demo22.test4.IService; import org.springframework.core.type.ClassMetadata; import org.springframework.core.type.classreading.MetadataReader; import org.springframework.core.type.classreading.MetadataReaderFactory; import org.springframework.core.type.filter.TypeFilter;  import java.io.IOException;  public class MyFilter implements TypeFilter {     /**      * @param metadataReader      * @param metadataReaderFactory      * @return      * @throws IOException      */     @Override     public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory) throws IOException {         Class curClass = null;         try {             //当前被扫描的类             curClass = Class.forName(metadataReader.getClassMetadata().getClassName());         } catch (ClassNotFoundException e) {             e.printStackTrace();         }         //判断curClass是否是IService类型         boolean result = IService.class.isAssignableFrom(curClass);         return result;     } } 来一个@CompontentScan标注的类package com.javacode2018.lesson001.demo22.test5;  import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.FilterType;  @ComponentScan(         basePackages = {"com.javacode2018.lesson001.demo22.test4"},         useDefaultFilters = false, //不启用默认过滤器         includeFilters = {                 @ComponentScan.Filter(type = FilterType.CUSTOM, classes = MyFilter.class) //@1         }) public class ScanBean5 { }
  @1:type为FilterType.CUSTOM,表示Filter是用户自定义的,classes为自定义的过滤器   再来个测试用例
  ComponentScanTest中新增个测试用例 @Test public void test5() {     AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ScanBean5.class);     for (String beanName : context.getBeanDefinitionNames()) {         System.out.println(beanName + "->" + context.getBean(beanName));     } } 运行输出service1->com.javacode2018.lesson001.demo22.test4.Service1@4cc451f2 service2->com.javacode2018.lesson001.demo22.test4.Service2@6379eb excludeFilters
  配置排除的过滤器,满足这些过滤器的类不会被注册到容器中,用法上面和includeFilters用一样,这个我就不演示了,可以自己玩玩  @ComponentScan重复使用
  从这个注解的定义上可以看出这个注解可以同时使用多个,如: @ComponentScan(basePackageClasses = ScanClass.class) @ComponentScan(         useDefaultFilters = false, //不启用默认过滤器         includeFilters = {                 @ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE, classes = IService.class)         }) public class ScanBean7 { }
  还有一种写法,使用@ComponentScans的方式: @ComponentScans({         @ComponentScan(basePackageClasses = ScanClass.class),         @ComponentScan(                 useDefaultFilters = false, //不启用默认过滤器                 includeFilters = {                         @ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE, classes = IService.class)                 })}) public class ScanBean7 { } Spring中这块的源码
  @CompontentScan注解是被下面这个类处理的  org.springframework.context.annotation.ConfigurationClassPostProcessor
  这个类非常非常关键,主要用户bean的注册,前面我们介绍的@Configuration,@Bean注解也是被这个类处理的。
  还有下面这些注解: @PropertySource @Import @ImportResource @Compontent
  以上这些注解都是被ConfigurationClassPostProcessor这个类处理的,内部会递归处理这些注解,完成bean的注册。
  以@CompontentScan来说一下过程,第一次扫描之后会得到一批需要注册的类,然后会对这些需要注册的类进行遍历,判断是否有上面任意一个注解,如果有,会将这个类交给ConfigurationClassPostProcessor继续处理,直到递归完成所有bean的注册。
  想成为高手,这个类是必看的。  总结@ComponentScan用于批量注册bean,spring会按照这个注解的配置,递归扫描指定包中的所有类,将满足条件的类批量注册到spring容器中 可以通过value、basePackages、basePackageClasses 这几个参数来配置包的扫描范围 可以通过useDefaultFilters、includeFilters、excludeFilters这几个参数来配置类的过滤器,被过滤器处理之后剩下的类会被注册到容器中 指定包名的方式配置扫描范围存在隐患,包名被重命名之后,会导致扫描实现,所以一般我们在需要扫描的包中可以创建一个标记的接口或者类,作为basePackageClasses的值,通过这个来控制包的扫描范围 @CompontScan注解会被ConfigurationClassPostProcessor类递归处理,最终得到所有需要注册的类。 作者:路人甲Java
  链接:https://mp.weixin.qq.com/s/MPMouZPsdLww7RgDZNtzLQ

意天空姆巴佩在美国踢橄榄球,公羊队巨星库珀库普为他扶球直播吧6月29日讯据意大利天空体育报道,姆巴佩目前正在美国度假,他还尝试了踢橄榄球。姆巴佩在今夏来到美国度假,此前他观看了NBA的选秀大会,又现场看了美职联洛杉矶FC对纽约红牛的比夏季口腔溃疡高发,疼的厉害怎么办?夏季天气炎热,正是口腔溃疡的高发季。口腔溃疡俗称口疮,是口腔黏膜的一种常见疾病,一旦发作,疼起来往往让人苦不堪言又生无可恋。在许多人看来,患口腔溃疡是件小事,忍一段时间就好了。然而索尼布局激光雷达获进展SPAD传感器即将出货国内赛道已现华为宁王身影科创板日报29日讯(编辑邱思雨),据日本经济新闻报道,索尼将于2023年开始量产车载传感器LiDAR零部件。索尼表示,继车载摄像头图像传感器之后,将致力于LiDAR零部件。这将是该李崇霄一路走好,天堂没有伤痛当听到李崇霄逝去的消息,心里真不是滋味,51岁的青春年华就此终结,怎能不让人心痛。她女儿的一席话更加刺痛众多网民的心以后的六月再也没无法和您过父亲节了,女儿惟有泪两行,感谢对女儿的红米K50出现最强竞争对手,6400万三摄只要1799红米K50搭载天玑8100以出色的口碑,稳居中端手机销量排行榜前列!那么此次敢于挑战红米K50销量口碑的手机是哪一部,为什么它被称为红米K50最强的对手。首先来介绍一下红米K50的家的感觉?图片报塞内加尔国家体育场几乎是安联的翻版拜仁慕尼黑新援马内最近刚刚加盟俱乐部,尽管新的国家新的联赛和新的球队需要他花时间来适应,但是拜仁慕尼黑的安联球场对于马内来说一定不会太陌生,因为在他的家乡塞内加尔也有一座翻版的安联三伏天常饮绿豆汤,解暑去火,如何喝上绿豆汤?简单4步一年中最热最难熬的时段三伏天即将出现,各地将进入持续高温的模式,人体受到暑气影响,容易感到烦躁没有食欲倦怠懒散等,每到这个时候,就很适合来一碗可口的绿豆汤。特别是对于肝脏有问题的人滑倒的传球,疑似犯规后的点球,嵩山龙门不是输给了山东而是运气河南队与山东队的二番战,最后的结果是中原铁军输球,但是看过比赛后,河南球迷都有点不甘心,首先就是多拉多的身体不适,在赛前高挂免战牌,在阵容上,哈维尔的球队就很吃亏,随后就是看似合理山东队多了两个新面孔,张云松透露一个计划,两名球员试训福建队山东队多了两个新面孔,张云松透露一个计划,两名球员试训福建队CBA休赛期最忙乎的恐怕就是球员签约这件事了,联盟就是一个生意场,每一支球队为了各自的利益,必然要对球队作出新的调整,该巩晓彬疑似帮山东招募小高,夸杨鸣很帅,评价郭艾伦说得很直白巩晓彬疑似帮山东招募小高,夸杨鸣很帅,评价郭艾伦说得很直白山东男篮原主帅巩晓彬,最近是连续直播,在和广东网友的互动中,巩指导是敞开心扉,透露了很多鲜为人知的故事,话说和王治郅在国家CBA三消息王薪凯加盟山东,林书豪无球可打,浙江男篮受到重创王薪凯加盟山东男篮。北京时间6月29日,CBA休赛期的第二个月,目前各支球队进行到紧张的补强集训当中,广东男篮离队第一人正式出炉,虽然目前广东男篮方面还没有正式官宣,但王薪凯已经被
地球内核开始反着转会对我们的生活产生多大影响?科普中国微信公号原标题惊爆!地球这里开始反着转!会对我们的生活产生多大影响?北京大学地球与空间科学学院杨翼博士与宋晓东教授最近在自然地球科学国际期刊上发表论文,研究发现在过去十年左电脑被勒索病毒攻击了,软件无法运行怎么办电脑病毒有很多种,主要就是黑客用来攻击用户服务器的恶意代码,来瘫痪计算机系统,或是获取用户账号密码,或是获取数据。而勒索病毒则是最为让人讨厌和恐惧的一种电脑病毒。勒索病毒并不会对计贾跃亭还钱了,一次性偿还近6亿,融资1。35亿美元!FF91将量产!形势不断发生改变,商场是没有硝烟的战场,各个行业的发展变幻莫测,或许有些商业大佬在一天之内从云端跌到谷底。贾跃亭曾是大家眼中的商业大佬,但是他背负了债务,由于短时间内未偿还,口碑直风向变了?杀疯了的ChatGPT引发争议,阿里冷静的可怕近段时间,OpenAI发布的ChatGPT产品火爆全球,风靡一时。短短2个月时间,其月活跃用户就突破了1亿。但随着ChatGPT的热度持续上涨,杀疯了的ChatGPT也暴露了其漏洞台积电内部的猛料爆出,美半导体的落后原因找到了!头条创作挑战赛台积电内部的猛料爆出,美半导体的落后原因不在技术上!一提起芯片规则的实施,很多的国人都是咬牙切齿,但实际上凡事都有两面性,老美用尽全力去限制中国半导体产业,实则是为了吴金贵新投资人来后俱乐部发展会越来越好,今年目标会更远直播吧2月24日讯今天,上海申花时隔一年在康桥体育基地再次开放训练,基地目前已经更名为久事康桥体育基地。据西北望看台报道,申花主帅吴金贵在接受采访时表示,新的投资人来了之后,从资金中老年男性的眉毛变长,通常意味着什么?医生原因要心里有数人体五官耳眉口鼻眼,五官对于我们的面部形象起到非常关键的作用。耳朵可以修饰面部脸颊,嘴巴可以起到衬托作用,鼻子可以使脸更加精致,眼睛使我们看起来更加有精气!而眉毛对我们的作用更是不欧联杯16强对阵揭晓曼联再战西甲球队,阿森纳面对葡超劲旅北京时间2月24日晚上,欧联杯16强抽签仪式进行,8支小组阶段头名球队和8支通过附加赛晋级的队伍会师本赛季欧联杯16强。欧联杯16强抽签遵循同协会球队回避原则,小组赛阶段头名球队先A股今天A股十分反常,这背后是有原因的,下周行情会这样走今天大盘的杀伤力确实非常的强,我相信绝大部分的兄弟们都出现了较大的亏损。而今天大盘下跌是有原因的,一个是北向资金大幅度净流出,而且是持续性的流出状态。另外,大股东非坚持,特别是券商为什么正骨店如此受欢迎?探究人们喜欢去正骨店的原因正骨按摩作为一种传统的治疗方法,近年来越来越受到人们的青睐,正骨店也随之而兴起。那么,为什么越来越多的人喜欢去正骨店呢?下面就让我们一起探究一下人们喜欢去正骨店的原因。缓解身体不适穆帅笑了!20罗马逆袭,闯入欧联淘汰赛正赛,6万现场球迷沸腾了赛前在意甲联赛排名第3的罗马队迎来了欧联淘汰赛附加赛第二回合的较量,罗马队此役坐镇自己的主场(能容纳6万多人的罗马奥林匹克球场)PK奥超领头羊萨尔茨堡红牛队。首回合罗马队没有发挥出