(xml configの代わりに)AOPのアノテーションの使用について友人に示すための簡単なクラスのセットを作成しました。@ComponentScanを機能させることができず、AnnotationConfigApplicationContextgetBeanも誤動作します。私は2つのことを理解したかった。以下のコードを参照してください:
PersonOperationsI.java
package samples.chapter3;
import org.springframework.stereotype.Component;
@Component
public interface PersonOperationsI {
public String getName();
}
PersonOperations.java
/**
*
*/
package samples.chapter3;
import org.springframework.stereotype.Component;
@Component
public class PersonOperations implements PersonOperationsI {
public String getName() {
return "";
}
}
PersonOperationsConfigClass.java
package samples.chapter3;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
@Configuration
//question2 - Below Component Scan didnt work - Test Case failing in setup()
//@ComponentScan(basePackages = {"samples.chapter3"})
@EnableAspectJAutoProxy
public class PersonOperationsConfigClass {
}
PersonOperationsAdvice.java
/**
*
*/
package samples.chapter3;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;
@Component
@Aspect
public class PersonOperationsAdvice {
/**
* execution( [Modifiers] [ReturnType] [FullClassName].[MethodName]
([Arguments]) throws [ExceptionType])
* @param joinPoint
* @return
*/
@Before("execution(public * samples.chapter3.PersonOperations.getName()))")
public String beforeGetName(JoinPoint joinPoint) {
System.out.println("method name = " + joinPoint.getSignature().getName());
return null;
}
}
PersonOperationsTest.java
package samples.chapter3;
import org.junit.Assert;
import org.junit.Before;
import org.junit.Ignore;
import org.junit.Test;
import org.junit.jupiter.api.extension.ExtendWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit.jupiter.SpringExtension;
@ExtendWith(SpringExtension.class)
@ContextConfiguration(classes = { PersonOperationsConfigClass.class })
public class PersonOperationsTest {
//@Autowired
private PersonOperationsI obj;
@Before
public void setUp() {
AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext();
ctx.scan("samples.chapter3");
ctx.refresh();
obj = ctx.getBean(PersonOperationsI.class);
//obj = ctx.getBean(PersonOperations.class);//getBean of Child class not working - why ?
Assert.assertNotNull(obj);
ctx.close();
}
@Test
public void test() {
System.out.println(obj.getName());
}
}
質問1- @ componentscanが機能しない理由。テストケースでAnnotationConfigApplicationContextを使用せず、@ componentscanとautowiredに依存している場合、テストケースのオブジェクトはnullです。
質問2-子クラスのctx.getBean(PersonOperations.class); // getBeanが機能しない-なぜですか?
通常@ComponentScan
、@Configuration
アノテーション付きクラスと一緒に使用する必要があります。引数なしの@ComponentScanは、現在のパッケージとそのすべてのサブパッケージをスキャンするようにSpringに指示することに注意してください。。
@Componentクラスはとても不要になった使用のXML設定にそのタイプのBeanを作成するために春を告げる、そして豆は=>なしインターフェース/抽象クラスをインスタンス化することができるクラスです。したがって、あなたの場合は、@ Componentをから削除し、。PersonOperationsI
にのみ残す必要がありPersonOperations
ます。クラスに@Componentアノテーションを付ける場合、Beanに付けられるデフォルトの名前は、最初の文字が下のクラス名であるため、ctx.getBean("personOperationsI")
またはを呼び出す必要があります。ctx.getBean(PersonOperations.class)
そして将来のために、インターフェースと実装のためのこれらの命名規則を読んでください。あなたの場合、私は以下を変更します:PersonOperationsI
toOperations
質問2
あなたが言ったように、Beanスキャンプロセスは完了していなかったので、コンテキストにBeanはなく、どちらの方法でもコンテキストからBeanを期待するべきではありません(どちらの方法もnullを返します)。@Autowired
context.getBean
以下のリンクには、Beanスキャンの詳細があります(役立つ場合があります)
Springコンポーネントのスキャン
A1、「ApplicationContextのロードに使用するコンポーネントクラス」@ComponentScan
からコメントアウトされているため、機能しませんでした。またはPersonOperationsConfigClass
@Configuration
//@ComponentScan(basePackages = {"samples.chapter3"})
@EnableAspectJAutoProxy
public class PersonOperationsConfigClass {}
テストクラスは、@ ContextConfigurationアノテーションで指定されたコンポーネントクラスから作成されたApplicationContextを取得します。コンポーネントが作成されていないか、自動検出されていないため、@Autowired
失敗しました。
AnnotationConfigApplicationContext
アノテーションが付けられたメソッド内でが使用された場合@Before
、ApplicationContextがプログラムで作成されました。ctx.scan("samples.chapter3");
スキャンされ、。でPersonOperations
注釈が付けられた自動検出@Component
。obj
参照はコードで設定されましたobj = ctx.getBean(PersonOperationsI.class);
。このオブジェクトは「自動配線」ではありませんでした。
OPからのコメントに基づいて更新
Junit 4アノテーションと@ExtendWith(SpringExtension.class)の組み合わせが機能していません。
次のテストクラスは、エラー/失敗なしで正常に実行されます。obj
自動配線され、nullではありません。Junit5の対応するアノテーションを使用しました。
package rg.app.aop.so.q1;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.extension.ExtendWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit.jupiter.SpringExtension;
@ExtendWith(SpringExtension.class)
@ContextConfiguration(classes= {PersonOperationsConfigClass.class})
public class PersonOperationsTest {
@Autowired
private PersonOperationsI obj;
@BeforeEach
public void setUp() {
System.out.println("init ::"+ obj);
Assertions.assertNotNull(obj);
}
@Test
public void testPersonOps() {
Assertions.assertNotNull(obj);
}
}
構成クラス
package rg.app.aop.so.q1;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
@Configuration
@EnableAspectJAutoProxy
@ComponentScan(basePackages = {"rg.app.aop.so.q1"})
public class PersonOperationsConfigClass {
}
A2、以下は私の分析です。
覚えておいてください、属性の@EnableAspectJAutoProxy
デフォルト値は「false」proxyTargetClass
です。この属性は、プロキシメカニズムを決定します:JDKプロキシ(false)またはCGLIBプロキシ(true)。
ここで、有効なアドバイスを持つ有効なアスペクトが存在すると、実際にプロキシが開始されます。コンポーネントは、アドバイスがそれに影響を与える場合にのみプロキシされます。つまり、コンポーネントのプロキシは、必要な場合にのみ発生します。
ケース1
いつ:@EnableAspectJAutoProxy
/@EnableAspectJAutoProxy(proxyTargetClass = false )
ctx.getBean(InterfaceType)
Beanを返しますctx.getBean(ImplementationClassType)
Beanを返さないケース2
いつ : @EnableAspectJAutoProxy(proxyTargetClass = true )
ctx.getBean(InterfaceType)
Beanを返しますctx.getBean(ImplementationClassType)
Beanを返しますケース3
いつ:@EnableAspectJAutoProxy
注釈がない
ctx.getBean(InterfaceType)
Beanを返しますctx.getBean(ImplementationClassType)
Beanを返しますケース1、SpringAOPはproxyTargetClass
falseとして有効になっています。JDKプロキシは、インターフェースタイプのプロキシBeanを作成します。作成されるBeanのタイプはInterfaceTypeであり、ImplementationClassTypeではありません。これは、ctx.getBean(ImplementationClassType)がBeanを返さない理由を説明しています。
ケース2、SpringAOPはproxyTargetClass
trueとして有効になっています。CGLIBは、アノテーションが付けられたクラスをサブクラス化することによってプロキシBeanを作成し@Component
ます。作成されたBeanはImplementationClassType型であり、InterfaceTypeとしての資格もあります。したがって、両方のgetBean()呼び出しは、このBeanを正常に返します。
ケース3、
Springは、特別な処理が必要な場合にのみ「プロキシ」オブジェクトを作成します(例:AOP、トランザクション管理)。
このロジックでは、@EnableAspectJAutoProxy
が存在し@Component
ないため、プロキシなしでアノテーションが付けられたクラスに対してBeanが作成されます。作成されたBeanはImplementationClassType型であり、InterfaceTypeとしての資格もあります。したがって、両方のgetBean()呼び出しは、このBeanを正常に返します。
次のコードで分析を行います。
package rg.app.aop.so.q1;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
public class AppMain {
public static void main(String[] args) {
AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext();
ctx.scan("rg.app.aop.so.q1");
ctx.refresh();
System.out.println();
for(String name:ctx.getBeanNamesForType(PersonOperationsI.class)) {
System.out.println(name);
}
for(String name:ctx.getBeanNamesForType(PersonOperations.class)) {
System.out.println(name);
}
PersonOperationsI obj = ctx.getBean(PersonOperationsI.class);
System.out.println(obj.getClass());
obj = ctx.getBean(PersonOperations.class);
System.out.println(obj.getClass());
ctx.registerShutdownHook();
}
}
ケース1のプリント
personOperations
class com.sun.proxy.$Proxy18
Exception in thread "main" org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'rg.app.aop.so.q1.PersonOperations' available
ケース2プリント
personOperations
personOperations
class rg.app.aop.so.q1.PersonOperations$$EnhancerBySpringCGLIB$$c179e7f2 class rg.app.aop.so.q1.PersonOperations$$EnhancerBySpringCGLIB$$c179e7f2
ケース3プリント
personOperations
personOperations
class rg.app.aop.so.q1.PersonOperations
class rg.app.aop.so.q1.PersonOperations
お役に立てれば
特徴的なスターのコリン・エッグレスフィールドは、RomaDrama Liveでのスリル満点のファンとの出会いについて料理しました!加えて、大会での彼のINSPIREプログラム。
ノーザンエクスポージャーが90年代の最も人気のある番組の1つになった理由を確認するには、Blu-rayまたはDVDプレーヤーをほこりで払う必要があります。
ドミニカのボイリング湖は、世界で2番目に大きいボイリング湖です。そこにたどり着くまでのトレッキングは大変で長いですが、努力する価値は十分にあります。
カリフォルニア州カレキシコでは、6月上旬に気温が約115度に達すると、人々はダウンタウンを歩きます。私たちは夏にたった2週間で、熱波はすでに米国を灼熱しています。
庵野秀明は、ついに新世紀エヴァンゲリオンを締めくくった後、自分のやりたいことを何でもしているようだ(少なくとも今のところ)。彼はウルトラマン映画を作るようになり、現在は仮面ライダー1号を監督しており、生涯にわたる2つのスーパーヒーローの執着のために新しい章を作成することを彼に任せています。
io9の最新かつ最高のオタク玩具ニュースの定期的なまとめであるToyAisleへようこそ。今週は、オビ=ワンの水曜日をさらに2つのハズブロフィギュアで締めくくります。Super7はスプリングフィールドに戻り、さらに多くのシンプソンズアルティメットを目指します。ダースベイダーをバップする勇気はありますか?見てみな!プレイメイトは、初期の頃からのシンプソンズのフィギュアの信じられないほど豊富なラインで素晴らしい仕事をしましたが、Super7のシンプソンズアルティメットほど詳細でアクセサリー化されていませんでした!数字—彼らがもっと早く彼らを追い出し始めてくれることを願っています。
その悪名高い5,000ドルのスターウォーズをテーマにした飲み物に何が入っているのか知りたいと思っているなら、それはもはや完全な謎ではありません。ハイパースペースラウンジのディズニーウィッシュクルーズ船で独占的に入手できるカイバークリスタルは、豪華に提示されたテーマカクテルで、提供されるカムトノに付属しているはずですが、残念ながらそうではありません。銀河系で最も高価なトップシェルフセレクション。
Zendaya shared a sweet photo in honor of boyfriend Tom Holland's 26th birthday Wednesday
シーレン「Ms.JuicyBaby」ピアソンは、先月脳卒中で入院した後、「もう一度たくさんのことをする方法を学ばなければならない」ため、言語療法を受けていることを明らかにしました。
オスカー受賞者の世紀半ばの家には、3つのベッドルーム、2つのバス、オーシャンフロントの景色があります。
1.ドラマを見た後、起業する考えはありますか?あなたのビジネスはボトルネックに遭遇しましたか?方向性がなくてわからない場合は、ドラマを追いかけて行くことを心からお勧めします。(?)ブラフではなく、最も完璧なビジネス例を隠すドラマがあります。2.ブレイキング・バッドとその弁護士ドラマ「ブレイキング・バッド」を見た友人たちは、演劇の中で、穏やかな表情で、弁護士のソウル・グッドマンに深く感銘を受けなければなりません。口を開けて、感覚の弱い傭兵の性格を持っています。道徳の面で、サル・グッドマンは無意識のうちに劇に欠かせない役割を果たし、彼自身のシリーズ「絶望的な弁護士」(ベター・コール・ソール)を生み出しました。ウェントウのテキストとビデオは、劇中のソウル・グッドマンのテレビコマーシャルです。製品(サービス)、競争戦略、市場ポジショニング、ブランド名、ターゲット顧客グループ、コミュニケーション軸から広告まで、サル・グッドマンの役割のビジネス設定は、「最低」と見なすことができる超超超超超超完全です。ブランドコミュニケーションのコスト」「変化」のモデル。なぜ?私の分析をご覧ください。3.ソウル・グッドマンの「事業戦略」1.基本情報ブランド名:Saul Goodman製品:法律相談サービス対象顧客:麻薬中毒、飲酒運転、事故など。法律知識の欠如は、一般的に公立弁護士にしか余裕がなく、真面目な弁護士も「特別な法律を持つ消費者」を避けます。恐れてはいけない「ニーズ」。コミュニケーションの主軸:この国のすべての男性、女性、子供は有罪判決を受けるまで無実だと思います。地域:アルバカーキ市スローガン:Thrallに電話したほうがいいです!(ベター・コール・ソール)広告:2つの可能性のある犯罪状況をシミュレートします+サウルの主張+サウルのスローガン2をより適切に呼び出します。