3

好吧,我在 MVC Spring 3 设计中从 Controller->Service->Repository 有 3 层。现在,我的问题是,由于默认范围定义为 Singleton,它们是线程安全的吗?

这是代码的样子:

用户控制器.java

@Controller
@RequestMapping("/users")
public class UserController extends ExceptionExtension {
     @Autowired
     private IUserService userService;

     @RequestMapping(value = { "/update", "/update/" }, method = RequestMethod.GET)
     public String updateUser(@RequestParam("email") String eMail, ModelMap model)
        throws Exception {

          if (eMail.isEmpty() || eMail == null) {
              throw new ArgumentIsEmptyException("Required String parameter 'email' is empty");
          } else {
              UserModel userModel = userService.setUser(eMail);

              if (userModel != null) {
                  model.put("roleList", userService.setRoleList());
                  model.put("title", "Update Existing User");
                  model.put("post", "/users/post/update");
                  model.put("userForm", userModel);

                  return "users.update";
              } else {
                  model.put("title", "Update Existing User");
                  model.put("result", "<font color='red'><u>" + eMail + "</u> does not exist in the database.</font>");
                  model.put("flag", "Error");

                  return "users.result";
              }
          }
     }
}

用户服务.java

public class UserService implements IUserService {

    @Autowired
    private IUserManager userManager;

    public UserModel setUser(String eMail) {
        UserModel userModel = new UserModel();
        Entity userEntity = userManager.getUser(eMail);

        if (userEntity != null) {
            userModel.setEMail(eMail);
            userModel.setRole(userEntity.getProperty("role").toString());
            userModel.setEnable((Boolean)userEntity.getProperty("enable"));

            return userModel;
        } else {
            return null;
        }
    }
}

假设用户 A 和用户 B 同时运行相同的 url 但参数不同。
用户 A 请求 =>“http://domain.com/users/update?user=myname1@domain.com”。
用户 B 请求 =>“http://domain.com/users/update?user=myname2@domain.com”。

由于控制器是单例的,用户 A 的电子邮件变量是否会与用户 B 重叠,反之亦然?

我很难理解单例线程安全在这种情况下是如何工作的。我的 @Service 和 @Repository 是否应该声明为 @Scope("prototype) 以便内部方法变量与新实例隔离?

====>

使用 Scope("request") 到我的 @Service 层,我遇到了这个错误消息:/

3328 [main] 错误 org.springframework.web.context.ContextLoader - 上下文初始化失败 org.springframework.beans.factory.BeanCreationException:创建名为“roleController”的 bean 时出错:注入自动装配的依赖项失败;嵌套异常是 org.springframework.beans.factory.BeanCreationException:无法自动装配字段:私有 com.company.dashboard.service.IRoleService com.company.dashboard.controller.RoleController.roleService;嵌套异常是 org.springframework.beans.factory.BeanCreationException:创建名称为“roleService”的 bean 时出错:范围“请求”对于当前线程无效;如果您打算从单例中引用它,请考虑为该 bean 定义一个作用域代理;嵌套异常是 java.lang.IllegalStateException:未找到线程绑定请求:您是指实际 Web 请求之外的请求属性,还是在原始接收线程之外处理请求?如果您实际上是在 Web 请求中操作并且仍然收到此消息,则您的代码可能在 DispatcherServlet/DispatcherPortlet 之外运行:在这种情况下,请使用 RequestContextListener 或 RequestContextFilter 来公开当前请求。

4

2 回答 2

1

这里线程安全取决于userService范围的内部锁定singleton

这就是最终发生的情况 => 相同的userSpacebean 用于两个请求并且两个请求同时访问userServicebean。因此,IUserService需要线程安全才能使整个操作执行良好。

或者

把范围放到request.

在这种情况下,为每个请求分配一个新 bean,并且由于线程限制,整个操作是线程安全的。只要您在IUserSpace

于 2012-11-26T16:01:14.527 回答
0

如果您制作任何层范围:原型,就像评论中建议的那样,您只是将问题再次移动到下面的层。

您需要决定您希望在哪里进行事务分界。通常,这是在对服务的调用级别完成的。Spring 可以为您处理。

控制器、服务和存储库通常应该以无状态方式实现,也就是说,这些类没有任何您的方法可以修改的实例变量。否则你将变得容易受到竞争条件的影响。最后,您的数据存储必须支持使用事务。

您使用 Google Appengine 数据存储。与任何 RDBMS 不同,它有一种非常具体的事务处理方法。要从 Java 应用程序与 appengine 交谈,我推荐使用“Objectify”。首先阅读 [关于它的概念](以及底层 BigTable 数据存储1的概念。

于 2012-11-26T17:25:46.300 回答