草庐IT

@Configuration的proxyBeanMethods的意义

lgtn 2023-03-28 原文

一、首先理解@Bean注解

1、@Bean表示方法产生一个由Spring管理的bean,属性的名称语义与Spring XML中的 标签配置的一样,源码如下:

  @Target({ElementType.METHOD, ElementType.ANNOTATION_TYPE})
  @Retention(RetentionPolicy.RUNTIME)
  @Documented
  public @interface Bean {
      @AliasFor("name")
      String[] value() default {};


      @AliasFor("value")
      String[] name() default {};


      @Deprecated
      Autowire autowire() default Autowire.NO;


      boolean autowireCandidate() default true;


      String initMethod() default "";


      String destroyMethod() default AbstractBeanDefinition.INFER_METHOD;
 }

@Bean 用在方法上(此方法需要有返回值),表示可以通过编程手段自行对需要初始化的对象进行赋值。

例如:

@Bean
public RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory connectionFactory)
{
    RedisTemplate<Object, Object> template = new RedisTemplate<>();
    template.setConnectionFactory(connectionFactory);

    FastJson2JsonRedisSerializer serializer = new FastJson2JsonRedisSerializer(Object.class);

    ObjectMapper mapper = new ObjectMapper();
    mapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
    mapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
    serializer.setObjectMapper(mapper);

    template.setValueSerializer(serializer);
    // 使用StringRedisSerializer来序列化和反序列化redis的key值
    template.setKeySerializer(new StringRedisSerializer());
    template.afterPropertiesSet();

    return template;
}

2、IOC容器会扫描到有@Bean注解的方法,会将此方法的返回值作为一个bean管理起来,并会执行整个bean生命周期。

二、@Configuration注解

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface Configuration {

    @AliasFor(annotation = Component.class)
    String value() default "";

    boolean proxyBeanMethods() default true;

}

使用proxyBeanMethods的流程图如下,具体的可以阅读源码

image.png
void validate(ProblemReporter problemReporter) {
        // A configuration class may not be final (CGLIB limitation) unless it declares proxyBeanMethods=false
        Map<String, Object> attributes = this.metadata.getAnnotationAttributes(Configuration.class.getName());
        if (attributes != null && (Boolean) attributes.get("proxyBeanMethods")) {
            if (this.metadata.isFinal()) {
                problemReporter.error(new FinalConfigurationProblem());
            }
            for (BeanMethod beanMethod : this.beanMethods) {
                beanMethod.validate(problemReporter);
            }
        }
    }

测试代码,可调整SpringConfig类上的proxyBeanMethods的值,观察输出结果


public class User {

    String id;
    String name;

    public User(String id, String name) {
        this.id = id;
        this.name = name;
    }

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}


@Configuration(proxyBeanMethods = false)
public class SpringConfig {

    @Bean
    public User myUser() {
        User user = new User("1","green");
        return user;
    }
}

@SpringBootApplication(scanBasePackages={"cn.hutool.extra.spring"})
public class MySpringApplication {

    public static void main(String[] args) {
        SpringApplication.run(MySpringApplication.class,args);

        User user1 = SpringUtil.getBean(User.class);
        User user2 = SpringUtil.getBean(User.class);

        System.out.println(user1);
        System.out.println(user2);

        SpringConfig config = SpringUtil.getBean(SpringConfig.class);
        User user3 = config.myUser();
        System.out.println(user3);

        User user4 = SpringUtil.getBean(User.class);
        System.out.println(user4);
    }
}

最终结论:
1、配置类上有了proxyBeanMethods=false后,表示配置类不会被代理,可以提高容器的初始化性能。其差别,主要再使用@Bean的方法上

(1)proxyBeanMethods=true,会给config类生成代理对象,当直接调用方法时,不会生成新的对象,会从spring 容器中获取
(2)proxyBeanMethods=false,不会给config类生成代理对象,当直接调用方法时,会生成新的对象,

2、如果配置类中的方法没有调用关系,则可以把proxyBeanMethods 设置为 false。如果存在以下情况,则需要将proxyBeanMethods 设置为 true(如果是fasle,Idea中会提示编译报错)

@Bean
    public User hisUser() {
        // 内部调用
        User yourUser = this.yourUser();
        User hUser = new User("2","his");
        hUser.setName(hUser.getName() + yourUser.getName());
        return hisUser();
    }
    
    @Bean
    public User yourUser() {
        User user = new User("1","your");
        return user;
    }

