春の中高音(1)春のスタートアップに参加したのですが、面接はばかげているかもしれません

知識のポイント:BeanPostProcessorとApplicationListener

春の初歩的な知識のレビュー

春といえば、IOCとAOPが最初に言及されますが、知識のポイントは何ですか?

1. Spring Bootプロセスで、Bをインストールするためのソースコードを確認します。(Bean定義BeanDefinitionのロード、Beanファクトリの作成など)---ソースコードを覚えるのはあまり意味がありません。読んで忘れてしまいました。インタビューが怖いだけです。

2. Beanスコープ(単一のケース、複数のケース)、Beanのライフサイクル:1。インスタンス化2.初期化3.破棄の使用[属性の割り当て、つまり依存性の注入、および認識されたインターフェイスの拡張( BeanNameAware)、BeanPostProcessorプロセッサなど]

  (1)Springはデフォルトでシングルトンであり、Springコンテナに追加されたすべてのBeanはシングルトンプールに保持されます。

  (2)Beanの初期化には3つのメソッドがあります。@ PostConstruct>は、InitializingBeanメソッドを実装します。afterPropertiesSet> init-メソッドのパラメーター指定

  (3)依存性注入には、コンストラクター注入、setメソッド注入、属性注入(autowireまたはresorce)、インターフェース注入(一般的には使用されない)の4つの方法があります。

3.注解:@ Restcontroller / @ Controller、@ getMapping / @ RequestMapping / @ Pathvariable、@ Autowired / @ Resource、Scope、RequestBody / ResponseBody、@ Bean

            (4.1)@ Restcontroller / @ Controllerの違いは何ですか?

          (4.2)@RequestMappingのパス属性と値属性の違いは?

          (4.3)@ Autowired / @ Resourceの違いは?

4.スプリングコンテナBeanFactory ApplicationContextを 理解ます。前者は基本的な関数を実装しますが、後者はAOPを含むより多くの関数を備えています。前者はgetBeanのときにインスタンス化され、すべてのBeanは後者の開始後にインスタンス化されます。

5. Springは循環依存関係を解決します:3レベルキャッシュ singletonObjects / EarlySingletonObjects / singletonFactories

6.springmvcプロセスインターセプター/フィルター違い

7.動的プロキシJDKおよびCGLIBに関しては、AOP原則の動的プロキシ(デフォルトのJDK)。前者は、インターフェイスを実装するクラスへリフレクション生成エージェントあり、後者は、サブクラスカバレッジ生成するためのバイトコードテクノロジの使用です

    動的エージェントと言えば、Springトランザクションが最も一般的な動的エージェントです。@Transitionalが失敗するとどうなりますか?

8. Springbootの改善:自動構成、起動の依存関係、組み込みのTomcat、アクチュエーターの動作監視スプリングブート自動構成原理

以上が基本的にSpringの要点を踏襲しており、Springの予備知識があります!


長編映画が始まります!

実際の戦闘:BeanPostProcessorとApplicationListenerを使用して、APIインターフェースが新しいかどうかを確認します

背景はこれです。マイクロサービスのRequestMappingは、一意に識別されたインターフェイスを形成します。APIが外部にリリースされていない場合は、新しいインターフェイスを外部にリリースする必要があります。新しい開発者が新しいインターフェースを受け取ったが、APIをリリースするようにメンテナンスに通知するのを忘れた場合、それは無駄になります。

   したがって、APIインターフェース(ファイル/データベース)は、Springの開始時に一度永続化でき、次にプログラムを起動したときに、プログラムはテスト環境で起動したときに新しいAPIがあるかどうかを確認できます。どうすればよいですか。行う?

ステップ1:MyBeanPostProcessorを追加してBeanPostProcessorを実装する


import org.apache.servicecomb.provider.rest.common.RestSchema;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;
import org.springframework.core.Ordered;
import org.springframework.stereotype.Component;
import org.springframework.util.ReflectionUtils;
import org.springframework.web.bind.annotation.RequestMapping;

