我被面试官给虐懵了,竟然是因为我不懂Spring中的@Configuration

作者 l cxuan

来源 l Hollis(ID:hollischuang)

现在大部分的Spring项目都采用了基于注解的配置,采用了@Configuration 替换标签的做法。一行简单的注解就可以解决很多事情。但是,其实每一个注解背后都有很多值得学习和思考的内容。这些思考的点也是很多大厂面试官喜欢问的内容。

在一次关于Spring注解的面试中,可能会经历面试官的一段夺命连环问:

@Configuration有什么用?

@Configuration和XML有什么区别?哪种好?

Spring是如何基于来获取Bean的定义的?

@Autowired 、 @Inject、@Resource 之间有什么区别?

@Value、@PropertySource 和 @Configuration?

Spring如何处理带@Configuration @Import的类?

@Profile有什么用?

@Configuration 如何嵌套?

Spring如何对Bean进行延迟初始化?

Spring项目怎么进行单元测试?

@Configuration 使用上有哪些约束?

本文就来尝试回答下以上问题。简单介绍下@Configuration 注解,并且你看一下他的基本用法以及和其他注解产生化学反应。文章内容较长,建议收藏。

@Configuration 基本说明

定义:指示一个类声明一个或者多个@Bean 声明的方法并且由Spring容器统一管理,以便在运行时为这些bean生成bean的定义和服务请求的类。例如:

@Configuration
public class AppConfig {@Beanpublic MyBean myBean(){return new MyBean();}
}

上述AppConfig 加入@Configuration 注解,表明这就是一个配置类。有一个myBean()的方法,返回一个MyBean()的实例,并用@Bean 进行注释,表明这个方法是需要被Spring进行管理的bean。@Bean 如果不指定名称的话,默认使用myBean名称,也就是小写的名称。

通过注解启动

通过启动一个AnnotationConfigApplicationContext 来引导这个@Configuration 注解的类,比如:

AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext();
ctx.register(AppConfig.class);
ctx.refresh();

在web项目中,也可以使用AnnotationContextWebApplicationContext或者其他变体来启动。

新建一个SpringBoot项目(别问我为什么,因为这样创建项目比较快)。

pom.xml 文件如下:

<?xml version="1.0" encoding="UTF-8"?>
<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><parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.1.5.RELEASE</version><relativePath/> <!-- lookup parent from repository --></parent><groupId>com.spring.configuration</groupId><artifactId>spring-configuration</artifactId><version>0.0.1-SNAPSHOT</version><name>spring-configuration</name><description>Demo project for Spring Boot</description><properties><java.version>1.8</java.version></properties><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>5.0.6.RELEASE</version></dependency></dependencies><build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin></plugins></build></project>

在config 包下新建一个MyConfiguration环境配置,和上面的示例代码相似,完整的代码如下:

@Configuration
public class MyConfiguration {@Beanpublic MyBean myBean(){System.out.println("myBean Initialized");return new MyBean();}
}

说明MyConfiguration 是一个配置类,能够在此类下面声明管理多个Bean,我们声明了一个MyBean的bean,希望它被容器加载和管理。

在pojo包下新建一个MyBean的类,具体代码如下

public class MyBean {public MyBean(){System.out.println("generate MyBean Instance");}public void init(){System.out.println("MyBean Resources Initialized");}
}

新建一个SpringConfigurationApplication类,用来测试MyConfiguration类,具体代码如下:

public class SpringConfigurationApplication {public static void main(String[] args) {
//        AnnotationConfigApplicationContext context = = new AnnotationConfigApplicationContext(MyConfiguration.class)// 因为我们加载的@Configuration 是基于注解形式的,所以需要创建AnnotationConfigApplicationContextAnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();// 注册MyConfiguration 类并刷新bean 容器。context.register(MyConfiguration.class);context.refresh();}
}

输出:

myBean Initialized
generate MyBean Instance


从输出的结果可以看到,默认名称为myBean 的bean随着容器的加载而加载,因为myBean方法返回一个myBean的构造方法,所以myBean被初始化了。

通过XML 的方式来启动

可以通过使用XML方式定义的开启基于注解的启动,然后再定义一个MyConfiguration的bean,在/resources 目录下新建 application-context.xml 代码如下:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"xmlns:util="http://www.springframework.org/schema/util"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsdhttp://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-4.2.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.2.xsd"
><!-- 相当于基于注解的启动类 AnnotationConfigApplicationContext--><context:annotation-config /><bean class="com.spring.configuration.config.MyConfiguration"/></beans>

需要引入applicationContext.xml ,在SpringConfigurationApplication 需要进行引入,修改后的SpringConfigurationApplication如下:

public class SpringConfigurationApplication {public static void main(String[] args) {ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");}
}

输出:

myBean Initialized
generate MyBean Instance

基于ComponentScan() 来获取Bean的定义

@Configuration 使用@Component 进行原注解,因此@Configuration 类也可以被组件扫描到(特别是使用XML元素)。

在这里认识几个注解: @Controller, @Service, @Repository, @Component

