【项目】深入理解IOC与AOP

IOC(控制反转、依赖注入)

IOC这种思想,是说,我把一个类它的实现写在一个单独的地方,在用到的地方,用一个最简单的方式把它注入进来。(比如Spring中就是使用的@Autowired)

然后就会在整个容器中去找,谁已经注册为一个toutiao.service。

发现有一个对象,它就把这俩这连起来。

这样,就不会被初始化的顺序啊,类与类之间的依赖啊所困扰。

写完之后只需要注入进来就可以了。

我们首先定义一个ToutiaoService,里面定义一个say的方法,返回一串字符串“This is from ToutiaoService”。

package com.test.service;

import org.springframework.stereotype.Service;

@Service
public class ToutiaoService {
    public String say(){
        return "This is from ToutiaoService";
    }
}

 

这里附上整个IndexController的代码:

package com.test.controller;

import com.nowcoder.model.User;
import com.nowcoder.service.ToutiaoService;
import org.omg.CosNaming.NamingContextExtPackage.StringNameHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.HttpStatus;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.*;
import org.springframework.web.servlet.view.RedirectView;

import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.util.*;


@Controller
public class IndexController {
    private static final Logger logger = LoggerFactory.getLogger(IndexController.class);

    @Autowired
    private ToutiaoService toutiaoService;

    @RequestMapping(path = {"/", "/index"})
    @ResponseBody
    public String index(HttpSession session){
        logger.info("Visit Index");
        return "Hello nowcoder" + session.getAttribute("msg")
                + "</br>Say:" + toutiaoService.say();
    }
    

    @RequestMapping(value = {"/profile/{groupId}/{userId}"})
    @ResponseBody
    public String profile(@PathVariable("groupId") String groupId,
                          @PathVariable("userId") int userId,
                          @RequestParam(value = "type", defaultValue = "1")int type,
                          @RequestParam(value = "key", defaultValue = "nowcoder") String key){
        return String.format("GID{%s}, UID{%d}, TYPE{%d}, KEY{%s}", groupId, userId, type, key);
    }

    @RequestMapping(value = {"/vm"})
    public String news(Model model){
        model.addAttribute("value1", "vv1");
        List<String> colors = Arrays.asList(new String[]{"RED", "GREEN", "YELLOW", "BLUE"});
        Map<String, String> map = new HashMap<String, String>();
        for(int i = 0; i < 4; i++){
            map.put(String.valueOf(i), String.valueOf(i * i));
        }
        model.addAttribute("colors", colors);
        model.addAttribute("map", map);
        model.addAttribute("user", new User("Jim"));

        return "news";
    }

    @RequestMapping(value = {"/request"})
    @ResponseBody
    public String request(HttpServletRequest request,
                          HttpServletResponse response,
                          HttpSession session){
          StringBuilder sb = new StringBuilder();
          Enumeration<String> headerNames = request.getHeaderNames();
          while(headerNames.hasMoreElements()){
               String name = headerNames.nextElement();
               sb.append(name + ":" + request.getHeader(name) + "<br>");
          }

          for(Cookie cookie : request.getCookies()){
              sb.append("Cookie:");
              sb.append(cookie.getName());
              sb.append(":");
              sb.append(cookie.getValue());
              sb.append("<br>");
          }

          sb.append("getMethod: " + request.getMethod() + "<br>");
          sb.append("pathInfo: " + request.getPathInfo() + "<br>");
          sb.append("getQueryString: " + request.getQueryString() + "<br>");
          sb.append("getRequestURI: " + request.getRequestURI() + "<br>");

          return sb.toString();
    }

    @RequestMapping(value = {"/response"})
    @ResponseBody
    public String response(@CookieValue(value = "nowcoderid", defaultValue = "a") String nowcoderId,
                           @RequestParam(value = "key", defaultValue = "key") String key,
                           @RequestParam(value = "value", defaultValue = "value")String value,
                           HttpServletResponse response){
        response.addCookie(new Cookie(key, value));
        response.addHeader(key, value);
        return  "NowCoderId From Cookie: " + nowcoderId;
    }

    @RequestMapping("/redirect/{code}")
    public String redirectView(@PathVariable("code") int code,
                               HttpSession session){
        /*RedirectView red = new RedirectView("/", true);
        if(code == 301){
            red.setStatusCode(HttpStatus.MOVED_PERMANENTLY);
        }
        return red;
        */
        session.setAttribute("msg", "jump from direct");
        return "redirect:/";
    }

    @RequestMapping("/admin")
    @ResponseBody
    public String admin(@RequestParam( value = "key", required = true)String key){
        if("admin".equals(key)){
            return "hello admin";
        }
        throw new IllegalArgumentException("Key 错误");
    }

    @ExceptionHandler()
    @ResponseBody
    public String error(Exception e){
        return "error:" + e.getMessage();
    }
}

AOP(面向切面)

