@component和@autowired区别
时间: 2023-05-03 08:00:15 浏览: 98
@component和@Autowired是Spring框架中的注解,@Component用于将一个类定义为组件,交给Spring进行管理,@Autowired用于进行依赖注入,将需要的组件自动装配进来。二者的作用不同,但都可以实现组件的自动化管理。
相关问题
1.利用注解、反射和工厂模式设计一个简单的IoC容器。 2.该IoC容器包含3个注解和一个IoC容器类(AnnotationConfigApplicationContext),其定义如下。 注解 含义 @Component 标注Bean @Autowired 标注需要被注入的对象 @Configuration 标注为配置类 @ComponentScan 注解扫描器
好的,针对您的问题,我可以给您提供一些思路和代码实现。
首先,我们来看一下注解的定义:
1. @Component
该注解用于标注Bean,表示该类需要被IoC容器管理。
```
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Component {
String value() default "";
}
```
2. @Autowired
该注解用于标注需要被注入的对象,表示该对象需要从IoC容器中获取。
```
@Target({ElementType.FIELD, ElementType.CONSTRUCTOR})
@Retention(RetentionPolicy.RUNTIME)
public @interface Autowired {
String value() default "";
}
```
3. @Configuration
该注解用于标注为配置类,表示该类中定义的Bean需要被IoC容器管理。
```
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Configuration {
String value() default "";
}
```
接下来,我们来看一下IoC容器类的实现。我们可以定义一个AnnotationConfigApplicationContext类,其中包含一个Map用于存储Bean的定义和实例。
```
public class AnnotationConfigApplicationContext {
private Map<String, Object> beanMap = new HashMap<>();
public AnnotationConfigApplicationContext(Class<?> configClass) throws Exception {
// 扫描所有的Component注解,并将类定义加入到beanMap中
scanComponents(configClass);
// 处理所有的Autowired注解,并注入Bean依赖
injectDependencies();
}
// 扫描所有的Component注解
private void scanComponents(Class<?> configClass) throws Exception {
ComponentScan componentScan = configClass.getAnnotation(ComponentScan.class);
if (componentScan != null) {
String[] basePackages = componentScan.value();
for (String basePackage : basePackages) {
String packagePath = basePackage.replace(".", "/");
URL url = Thread.currentThread().getContextClassLoader().getResource(packagePath);
if (url == null) {
continue;
}
File basePackageFile = new File(url.toURI());
for (File file : basePackageFile.listFiles()) {
String fileName = file.getName();
if (fileName.endsWith(".class")) {
String className = fileName.substring(0, fileName.lastIndexOf("."));
Class<?> clazz = Class.forName(basePackage + "." + className);
Component component = clazz.getAnnotation(Component.class);
if (component != null) {
String beanName = component.value();
if (beanName.equals("")) {
beanName = className.substring(0, 1).toLowerCase() + className.substring(1);
}
Object bean = clazz.getDeclaredConstructor().newInstance();
beanMap.put(beanName, bean);
}
}
}
}
}
}
// 处理所有的Autowired注解
private void injectDependencies() throws Exception {
for (Object bean : beanMap.values()) {
for (Field field : bean.getClass().getDeclaredFields()) {
Autowired autowired = field.getAnnotation(Autowired.class);
if (autowired != null) {
String beanName = autowired.value();
if (beanName.equals("")) {
beanName = field.getType().getSimpleName().substring(0, 1).toLowerCase() + field.getType().getSimpleName().substring(1);
}
field.setAccessible(true);
field.set(bean, beanMap.get(beanName));
}
}
}
}
// 获取Bean实例
public Object getBean(String beanName) {
return beanMap.get(beanName);
}
}
```
最后,我们来看一下如何使用该IoC容器。我们可以定义一个配置类,并在该类中定义需要被IoC容器管理的Bean。
```
@Configuration
@ComponentScan("com.example")
public class AppConfig {
@Component
public class UserService {
public void sayHello() {
System.out.println("Hello, world!");
}
}
@Component
public class UserController {
@Autowired
private UserService userService;
public void sayHello() {
userService.sayHello();
}
}
}
```
然后,我们可以创建一个AnnotationConfigApplicationContext实例,并通过getBean方法获取Bean实例。
```
public static void main(String[] args) throws Exception {
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
UserController userController = (UserController) context.getBean("userController");
userController.sayHello();
}
```
该IoC容器包含3个注解和一个IoC容器类(AnnotationConfigApplicationContext),其定义如下。 注解 含义 @Component 标注Bean @Autowired 标注需要被注入的对象 @Configuration 标注为配置类 @ComponentScan 注解扫描器 自定义两个业务类Group和User,创建一个测试类Test,对IoC容器进行测试。
好的,根据你的需求,我可以给出一个简单的实现方案,具体步骤如下:
1. 定义三个注解:@Component、@Autowired和@Configuration。
@Component注解用于标记需要被IoC容器管理的Bean组件,@Autowired注解用于标记需要被注入的对象,@Configuration注解用于标记配置类。
```
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Component {
}
```
```
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Autowired {
}
```
```
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Configuration {
}
```
2. 定义一个注解扫描器@ComponentScan,用于指定需要扫描的包路径。
```
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface ComponentScan {
String[] value() default {};
}
```
3. 定义IoC容器类AnnotationConfigApplicationContext,实现以下功能:
- 扫描指定包下的所有类,并将所有被@Component注解的类实例化并加入IoC容器中;
- 根据@Autowired注解自动注入依赖关系;
- 支持@Configuration注解,即使用@Configuration注解标记的类应该被解析为配置类,并按照其中的@Bean方法实例化并加入IoC容器中;
- 支持@ComponentScan注解,即指定需要扫描的包路径。
```
public class AnnotationConfigApplicationContext {
private Map<String, Object> beans = new HashMap<>();
public AnnotationConfigApplicationContext(Class<?> configClass) throws Exception {
// 扫描包路径
ComponentScan componentScan = configClass.getDeclaredAnnotation(ComponentScan.class);
String[] basePackages = componentScan.value();
List<Class<?>> classes = new ArrayList<>();
for (String basePackage : basePackages) {
classes.addAll(getClasses(basePackage));
}
// 实例化被@Component标注的类
for (Class<?> clazz : classes) {
if (clazz.isAnnotationPresent(Component.class)) {
Object bean = clazz.getDeclaredConstructor().newInstance();
beans.put(clazz.getName(), bean);
}
}
// 自动注入被@Autowired标注的属性
for (Object bean : beans.values()) {
for (Field field : bean.getClass().getDeclaredFields()) {
if (field.isAnnotationPresent(Autowired.class)) {
Object dependency = beans.get(field.getType().getName());
field.setAccessible(true);
field.set(bean, dependency);
}
}
}
// 实例化@Configuration标注的类中的@Bean方法
for (Class<?> clazz : classes) {
if (clazz.isAnnotationPresent(Configuration.class)) {
Object config = clazz.getDeclaredConstructor().newInstance();
for (Method method : clazz.getDeclaredMethods()) {
if (method.isAnnotationPresent(Bean.class)) {
Object bean = method.invoke(config);
beans.put(bean.getClass().getName(), bean);
}
}
}
}
}
public Object getBean(Class<?> clazz) {
return beans.get(clazz.getName());
}
private List<Class<?>> getClasses(String packageName) throws Exception {
List<Class<?>> classes = new ArrayList<>();
String path = packageName.replace(".", "/");
URL url = Thread.currentThread().getContextClassLoader().getResource(path);
File directory = new File(url.toURI());
if (directory.exists()) {
File[] files = directory.listFiles();
for (File file : files) {
String fileName = file.getName();
if (fileName.endsWith(".class")) {
String className = packageName + "." + fileName.substring(0, fileName.length() - 6);
Class<?> clazz = Class.forName(className);
classes.add(clazz);
}
}
}
return classes;
}
}
```
4. 自定义两个业务类Group和User,并在类上使用@Component注解进行标记。
```
@Component
public class Group {
private String name;
public Group() {
this.name = "defaultGroup";
}
public Group(String name) {
this.name = name;
}
public String getName() {
return name;
}
}
```
```
@Component
public class User {
private String name;
@Autowired
private Group group;
public User() {
this.name = "defaultUser";
}
public User(String name) {
this.name = name;
}
public String getName() {
return name;
}
public Group getGroup() {
return group;
}
}
```
5. 创建一个测试类Test,通过AnnotationConfigApplicationContext类实例化IoC容器,并从容器中获取Group和User实例进行测试。
```
public class Test {
public static void main(String[] args) throws Exception {
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
Group group = (Group) context.getBean(Group.class);
System.out.println(group.getName());
User user = (User) context.getBean(User.class);
System.out.println(user.getName());
System.out.println(user.getGroup().getName());
}
}
```
6. 创建一个配置类AppConfig,使用@Configuration和@Bean注解来实例化Bean组件。
```
@Configuration
@ComponentScan("com.example")
public class AppConfig {
@Bean
public Group group() {
return new Group("testGroup");
}
@Bean
public User user() {
return new User("testUser");
}
}
```
通过以上步骤,可以实现一个简单的IoC容器以及依赖注入功能。注:以上代码仅供参考,可能存在错误和不足之处,需要根据实际情况进行调整和完善。
阅读全文