import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;

@Component
public class MyBeanPostProcessor implements BeanPostProcessor, Ordered {


    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        //初始化前执行当前方法
        //doWithMethos的功能是可以执行类中的所有方法,并且提供回调函数方法
        Class<?> cl = bean.getClass();
        RestSchema restSchema = cl.getAnnotation(RestSchema.class);
        String classRequestMappingValue = "";

        if (restSchema != null) {
            List<String> valueList = new ArrayList<>();
            RequestMapping classMapping = cl.getAnnotation(RequestMapping.class);
            if (classMapping != null) {
                classRequestMappingValue = classMapping.path()[0];
                MyCache.mappingMapList.put(classRequestMappingValue, valueList);
            }
            ReflectionUtils.doWithMethods(bean.getClass(), new ReflectionUtils.MethodCallback() {
                public void doWith(Method method) throws IllegalArgumentException, IllegalAccessException {
                    MyBeanPostProcessor.this.processConsumerMethod(bean, method, valueList);
                }
            });
        }
        return bean;
    }

    private void processConsumerMethod(Object bean, Method method, List<String> valueList) {
        //获取方法上的注解
        if (method != null) {
            RequestMapping methodRequestMapping = method.getAnnotation(RequestMapping.class);
            if (methodRequestMapping != null) {
                String[] values = methodRequestMapping.value();
                if (values != null && values.length > 0) {
                    valueList.add(values[0]);
                }
            }
        }
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        //初始化后执行当前方法  ,可以做什么事情?
        // 1.修改注解中的属性值,比如增加前缀标识,比如有调用多个公司的项目,可以增加分支标识自己的公司,这样就可以区分
        //获取Class对象中的注解信息


        return bean;
    }

    @Override
    public int getOrder() {
        return 0;
    }
}

重要な内容について話します。

1. postProcessBeforeInitializationこれは初期化前の操作であり、postProcessAfterInitializationは初期化後の操作です。

2.主にClassオブジェクトを使用して内部のアノテーション情報を取得します。getAnnotation()メソッド。RestSchemaアノテーションはクラスの一意の識別子です(これはservicecombオープンソースマイクロサービスフレームワークのアノテーションです)。これはクラスと同等です。 RequestMapping。

3.ReflectionUtils.doWithMethodsこれは、スプリングに付属するリフレクションツールクラスです。これは何のためにありますか?主に、メソッドのアノテーションを取得するために、クラスのすべてのメソッドをトラバースできます。

ソースコードのorg.springframework.util.ReflectionUtils部分

	public static void doWithMethods(Class<?> clazz, MethodCallback mc, MethodFilter mf) {
		// Keep backing up the inheritance hierarchy.
		Method[] methods = getDeclaredMethods(clazz);
		for (Method method : methods) {
			if (mf != null && !mf.matches(method)) {
				continue;
			}
			try {
				mc.doWith(method);
			}
			catch (IllegalAccessException ex) {
				throw new IllegalStateException("Not allowed to access method '" + method.getName() + "': " + ex);
			}
		}
		if (clazz.getSuperclass() != null) {
			doWithMethods(clazz.getSuperclass(), mc, mf);
		}
		else if (clazz.isInterface()) {
			for (Class<?> superIfc : clazz.getInterfaces()) {
				doWithMethods(superIfc, mc, mf);
			}
		}
	}

4. MethodCallbackはコールバックインターフェイスです。ここでは、RequestMappingアノテーションをリストに取得した後にいくつかの操作を行います。

5.注文されたのはスプリングのインターフェースですが、用途は何ですか?BeanPostProcessorは、春までに公開されるインターフェイスです。複数の実装クラスが存在する可能性があります。Orderedの役割は、これらの実装クラスの実行順序を決定することです。値が小さいほど、早く実行されます。ここでの注文は気にしません。

ステップ2:MyApplicationListenerはApplicationListener <ContextRefreshedEvent>を実装します

