18-第17课:Spring Cloud 实例详解——业务代码实现

  1. 公共模块
    1. Bean
    2. Service和Controller
  2. 用户模块
  3. 总结

本文开始,我们将实现具体的业务,由于篇幅问题,本文将贴出部分实例代码,其余会提供一般思路。

公共模块

我们的接口会分别放在不同的工程下,其中会有公共代码,在此我们考虑将公共代码抽象出来放到公共模块 common 下。

Bean

我们提供的接口分为输入参数(request)和输出参数(response),输入参数为客户端请求时传入,输出参数为后端接口返回的数据。我们在定义接口时最好将输入参数和输出参数放到 request 和 response 包下,在定义的 Bean 下抽象出 Base 类来,如下代码:

@Data
public abstract class BaseModel {
    private Long id;
}
public abstract class BaseResponse extends BaseModel { }
public abstract class BaseRequest { }

Service和Controller

同样地,我们也可以定义出 BaseService 和 BaseController,在 BaseService 中实现公共方法:

public abstract class BaseService {
  // 密码加密算法
  protected String encryptPassword(String password){
    return MessageDigestUtils.encrypt(password, Algorithm.SHA1);
  }

  // 生成API鉴权的Token
  protected String getToken(String mobile, String password){
    return MessageDigestUtils.encrypt(mobile + password, Algorithm.SHA1);
  }
}

在 BaseController 里写公共方法:

public abstract class BaseController {
  // 接口输入参数合法性校验
  protected void validate(BindingResult result){
    if(result.hasFieldErrors()){
      List<FieldError> errorList = result.getFieldErrors();
      errorList.stream().forEach(item -> Assert.isTrue(false,item.getDefaultMessage()));
    }
  }
}

接下来,我们就可以来实现具体的业务了。

用户模块

根据第14课提供的原型设计图,我们可以分析出,用户模块大概有如下几个接口:

  • 登录
  • 注册
  • 获得用户评论

接下来我们来实现具体的业务(以登录为例),首先是 Bean:

@Data
public class UserBean extends BaseModel{
    private String mobile;
    private String password;
}
@Data
public class LoginRequest {
    @NotEmpty
    private String mobile;
    @NotEmpty
    private String password;
}

其次是 Mapper(框架采用 Mybatis 的注解方式):

@Mapper
public interface UserMapper {
    @Select("select id,mobile,password from news_user where mobile = #{mobile} and password = #{password}")
    List<UserBean> selectUser(String mobile, String password);
}

然后是 Service(具体的业务实现):

@Transactional(rollbackFor = Exception.class)
@Service
public class UserService extends BaseService {
  @Autowired
  private UserMapper userMapper;

  public SingleResult<TokenResponse> login(LoginRequest request) {
    List<UserBean> userList = userMapper.selectUser(request.getMobile(),request.getPassword());
    if(CollectionUtils.isEmpty(userList)) {
      return SingleResult.buildFailure(Code.ERROR, "手机号或密码输入不正确!");
    }
    String token = getToken(request.getMobile(),request.getPassword());
    TokenResponse response = new TokenResponse();
    response.setToken(token);
    return SingleResult.buildSuccess(response);
  }

我们写的接口要提供给客户端调用,因此最后还需要添加 Controller:

@RequestMapping("user")
@RestController
public class UserController extends BaseController {
  @Autowired
  private UserService userService;

  @RequestMapping("login")
  public SingleResult<TokenResponse> login(@Valid @RequestBody LoginRequest request, BindingResult result) {
    // 必须要调用validate方法才能实现输入参数的合法性校验
    validate(result);
    return userService.login(request);
  }
}

这样一个完整的登录接口就写完了。

为了校验我们写的接口是否有问题可以通过 JUnit 来进行单元测试:

@RunWith(SpringRunner.class)
@SpringBootTest(classes = Application.class)
public class TestDB {
  @Autowired
  private UserService userService;

  @Test
  public void test() {
    LoginRequest request = new LoginRequest();
    request.setMobile("13800138000");
    request.setPassword("1");
    System.out.println(userService.login(request));
  }
}

总结

在定义接口之前首先应该分析该接口的输入参数和输出参数,分别定义到 request 和 response 里,在 request 里添加校验的注解,如 NotNull(不能为 null)、NotEmpty(不能为空)等等。

在定义具体的接口,参数为对应的 request,返回值为 SingleResult<Response>MultiResult<Response>,根据具体的业务实现具体的逻辑。

最后添加 Controller,就是调用 Service 的代码,方法参数需要加上 @Valid,这样参数校验才会生效,在调 Service 之前调用 validate(BindResult) 方法会抛出参数不合法的异常。最后,通过 JUnit 进行单元测试。


转载请注明来源,欢迎对文章中的引用来源进行考证,欢迎指出任何有错误或不够清晰的表达。可以在下面评论区评论,也可以邮件至 tuyrk@qq.com

文章标题:18-第17课:Spring Cloud 实例详解——业务代码实现

文章字数:880

本文作者:神秘的小岛岛

发布时间:2020-07-07, 10:14:46

最后更新:2020-07-14, 23:05:28

原始链接:https://www.tuyrk.cn/gitchat/springcloud-quickly/18-bussiness-code/

版权声明: "署名-非商用-相同方式共享 4.0" 转载请保留原文链接及作者。

目录
×

喜欢就点赞,疼爱就打赏