gpt4 book ai didi

java - 用于确定主页部分的简洁架构用例——已更新

转载 作者:行者123 更新时间:2023-12-01 18:48:18 25 4
gpt4 key购买 nike

我正在试验 Robert“Uncle Bob”Martin 的 Clean Architecture,以便在即将进行的项目中使用。这是一个基于 Spring MVC 的 Web 应用程序。

其中一个功能是,一旦用户登录,他们的主页就会根据其用户类型显示不同的部分。

在我的小原型(prototype)中,我的工作一切正常,但我对我的解决方案不满意。特别是,我应该在哪里确定我正在处理的用户类型?

我也尝试遵循单一职责原则,因此我创建了一个抽象的 ViewHomePageUseCase,它处理每个人主页上的公共(public)部分。我还有 ViewUserType1HomePageUseCase 等,用于特定的用户类型,它扩展了基类。

虽然目前,在我的 Spring @Controller 中,我有这样的东西:

@RequestMapping(value = {"/home"}, method = RequestMethod.GET)
public String getHomePage(Model model) {

ViewHomePageResponseModel response;
String userId = "test";

int userType = this.getUserProfileUseCase.getUserType(userId);
switch(s) {
case 1:
response = useCaseFactory.viewUserType1_HomePageUseCase().getHomePage(userId);
break;
case 2:
response = useCaseFactory.viewUserType2_HomePageUseCase().getHomePage(userId);
break;
default:
response = useCaseFactory.viewSimpleHomePageUseCase().getHomePage(userId);
}

HomePageViewModel viewModel = presenter.presentHomePage(response);

model.addAttribute("viewModel", viewModel);

return "home";
}

虽然我不喜欢这个 Controller ,因为它里面有业务逻辑。

在原型(prototype)的早期版本中,我确实尝试将“确定用户类型”功能放入用例本身中,然后委托(delegate)给正确的用例。虽然它有效,但我也不喜欢它。

我在当前版本中喜欢的一件事是通过继承拥有多个 ViewHomePageUseCase 类,但我不清楚如何或在哪里根据用户类型选择用例。

我希望我说得清楚!

感谢任何帮助!

克里斯

更新--

我最终所做的是为每个主页创建单独的用例,以及用于决定需要哪个主页的用例。我还使用工厂来实例化用例。每个主页用例仅获取特定主页所需的数据。然后 Spring Controller 就被简化了:

@Controller
public class HomeController {

private IUseCaseFactory useCaseFactory;

@Autowired
public HomeController(IUseCaseFactory useCaseFactory) {
this.useCaseFactory = useCaseFactory;
}

@RequestMapping(value = {"/home"}, method = RequestMethod.GET)
public String getHomePage(Model model) {

String userId = "steven"; // User Type 2
// hardcoded userId's are used in the GetHomePageUseCase class.

IHomePagePresenter presenter = this.useCaseFactory.getHomePageUseCase().getHomePage(userId);
HomePageViewModel viewModel = presenter.presentHomePage();

model.addAttribute("viewModel", viewModel);
return "home";
}
}

我的 GetHomePageUseCase 决定需要哪个主页,如下所示:

public class GetHomePageUseCase implements IViewHomePageUseCase {

private IUseCaseFactory useCaseFactory;

public GetHomePageUseCase(IUseCaseFactory useCaseFactory) {
this.useCaseFactory = useCaseFactory;
}

@Override
public IHomePagePresenter getHomePage(String userId) {
IHomePagePresenter presenter;

//temporary
//TODO - once users are in place, rewrite this function.
String userRole;
switch(userId) {
case "chris":
userRole = "UserType1";
break;
case "steven":
userRole = "UserType2";
break;
default:
userRole = "Public";
break;
}

switch(userRole) {
case "UserType1":
presenter = useCaseFactory.viewUserType1HomePageUseCase().getHomePage(userId);
break;
case "UserType2":
presenter = useCaseFactory.viewUserType2HomePageUseCase().getHomePage(userId);
break;
default:
presenter = useCaseFactory.viewPublicHomePageUseCase().getHomePage(userId);
break;
}

return presenter;
}

}

我仍然不确定这种方法。虽然它比第一个版本更整洁,但我对此不太满意。

一个更普遍的问题:当您的网页(在本例中)其功能依赖于某些用户角色或权限时,我如何将这些用户权限纳入整洁架构方法中?例如,在另一个页面(我的“查看”页面)上,显示的基本数据对于所有用户来说都是相同的,但根据用户角色/权限,用户可能能够编辑、删除等,否则这些按钮不可用甚至显示。在这种情况下(对于我的 View 页面),我有像 showEditButton 这样的 boolean 标志,它将通过用例进行设置,并且 View 页面代码本身只是检查该标志以显示或隐藏按钮。这是在整洁架构中执行此操作的正确方法吗?

再次感谢!!

克里斯

最佳答案

主页是细节,因为用户界面是细节。因此,用例不应涉及选择主页或部分。

Chapter 22 - Use Cases

We also do not expect this layer to be affected by changes to externalities such as the database, the UI, or any of the common frameworks. The use cases layer is isolated from such concerns.

Martin, Robert C.. Clean Architecture: A Craftsman's Guide to Software Structure and Design (Robert C. Martin Series) Pearson Education.

您的用例getUserProfileUseCase没问题,但是根据类型选择主页是ui Controller 的一部分。

I'm trying to also follow Single Responsibility Principle, so I created an abstract ViewHomePageUseCase, which deals with the common sections on everyone's home page.

即使 ViewHomePageUseCase 遵循 SRP(单一责任原则),您的 Controller 也不遵循。每次添加新的用户类型时,您都必须更改 Controller 的 switch 语句。所以我要做的就是重定向到可以处理用户类型的 Controller 。

String redirect = MessageFormat.format("redirect:/home/{0}?userId={1}", userType, userId);
return new ModelAndView(redirect);

然后,您可以通过创建新的 Controller 类来添加对新用户类型的支持,例如

@Controller
@RequestMapping("home")
public class UserType3HomePageController {

@Autowired
private UserType3InfoUseCase useCase;

@GetMapping("/{userType}")
public ModelAndView getHomePage(@PathVariable int userType, @RequestParam userId) {
UserType3Info userInfo = useCase.getUserInfo(userId);
ModelAndView modelAndView = new ModelAndView("userType3Home");
modelAndView.addObject("userInfo ", userInfo);
return modelAndView;
}

}

然后您还可以在 Controller 中实现关注点分离。这也是有道理的,因为不同的主页通常显示不同的用户数据。因此,为每种类型创建一个 View 是有意义的。否则,您可能会在 View 中看到 switch 语句(或 if/else)。

关于java - 用于确定主页部分的简洁架构用例——已更新,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59776967/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com