  • @Controller: 表明一个注解的类是一个"Controller",也就是控制器,可以把它理解为MVC 模式的Controller 这个角色。这个注解是一个特殊的@Component,允许实现类通过类路径的扫描扫描到。它通常与@RequestMapping 注解一起使用。

  • @Service: 表明这个带注解的类是一个"Service",也就是服务层,可以把它理解为MVC 模式中的Service层这个角色,这个注解也是一个特殊的@Component,允许实现类通过类路径的扫描扫描到

  • @Repository: 表明这个注解的类是一个"Repository",团队实现了JavaEE 模式中像是作为"Data Access Object" 可能作为DAO来使用,当与 PersistenceExceptionTranslationPostProcessor 结合使用时,这样注释的类有资格获得Spring转换的目的。这个注解也是@Component 的一个特殊实现,允许实现类能够被自动扫描到

  • @Component: 表明这个注释的类是一个组件,当使用基于注释的配置和类路径扫描时,这些类被视为自动检测的候选者。

也就是说,上面四个注解标记的类都能够通过@ComponentScan 扫描到,上面四个注解最大的区别就是使用的场景和语义不一样,比如你定义一个Service类想要被Spring进行管理,你应该把它定义为@Service 而不是@Controller因为我们从语义上讲,@Service更像是一个服务的类,而不是一个控制器的类,@Component通常被称作组件,它可以标注任何你没有严格予以说明的类,比如说是一个配置类,它不属于MVC模式的任何一层,这个时候你更习惯于把它定义为 @Component。@Controller,@Service,@Repository 的注解上都有@Component,所以这三个注解都可以用@Component进行替换。

来看一下代码进行理解:

定义五个类,类上分别用@Controller, @Service, @Repository, @Component, @Configuration 进行标注,分别如下

@Component
public class UserBean {}@Configuration
public class UserConfiguration {}@Controller
public class UserController {}@Repository
public class UserDao {}@Service
public class UserService {}

在MyConfiguration上加上@ComponentScan 注解,扫描上面5个类所在的包位置。代码如下:

@Configuration
@ComponentScan(basePackages = "com.spring.configuration.pojo")
public class MyConfiguration {@Beanpublic MyBean myBean(){System.out.println("myBean Initialized");return new MyBean();}
}

修改 SpringConfigurationApplication 中的代码,如下:

public class SpringConfigurationApplication {public static void main(String[] args) {//        AnnotationConfigApplicationContext context = = new AnnotationConfigApplicationContext(MyConfiguration.class)
//        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();context.register(MyConfiguration.class);context.refresh();// 获取启动过程中的bean 定义的名称for(String str : context.getBeanDefinitionNames()){System.out.println("str = " + str);}context.close();}
}

输出:

myBean Initialized
generate MyBean Instance
str = org.springframework.context.annotation.internalConfigurationAnnotationProcessor
str = org.springframework.context.annotation.internalAutowiredAnnotationProcessor
str = org.springframework.context.annotation.internalRequiredAnnotationProcessor
str = org.springframework.context.annotation.internalCommonAnnotationProcessor
str = org.springframework.context.event.internalEventListenerProcessor
str = org.springframework.context.event.internalEventListenerFactory
str = myConfiguration
str = userBean
str = userConfiguration
str = userController
str = userDao
str = userService
str = myBean

由输出可以清楚的看到,上述定义的五个类成功被@ComponentScan 扫描到,并在程序启动的时候进行加载。

@Configuration 和 Environment

@Configuration 通常和Environment 一起使用,通过@Environment 解析的属性驻留在一个或多个"属性源"对象中,@Configuration类可以使用@PropertySource,像Environment 对象提供属性源

为了便于测试,我们引入junit4和spring-test 的依赖,完整的配置文件如下

<?xml version="1.0" encoding="UTF-8"?>
<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><parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.1.5.RELEASE</version><relativePath/> <!-- lookup parent from repository --></parent><groupId>com.spring.configuration</groupId><artifactId>spring-configuration</artifactId><version>0.0.1-SNAPSHOT</version><name>spring-configuration</name><description>Demo project for Spring Boot</description><properties><java.version>1.8</java.version><spring.version>5.0.6.RELEASE</spring.version><spring.test.version>4.3.13.RELEASE</spring.test.version><junit.version>4.12</junit.version></properties><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>${spring.version}</version></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-test</artifactId><version>${spring.test.version}</version></dependency><dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>${junit.version}</version></dependency></dependencies><build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin></plugins></build></project>

在config 包下定义一个 EnvironmentConfig 类,注入Environment 属性,完整代码如下:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = EnvironmentConfig.class)
@Configuration
@PropertySource("classpath:beanName.properties")
public class EnvironmentConfig {@AutowiredEnvironment env;@Testpublic void testReadProperty(){// 获取bean.name.controller 的属性System.out.println(env.getProperty("bean.name.controller"));// 判断是否包含bean.name.componentSystem.out.println(env.containsProperty("bean.name.component"));// 返回与给定键关联的属性值System.out.println(env.getRequiredProperty("bean.name.service"));}
}

在/resources 目录下新建beanName.properties 文件,如下:

bean.name.configuration=beanNameConfiguration
bean.name.controller=beanNameController
bean.name.service=beanNameService
bean.name.component=beanNameComponent
bean.name.repository=beanNameRepository

启动并进行Junit测试,输出如下:

…..……beanNameControllertruebeanNameService…..……

@Autowired 、 @Inject、@Resource 的区别

@Inject: 这是jsr330 的规范,通过AutowiredAnnotationBeanPostProcessor 类实现的依赖注入。位于javax.inject包内,是Java自带的注解。

@Inject
@Named("environment")
Environment env;

不加@Named注解,需要配置与变量名一致即可。

@Autowired: @Autowired 是Spring提供的注解,通过AutowiredAnnotationBeanPostProessor 类实现注入。位于org.springframework.beans.factory.annotation 包内,是Spring 中的注解

@Autowired
Environment env;

默认是通过byType 实现注入

@Resource: @Resource 是jsr250规范的实现,@Resource通过CommonAnnotationBeanPostProcessor 类实现注入。@Resource 一般会指定一个name属性,如下:

@Resource(name = "environment")
Environment env;

默认是通过byName 实现注入

区别:

@Autowired和@Inject基本是一样的,因为两者都是使用AutowiredAnnotationBeanPostProcessor来处理依赖注入。但是@Resource是个例外,它使用的是CommonAnnotationBeanPostProcessor来处理依赖注入。当然,两者都是BeanPostProcessor。

在介绍完上述三者的区别之后,可以对Environment的属性以上述注入方式进行改造

@Value、@PropertySource 和 @Configuration

@Configuration 可以和@Value 和@PropertySource 一起使用读取外部配置文件,具体用法如下:

在config 包下新建一个ReadValueFromPropertySource类,代码如下

@PropertySource("classpath:beanName.properties")
@Configuration
public class ReadValueFromPropertySource {@Value("bean.name.component")String beanName;@Bean("myTestBean")public MyBean myBean(){return new MyBean(beanName);}}

通过@PropertySource引入的配置文件,使@Value 能够获取到属性值,在给myBean()方法指定了一个名称叫做myTestBean。

修改MyBean类,增加一个name属性和一个构造器,再生成其toString() 方法

public class MyBean {String name;public MyBean(String name) {this.name = name;}public MyBean(){System.out.println("generate MyBean Instance");}public void init(){System.out.println("MyBean Resources Initialized");}@Overridepublic String toString() {return "MyBean{" +"name='" + name + '\'' +'}';}
}

在SpringConfigurationApplication中进行测试,如下

public class SpringConfigurationApplication {public static void main(String[] args) {// 为了展示配置文件的完整性,之前的代码没有删除。
//        AnnotationConfigApplicationContext context = = new AnnotationConfigApplicationContext(MyConfiguration.class)
//        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");//        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();
//        context.register(MyConfiguration.class);
//        context.refresh();
//
//        // 获取启动过程中的bean 定义的名称
//        for(String str : context.getBeanDefinitionNames()){
//            System.out.println("str = " + str);
//        }
//        context.close();ApplicationContext context =new AnnotationConfigApplicationContext(ReadValueFromPropertySource.class);MyBean myBean = (MyBean) context.getBean("myTestBean");System.out.println("myBean = " + myBean);}
}

使用Applicatio@InConntext 就能够获取myTestBean 这个bean,再生成myBean的实例。

输出:myBean = MyBean{name='bean.name.component'}

@Import 和 @Configuration

@Import的定义(来自于JavaDoc):表明一个或者多个配置类需要导入,提供与Spring XML中相等的功能,允许导入@Configuration 、@ImportSelector、@ImportBeanDefinitionRegistar的实现,以及常规组件类似于AnnotationConfigApplicationContext。可能用于类级别或者是原注解。如果XML或者其他非@Configuration标记的Bean资源需要被导入的话,使用@ImportResource。下面是一个示例代码:

在pojo 包下新建两个配置类,分别是CustomerBo, SchedualBo

@Configuration
public class CustomerBo {public void printMsg(String msg){System.out.println("CustomerBo : " + msg);}@Beanpublic CustomerBo testCustomerBo(){return new CustomerBo();}
}@Configuration
public class SchedulerBo {public void printMsg(String msg){System.out.println("SchedulerBo : " + msg);}@Beanpublic SchedulerBo testSchedulerBo(){return new SchedulerBo();}
}

在config 包下新建一个AppConfig,导入CustomerBo 和 SchedulerBo 。

@Configuration
@Import(value = {CustomerBo.class,SchedulerBo.class})
public class AppConfig {}

在config 包下新建一个ImportWithConfiguration ,用于测试@Import 和 @Configuration 的使用

public class ImportWithConfiguration {public static void main(String[] args) {ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);CustomerBo customerBo = (CustomerBo) context.getBean("testCustomerBo");customerBo.printMsg("System out println('get from customerBo')");SchedulerBo schedulerBo = (SchedulerBo) context.getBean("testSchedulerBo");schedulerBo.printMsg("System out println('get from schedulerBo')");}
}

输出:

CustomerBo : System out println('get from customerBo')
SchedulerBo : System out println('get from schedulerBo')

@Profile

@Profile: 表示当一个或多个@Value 指定的配置文件处于可用状态时,组件符合注册条件,可以进行注册。

三种设置方式:

  • 可以通过ConfigurableEnvironment.setActiveProfiles()以编程的方式激活

  • 可以通过AbstractEnvironment.ACTIVE_PROFILES_PROPERTY_NAME (spring.profiles.active )属性设置为

    JVM属性

  • 作为环境变量,或作为web.xml 应用程序的Servlet 上下文参数。也可以通过@ActiveProfiles 注解在集成测试中以声明方式激活配置文件。

作用域

  • 作为类级别的注释在任意类或者直接与@Component 进行关联,包括@Configuration 类

  • 作为原注解,可以自定义注解

  • 作为方法的注解作用在任何方法

注意:

如果一个配置类使用了Profile 标签或者@Profile 作用在任何类中都必须进行启用才会生效,如果@Profile({"p1","!p2"}) 标识两个属性,那么p1 是启用状态 而p2 是非启用状态的。

@ImportResource 和 @Configuration

@ImportResource: 这个注解提供了与@Import 功能相似作用,通常与@Configuration 一起使用,通过AnnotationConfigApplicationContext 进行启动,下面以一个示例来看一下具体用法:

在config下新建TestService 类,声明一个构造函数,类初始化时调用

public class TestService {public TestService(){System.out.println("test @importResource success");}
}

在/resources 目录下新建 importResources.xml ,为了导入TestService

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"xmlns:util="http://www.springframework.org/schema/util"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsdhttp://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-4.2.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.2.xsd"
><bean id = "testService" class="com.spring.configuration.config.TestService" /></beans>

然后在config 下新建一个ImportResourceWithConfiguration, 用于读取配置文件

@Configuration
@ImportResource("classpath:importResources.xml")
public class ImportResourceWithConfiguration {@Autowiredprivate TestService service;public void getImportResource(){new TestService();}public static void main(String[] args) {AnnotationConfigApplicationContext context =new AnnotationConfigApplicationContext(ImportResourceWithConfiguration.class);context.getBean("testService");}
}

输出:test @importResource success

@Configuration 嵌套

@Configuration注解作用在类上,就和普通类一样能够进行相互嵌套,定义内部类。

// 来自JavaDoc
@Configuration
public class AppConfig{@InjectDataSource dataSource;@Beanpublic MyBean myBean(){return new MyBean(dataSource);}@Configurationstatic class DataConfig(){@BeanDataSource dataSource(){return new EmbeddedDatabaseBuilder().build()}}
}

在上述代码中,只需要在应用程序的上下文中注册 AppConfig 。由于是嵌套的@Configuration 类,DatabaseConfig 将自动注册。当AppConfig 、DatabaseConfig 之间的关系已经隐含清楚时,这就避免了使用@Import 注解的需要。

@Lazy 延迟初始化

@Lazy : 表明一个bean 是否延迟加载,可以作用在方法上,表示这个方法被延迟加载;可以作用在@Component (或者由@Component 作为原注解) 注释的类上,表明这个类中所有的bean 都被延迟加载。如果没有@Lazy注释,或者@Lazy 被设置为false,那么该bean 就会急切渴望被加载;除了上面两种作用域,@Lazy 还可以作用在@Autowired和@Inject注释的属性上,在这种情况下,它将为该字段创建一个惰性代理,作为使用ObjectFactory或Provider的默认方法。下面来演示一下:

修改MyConfiguration类,在该类上添加@Lazy 注解,新增一个IfLazyInit()方法,检验是否被初始化。

@Lazy
@Configuration
@ComponentScan(basePackages = "com.spring.configuration.pojo")
public class MyConfiguration {@Beanpublic MyBean myBean(){System.out.println("myBean Initialized");return new MyBean();}@Beanpublic MyBean IfLazyInit(){System.out.println("initialized");return new MyBean();}
}

修改SpringConfigurationApplication 启动类,放开之前MyConfiguration 的启动类

public class SpringConfigurationApplication {public static void main(String[] args) {AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(MyConfiguration.class);
//        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");//        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();
//        context.register(MyConfiguration.class);
//        context.refresh();
//
//        // 获取启动过程中的bean 定义的名称for(String str : context.getBeanDefinitionNames()){System.out.println("str = " + str);}
//        context.close();//        ApplicationContext context =
//                new AnnotationConfigApplicationContext(ReadValueFromPropertySource.class);
//        MyBean myBean = (MyBean) context.getBean("myTestBean");
//        System.out.println("myBean = " + myBean);}
}

输出你会发现没有关于bean的定义信息,但是当吧@Lazy 注释拿掉,你会发现输出了关于bean的初始化信息:

myBean Initialized
generate MyBean Instance
initialized
generate MyBean Instance

@RunWith 和 @ContextConfiguration

Junit4 测试类,用于注解在类上表示通过Junit4 进行测试,可以省略编写启动类代码,是ApplicationContext 等启动类的替换。一般用@RunWith 和 @Configuration 进行单元测试,这是软件开发过程中非常必要而且具有专业性的一部分,上面EnvironmentConfig 类证实了这一点:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = EnvironmentConfig.class)
@Configuration
@PropertySource("classpath:beanName.properties")
public class EnvironmentConfig {//    @Autowired
//    Environment env;@InjectEnvironment env;@Testpublic void testReadProperty(){// 获取bean.name.controller 的属性System.out.println(env.getProperty("bean.name.controller"));// 判断是否包含bean.name.componentSystem.out.println(env.containsProperty("bean.name.component"));// 返回与给定键关联的属性值System.out.println(env.getRequiredProperty("bean.name.service"));}
}

@Enable 启动Spring内置功能

详情查阅@EnableAsync,@EnableScheduling,@EnableTransactionManagement,@EnableAspectJAutoProxy,@EnableWebMvc官方文档

@Configuration 使用约束

必须以类的方式提供(即不是从工厂方法返回的实例)

@Configuration 注解的类必须是非final的

配置类必须是非本地的(即可能不在方法中声明),native 标注的方法

任何嵌套的@Configuration 都必须是static 的。

@Bean 方法可能不会反过来创建更多配置类

参考资料:

https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/context/annotation/Configuration.html

https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/context/annotation/PropertySource.html

https://blog.csdn.net/u012734441/article/details/51706504

本文来自作者投稿,原作者:cxuan

关注下方二维码,订阅更多精彩内容。

转发朋友圈,是对我最大的支持。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/546591.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

jQuery中的.bind()、.live()和.delegate()之间区别分析,学习jquery

2019独角兽企业重金招聘Python工程师标准>>> DOM树 首先&#xff0c;可视化一个HMTL文档的DOM树是很有帮助的。一个简单的HTML页面看起来就像是这个样子&#xff1a; 事件冒泡(又称事件传播) 当我们点击一个链接时&#xff0c;其触发了链接元素的单击事件&#xff0…

Java中常用的4个Json库,哪个性能更牛逼?

来源&#xff1a;http://u6.gg/sDMab前言每次都在网上看到别人说什么某某库性能是如何如何的好&#xff0c;碾压其他的库。但是百闻不如一见&#xff0c;只有自己亲手测试过的才是最值得相信的&#xff0c;本篇通过JMH来测试一下Java中几种常见的JSON解析库的性能。JSON不管是在…

别在Java代码里乱打日志了,这才是正确的打日志姿势!

原文链接&#xff1a;http://t.cn/E9BkD7a使用slf4j使用门面模式的日志框架&#xff0c;有利于维护和各个类的日志处理方式统一。实现方式统一使用: Logback框架打日志的正确方式什么时候应该打日志当你遇到问题的时候&#xff0c;只能通过debug功能来确定问题&#xff0c;你应…

局域网IP地址查看软件V2.0使用教程 Python办公自动化

简介&#xff1a; 在企业、公司都有自己的局域网&#xff0c;作为维护网络运维人员还在使用纸质或电子excel表记录计算机IP信息。是不是很希望有一款工具软件能够管理局域网内的计算机IP地址&#xff0c;可以统计出不同VLAN下的已使用IP-MAC记录和未使用的IP地址&#xff1f;再…

Oracle 10g RAC 升级(CPU Update)之--升级CRS

Oracle 10g RAC 升级&#xff08;CPU Update&#xff09;之--升级CRS系统环境&#xff1a;操作系统&#xff1a;RedHat EL5Cluster&#xff1a; Oracle CRS 10.2.0.1.0Oracle&#xff1a; Oracle 10.2.0.1.0如图所示&#xff1a;RAC 系统架构案例&#xff1a;本案例是从 Oracl…

面试题:聊一聊设计模式的基本原则

设计模式&#xff08;Design pattern&#xff09;代表了程序开发的最佳实践&#xff0c;通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软件开发人员经过相当长的一段时间的试验和错误总结…

python爬虫程序requests采用get和post方式

第1个爬虫程序&#xff1a; main.py #! python # -*- coding: utf-8 -*- from urllib.request import urlopen url "https://www.hao123.com/"if __name__ __main__:resp urlopen(url)# 把读取到网页的页面源代码写入myGetFile.html文件with open("myGetFil…

带你走近AngularJS - 基本功能介绍

带你走近AngularJS系列&#xff1a; 带你走近AngularJS - 基本功能介绍带你走近AngularJS - 体验指令实例带你走近AngularJS - 创建自定义指令------------------------------------------------------------------------------------------------ AngularJS是Google推出的一款…

解决SpringBoot多模块发布时99%的问题?SpringBoot发布的8个原则和4个问题的解决方案

如果使用 SpringBoot 多模块发布到外部 Tomcat&#xff0c;可能会遇到各种各样的问题。本文归纳了以下 8 个原则和发布时经常出现的 4 个问题的解决方案&#xff0c;掌握了这些原则和解决方案&#xff0c;几乎可以解决绝大数 SpringBoot 发布问题。 SpringBoot 多模块发布的 8 …

python爬虫 正则表达式 re.finditer 元字符 贪婪匹配 惰性匹配

测试代码1&#xff1a; main5.py # -*- coding: utf-8 -*- import reif __name__ __main__:# findall匹配字符串中所有的符合正则的内容lst re.findall(r"\d", "濮阳电话区号&#xff1a;0393&#xff0c;郑州电话区号&#xff1a;0371")print(lst)# 【…

如何在路由器的局域网下使用IIS发布网页

这两天尝试着使用IIS发布网页&#xff0c;但是学校宿舍只有一个网线接口&#xff0c;所以宿舍只申请了一个IP然后使用路由器共享网络。但是问题出现了&#xff1a;发布后的网页通过“IP&#xff1a;端口号”只能被自己宿舍的电脑&#xff08;同一路由器&#xff09;访问&#x…

U盘安装ubuntu20.04 Linux系统分区方案 Invalid Partition Table

一、简介&#xff1a; 一般磁盘分区表有两种格式&#xff1a;MBR和GPT&#xff0c;目前主要的BIOS系统引导方式也有两种&#xff1a;传统的Legacy BIOS和新型的UEFI BIOS 如果主机BIOS系统引导方式采用传统的Legacy BIOS&#xff0c;那么安装ubuntu系统的磁盘分区表使用MBR格式…

面试珍藏:最常见的200多道Java面试题(2019年最新版)

作者&#xff1a;老王来自公众号&#xff1a;Java中文社群&#xff08;ID&#xff1a;javacn666&#xff09;2015 年&#xff0c;因为工作岗位的变动&#xff0c;开始负责给集团招聘一些技术人员&#xff0c;出于对公司的负责&#xff0c;也为了更好的胜任技术经理的职位&#…

ubuntu20.04使用终端命令安装谷歌Chrome浏览器稳定版

使用命令行安装谷歌浏览器稳定版 sudo wget http://www.linuxidc.com/files/repo/google-chrome.list -P /etc/apt/sources.list.d/wget -q -O - https://dl.google.com/linux/linux_signing_key.pub | sudo apt-key add -sudo apt-get updatesudo apt-get install google-chr…

面试官:数据量大的情况下分页查询很慢,有什么优化方案?

来源: http://uee.me/aVSnD当需要从数据库查询的表有上万条记录的时候&#xff0c;一次性查询所有结果会变得很慢&#xff0c;特别是随着数据量的增加特别明显&#xff0c;这时需要使用分页查询。对于数据库分页查询&#xff0c;也有很多种方法和优化的点。下面简单说一下我知道…

python爬虫获取中国天气网天气数据 requests BeautifulSoup re

python获取中国天气网天气数据:http://www.weather.com.cn/textFC/henan.shtml main.py # -*- coding: utf-8 -*- import requests from xpinyin import Pinyin from bs4 import BeautifulSoup import

使用Exchange反垃圾邮件组件解决内部仿冒邮件

最近一个朋友遇到一个很有意思的问题&#xff0c;在生产环境中我们装好Exchange之后&#xff0c;肯定需要配置外部邮件的收发。按照我们对邮件系统的理解&#xff0c;我们必须在default 接收连接器上的权限设置&#xff0c;允许匿名连接&#xff0c;才能实现外部邮件的接收。如…

VMware ubuntu20.04 server随win10自动启动与关闭

一、背景简介&#xff1a; 在win10下安装虚拟机&#xff0c;此虚拟机作为服务器使用&#xff0c;但是wi10电脑经常开关机&#xff0c;有的情况下&#xff0c;关机前没有主动关闭虚拟机&#xff0c;到此服务器需要收到重启&#xff0c;且数据容易丢失&#xff0c;配置 win10开机…

秋招注意事项和面试通关攻略

首先给大家说声抱歉&#xff0c;最近由于个人工作(突然996)和精力上的原因&#xff0c;导致公众号断更了一段时间&#xff0c;后面会逐渐恢复公众号的更新&#xff0c;在此感谢大家的理解和支持。国内主流的互联网公司在 8 月份都开放了秋招申请&#xff0c;那秋招有哪些需要注…

python爬虫 单线程的多任务异步协程

在input()、sleep(2)、request.get()等时&#xff0c;都会导致线程阻塞&#xff0c;协程可以解决IO等操作时的阻塞现象&#xff0c;提高CPU利用效率。 1.单线程的多任务异步协程 main.py """ coding: UTF8 """ import asyncio import timeasyn…