import org.springframework.context.ApplicationListener;
import org.springframework.context.event.ContextRefreshedEvent;
import org.springframework.stereotype.Component;

@Component
public class MyApplicationListener  implements ApplicationListener <ContextRefreshedEvent>{

    @Override
    public void onApplicationEvent(ContextRefreshedEvent event) {
        if(event.getApplicationContext().getParent()==null){//保证只执行一次
            System.out.println("============================================spring容器启动完毕,现在再做一些自己的操作!");
            new Thread(new ApiCheck()).start();
        }
    }
}

説明:ここでのApplicationListenerの役割は、他に何かする必要がある場合に、Springコンテナーがロードされた後にこのインターフェースを実装することです。ApplicationRunnerインターフェースは、そのような操作を実装するためにspringbootで提​​供されます。

3番目のステップ:事業運営ApiCheck.java

public class MyCache {
   public  static Map<String,List<String>> mappingMapList = new HashMap<>();

    public static Map<String, List<String>> getMappingMapList() {
        return mappingMapList;
    }

    public static void setMappingMapList(Map<String, List<String>> mappingMapList) {
        MyCache.mappingMapList = mappingMapList;
    }
}

import lombok.SneakyThrows;
import org.apache.servicecomb.core.SCBEngine;
import org.apache.servicecomb.core.SCBStatus;
import org.springframework.stereotype.Component;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.TimeUnit;

@Component
public class ApiCheck implements Runnable {

    @SneakyThrows
    @Override
    public void run() {
        for (; ; ) {
            SCBStatus currentStatus = SCBEngine.getInstance().getStatus();
            if (currentStatus.equals(SCBStatus.UP)) {//如果一开始就改成UP,启动流程还没走完,发现是UP,不会往下走
                Map<String, List<String>> mappingMapList = MyCache.getMappingMapList();
                List<String> apiListNew = new ArrayList<>();
                for (Map.Entry<String, List<String>> stringListEntry : mappingMapList.entrySet()) {
                  String schemaId = stringListEntry.getKey();
                    List<String> list = stringListEntry.getValue();
                    for (String s : list) {
                        apiListNew.add(schemaId+s);
                    }

                }
                File file = new File("C:\\....\\api.txt");
                BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(new FileInputStream(file), "gbk"));
                String s = "";
                List<String> apiOldList = new ArrayList<>();
                while (( s = bufferedReader.readLine()) != null) {
                    apiOldList.add(s);
                }
                for (String s1 : apiListNew) {
                    boolean haveFlag = false;
                    for (String s2 : apiOldList) {
                        if(s1.equals(s2)){
                            haveFlag = true;
                            break;
                        }
                    }
                   if(!haveFlag){
                      System.out.println("------------api不存在:"+s1);
                   }
                }
                System.out.println("============================================API检查分析完毕!");
              break;
            }
            TimeUnit.SECONDS.sleep(3);
        }
    }
}

api.txt

/aa/query
/bb/insert
/cc/delete
/dd/xxquery

説明:一般的に、APIはSpringの開始時にキャッシュマップに配置され、永続的なapi.txtと比較されます。api.txtよりも大きいことが判明した場合は、APIが追加されて達成されたことを意味します。検査の目的と機能が完了しました。

効果画像:

要約:あなたが知る必要があること

  • 基本的に、BeanPostProcessorとApplicationListenerおよびApplicationRunnerの使用法を知っています。
  • ReflectionUtils.doWithMethodsのツールを知っている
  • MethodCallbackコールバックインターフェイスを知っている
  • Orderedインターフェースの役割を理解する
  • Classオブジェクトは、クラスのアノテーションとメソッドのアノテーションをどのように取得しますか

これは高レベルの春とは見なされないかもしれませんが、少なくとも他の人に言うと、IOC、AOP、許可制御の反転、その他の理論的知識だけではありません。行きたいです。春の打ち上げに参加し、一歩踏み出しました!

おすすめ

転載: blog.csdn.net/x18094/article/details/115017978