只要不是一个特殊的服务,面向所有服务要做一个切下来的业务的时候,就可以通过这个面向切面的思路来编码。

当需要打印日志,控制访问权限的时候,可以采用这种方法。

我的目的是希望在执行这个方法之前或者是之后可以调用这个方法来打印相关的日志。

在这个类中,首先定义一个Logger变量。这个Logger有info、warn等方法,可以根据需求来使用。

然后定义在方法前需要打印的信息,以及在方法后需要打印的信息。写Before还有After方法,里面根据需求来写一些调用的方法。

比如这个beforeMethod中,准备一个字符串,然后把每次访问的目标方法的参数给添加到这个字符串中。在日志打印的时候,打印出“Before Method:” + 相关参数。

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;


@Aspect
@Component
public class LogAspect {
    private static final Logger logger = LoggerFactory.getLogger(LogAspect.class);

    @Before("execution(* com.nowcoder.controller.*Controller.*(..))")
    //这里采用的是通配符的形式,就是只要是带有Controller为后缀的,
    //在调用的时候,都可以打印出相关日志。该日志包含目标方法的参数。
    public void beforeMethod(JoinPoint joinPoint){
        StringBuilder sb = new StringBuilder();
        for(Object arg:joinPoint.getArgs()){
            sb.append("arg" + arg.toString() + "|");
        }
        logger.info("before method: " + sb.toString());
    }

    @After("execution(* com.nowcoder.controller.*Controller.*(..))")
    public void afterMethod(JoinPoint joinPoint){
        logger.info("after mrthod");
    }
}

我们在Controller中添加这个Logger变量,这样就可以打印日志:

 当访问这个网站的时候,

再查看控制台,得到输出的日志:

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring IOC (Inversion of Control) 是一种设计模式,用于管理和组织应用程序中的对象之间的依赖关系。它通过将控制权从应用程序代码中转移给一个容器,从而实现了松耦合和可维护性。在Spring框架中,IOC容器负责创建、组装和管理对象的生命周期。 举个例子来说明,假设我们有一个应用程序,其中包含了 User 和 Order 两个类。在传统的编程方式中,我们需要手动创建这些对象,并在需要的地方进行依赖注入。但在Spring IOC中,我们只需要在配置文件中定义好这些对象,并让容器管理它们的创建和销毁。当我们需要使用这些对象时,只需要从容器中获取即可。 例如,我们可以使用 `<bean>` 标签在配置文件中定义一个 User 对象: ``` <bean id="user" class="com.example.User"> <property name="name" value="John" /> <property name="age" value="25" /> </bean> ``` 然后,在代码中可以通过获取 `user` 对象的实例: ``` User user = (User) applicationContext.getBean("user"); ``` 这样,我们就实现了对象的创建和依赖注入,而不需要手动管理对象的生命周期。 接下来是AOP(Aspect-Oriented Programming),它是一种编程范式,用于解决在面向对象编程中横切关注点的问题。AOP通过将横切关注点(如日志记录、事务管理等)与主要业务逻辑分离,使得代码更加模块化和可维护。 举个例子来说明,假设我们有一个应用程序,其中包含了多个服务类,每个服务类都需要进行日志记录。在传统的编程方式中,我们需要在每个服务类的方法中添加日志记录的代码。但是在AOP中,我们可以通过定义一个切面(Aspect)来实现日志记录功能,并将其应用到需要的方法上。 例如,我们可以使用 Spring AOP 来实现日志记录功能。首先,我们需要引入`spring-boot-starter-aop`依赖。然后,在切面类中定义日志记录的逻辑: ``` @Aspect @Component public class LoggingAspect { @Before("execution(* com.example.service.*.*(..))") public void beforeMethod(JoinPoint joinPoint) { // 日志记录逻辑 System.out.println("Before method: " + joinPoint.getSignature().getName()); } } ``` 在上述例子中,我们使用 `@Before` 注解来定义在方法执行之前执行的逻辑。通过 `JoinPoint` 参数,我们可以获取到方法的签名信息并进行相应的处理。 最后,在需要应用日志记录的方法上添加相应的注解即可: ``` @Service public class UserService { @LogBeforeMethod public void addUser(User user) { // 添加用户逻辑 } } ``` 在上述例子中,我们使用 `@LogBeforeMethod` 注解来说明在该方法执行之前需要应用日志记录的切面。 通过以上的例子,我们可以看到Spring IOCAOP的作用和应用。IOC通过容器管理对象的创建和依赖注入,使得代码更加模块化和可维护;而AOP通过切面的方式实现横切关注点的功能,提高了代码的复用性和可维护性。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [SpringIOCAOP理解之深入浅出(建议收藏)](https://blog.csdn.net/weixin_45105261/article/details/112004029)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [SpringBoot 切面AOP实现权限校验:实例演示与注解全解](https://blog.csdn.net/chali1314/article/details/115317063)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值