有关@Configuration的proxyBeanMethods的意义的更多相关文章

  1. 旋转矩阵的几何意义 - 2

    点向量坐标矩阵的几何意义介绍旋转矩阵的几何含义之前,先介绍一下点向量坐标矩阵的几何含义点:在一维空间下就是一个标量,如同一条直线上,以任意某一个位置为0点,以一定的尺度间隔为1,2,3...,相反方向为-1,-2,-3...;如此就形成了一维坐标系,这时候任何一个点都可以用一个数值表示,如点p1=5,即即从原点出发沿着x轴正方向移动5个尺度;点p2=-3,负方向移动3个尺度;     在一维坐标系上过原点做垂直于一维坐标系的直线,则形成了二维坐标系,此时描述一个点需要两个数值来表示点p3=(3,2),即从原点出发沿着x轴正方向移动3个尺度,在此基础上沿着y轴正方向移动两个尺度的位置就是点p3。

  2. ruby - 如何将 Puma::Configuration 传递给 Sinatra? - 2

    这是我的网络应用:classFront我是这样开始的(请不要建议使用Rack):Front.start!这是我的Puma配置对象,我不知道如何传递给它:require'puma/configuration'Puma::Configuration.new({log_requests:true,debug:true})说真的,怎么样? 最佳答案 配置与您运行的方式紧密相关puma服务器。运行的标准方式puma-pumaCLI命令。为了配置puma配置文件config/puma.rb或config/puma/.rb应该提供(参见examp

  3. ruby - Ruby 中的 "method"方法有什么意义? - 2

    在Ruby中,有“method”方法,它创建一个方法对象,然后您可以将其视为Proc。如果您想进行元编程,这很有用:deffoobar(method_as_a_string)2.method(method_as_a_string).call(2)endfoobar("+")=>4foobar("-")=>0但是,通常情况下,如果您想进行这样的元编程,您会使用send而不是方法...deffoobar(method_as_a_string)2.send(method_as_a_string,2)endfoobar("+")=>4foobar("-")=>0Ruby中的“方法”方法有什么意

  4. ruby-on-rails - `method_missing':#<Rails::Application::Configuration:0x00> 的未定义方法 `action_mailer' - 2

    我正在构建一个Rails应用程序并且使用的是Rails4.0.1。我有一个错误,并注意到它在3个月前被称为rails上的一个错误,所以我决定:捆绑更新并获得rails4.0.3这样做之后,测试和服务器都不会启动,并且会抛出错误:gems/railties-4.0.3/lib/rails/railtie/configuration.rb:95:in`method_missing':undefinedmethod`action_mailer'for#(NoMethodError)目前我在config/environments/*中注释掉了action_mailer行,但最好能找到一个真正的

  5. ruby - 在方法调用中使用 Ruby 的 double-splat (`**` ) 有什么意义? - 2

    通过一个splat,我们可以将一个数组扩展为多个参数,这与直接传递数组有很大不同:deffoo(a,b=nil,c=nil)aendargs=[1,2,3]foo(args)#Evaluatestofoo([1,2,3])=>[1,2,3]foo(*args)#Evaluatestofoo(1,2,3)=>1然而,对于关键字参数,我看不出有什么区别,因为它们只是散列的语法糖:deffoo(key:)keyendargs={key:'value'}foo(args)#Evaluatestofoo(key:'value')=>'value'foo(**args)#Evaluatestofo

  6. ruby - Ruby 的方法解除绑定(bind)机制有什么意义? - 2

    Method#unbind返回对该方法的UnboundMethod引用,稍后可以使用UnboundMethod#bind将其绑定(bind)到另一个对象.classFooattr_reader:bazdefinitialize(baz)@baz=bazendendclassBardefinitialize(baz)@baz=bazendendf=Foo.new(:test1)g=Foo.new(:test2)h=Bar.new(:test3)f.method(:baz).unbind.bind(g).call#=>:test2f.method(:baz).unbind.bind(h).

  7. ruby - SAP 新实现Ruby 对Ruby 程序员有何意义? - 2

    SAP宣布BlueRuby,在ABAP虚拟机中运行的Ruby版本。这似乎为Ruby语言增加了可信度,但是,除了SAP开发人员之外,这是否适用于Ruby社区的其他人?我只是想知道这可能还有什么其他意义。可能会雇用Ruby开发人员从事SAP项目的额外工作机会?Ruby程序员还有其他潜在的好处吗?此外,还有一些我不清楚的地方:除了MRI和JRuby,还有多少种不同的实现,为什么我作为Ruby程序员需要MRI以外的任何实现?我知道如果我想与Java库集成,我可能需要JRuby。除了MRI或JRuby之外,还有什么时候我可能需要研究实现吗?我注意到BlueRuby已编译。这是一个很大的好处吗?这

  8. ruby-on-rails - Ruby 中的纤维有什么意义? - 2

    我不明白下面是怎么回事:counts=Hash.new(0)File.foreach("testfile")do|line|line.scan(/\w+/)do|word|word=word.downcasecounts[word]+=1endendcounts.keys.sort.each{|k|print"#{k}:#{counts[k]}"}远比:words=Fiber.newdoFile.foreach("testfile")do|line|line.scan(/\w+/)do|word|Fiber.yieldword.downcaseendendendcounts=Hash.

  9. ruby - Ruby 中的一元加运算符有什么意义? - 2

    除了与一元减号形成良好的对称性之外,为什么要在Numeric类上定义一元加号运算符?它是否有一些实用值(value),除了引起混淆允许编写类似++i的东西(与大多数非Rubyist认为的不同,它不会增加i).我可以想到在自定义类上定义一元加号的场景可能很有用(比如说,如果您正在创建一些性感的DSL),所以能够定义它是可以的,但为什么它已经在Rubynumbers上定义了? 最佳答案 也许这只是一个一致性问题,既与其他编程语言一致,又反射(reflect)一元减号。在TheRubyProgrammingLanguage中找到对此的支持

  10. ruby-on-rails - rails : Could not load database configuration. 没有这样的文件 - - 2

    我从GitHub存储库克隆了一个应用程序文件夹,在捆绑安装gems之后,我尝试使用rakedb:setup和rakedb:migrate命令,但都没有用,这是我的错误消息:**arun997@promanager:~/workspace(master)$rakedb:setuprequire'rails/all'...2.470sBundler.require...7.590srakeaborted!Cannotload`Rails.application.database_configuration`:Couldnotloaddatabaseconfiguration.Nosuchf

随机推荐