Web应用程序体系结构– Spring MVC – AngularJs堆栈

Spring MVC和AngularJs共同为构建表单密集型Web应用程序提供了一个真正高效且吸引人的前端开发堆栈。在这篇博客文章中,我们将看到如何使用这些技术构建表单密集型Web应用程序,并将这种方法与其他方法进行比较可用选项。 可以在此github 存储库中找到功能完整且受保护的示例Spring MVC / AngularJs Web应用程序。我们将研究以下主题:

  • Spring MVC + Angular单页应用程序的体系结构
  • 如何使用Angular构建Web UI
  • 哪个Javascript / CSS库与Angular相辅相成?
  • 如何使用Spring MVC构建REST API后端
  • 使用Spring Security保护REST API
  • 与使用完全基于Java的方法的其他方法相比,这又如何呢?

Spring MVC + Angular单页Web应用程序的体系结构

表单密集型企业级应用程序非常适合构建为单页Web应用程序。 与其他更传统的服务器端体系结构相比,主要思想是将服务器构建为一组无状态可重用的REST服务,并从MVC的角度将控制器从后端移出并移至浏览器中:

SpringMVCAngular2

客户端具有MVC功能,并包含在视图层,控制器层和前端服务层中分隔的所有表示逻辑。 初始应用程序启动后,只有JSON数据通过客户端和服务器之间的连接。

后端是如何构建的?

企业前端应用程序的后端可以以非常自然和类似于Web的方式构建为REST API。 可以使用相同的技术为第三方应用程序提供Web服务-在许多情况下,不需要单独的SOAP Web服务堆栈。

从DDD的角度来看,域模型保留在服务和持久层级别的后端。 通过有线只有DTO可以通过,而域模型则不可以。

如何使用Angular构建Web应用程序的前端

前端应围绕特定于视图的模型(不是域模型)构建,并且应仅处理表示逻辑,而不能处理业务逻辑。 这些是前端的三层:

视图层

视图层由HTML模板,CSS和代表不同UI组件的任何Angular指令组成。 这是登录表单的简单视图示例:

<form ng-submit="onLogin()" name="form" novalidate="" ng-controller="LoginCtrl">  <fieldset><legend>Log In</legend><div class="form-field"><input ng-model="vm.username" name="username" required="" ng-minlength="6" type="text"><div class="form-field"><input ng-model="vm.password" name="password" required="" ng-minlength="6" pattern="(?=.*\d)(?=.*[a-z])(?=.*[A-Z]).{6,}" type="password"></div></div></fieldset><button type="submit">Log In</button><a href="/resources/public/new-user.html">New user?</a></form>

控制器层

控制器层由Angular控制器组成,它们将从后端和视图检索的数据粘合在一起。 控制器初始化视图模型,并定义视图应如何对模型更改做出反应,反之亦然:

angular.module('loginApp', ['common',  'editableTableWidgets'])  .controller('LoginCtrl', function ($scope, LoginService) {$scope.onLogin = function () {console.log('Attempting login with username ' + $scope.vm.username + ' and password ' + $scope.vm.password);if ($scope.form.$invalid) {return;}LoginService.login($scope.vm.userName, $scope.vm.password);};});

控制器的主要职责之一是执行前端验证。 在前端进行的任何验证仅出于方便用户的目的-例如,它们对于立即通知用户必填字段很有用。

由于安全原因,任何前端验证都需要在后端在服务层级别重复进行,因为可以轻松绕过前端验证。

前端服务层

一组允许与后端交互并可以注入到Angular控制器中的Angular服务:

angular.module('frontendServices', [])  .service('UserService', ['$http','$q', function($http, $q) {return {getUserInfo: function() {var deferred = $q.defer();$http.get('/user').then(function (response) {if (response.status == 200) {deferred.resolve(response.data);}else {deferred.reject('Error retrieving user info');}});return deferred.promise;}

让我们看看还需要哪些其他库来启动前端并运行它。

哪些Javascript / CSS库是补充Angular所必需的?

Angular已经提供了构建应用程序前端所需的大部分功能。 Angular的一些很好的补充是:

  • 雅虎仅有4k的一个易于主题化的纯CSS库,名为PureCss 。 它的皮肤生成器可以轻松地基于基色生成主题。 它的BYOJ(自带Java脚本)解决方案有助于使事情保持“成角度的方式”。
  • 功能性的程序库来处理数据。 这些年来,似乎最常用,维护和记录最完善的工具就是lodash 。

使用这两个库和Angular, 几乎可以构建任何基于表单的应用程序 ,实际上不需要任何其他操作。 根据您的项目,其他一些库可能是一个选项:

  • 像requirejs这样的模块系统很不错,但是由于Angular模块系统不处理文件检索,因此在requirejs和angular模块的依赖项声明之间引入了一些重复。
  • CSRF Angular模块,用于防止跨站点请求伪造攻击。
  • 国际化模块

如何使用Spring MVC构建REST API后端

后端是使用通常的后端层构建的:

  • 路由器层:定义哪些服务入口点对应于给定的HTTP URL,以及如何从HTTP请求中读取参数
  • 服务层:包含任何业务逻辑(例如验证),定义业务交易的范围
  • 持久层:将数据库映射到内存中的域对象或从中映射

目前,最好仅使用Java配置来配置Spring MVC。 几乎不需要web.xml ,请参阅此处的仅使用Java config的完全配置的应用程序示例。

服务和持久层是使用通常的DDD方法构建的,因此让我们将注意力集中在路由器层上。

路由器层

用于构建JSP / Thymeleaf应用程序的相同Spring MVC批注也可以用于构建REST API。

最大的区别在于,控制器方法不返回定义应渲染哪个视图模板的String。 相反,
@ResponseBody批注指示应该直接呈现controller方法的返回值并成为响应主体:

@ResponseBody
@ResponseStatus(HttpStatus.OK)
@RequestMapping(method = RequestMethod.GET)
public UserInfoDTO getUserInfo(Principal principal) {User user = userService.findUserByUsername(principal.getName());Long todaysCalories = userService.findTodaysCaloriesForUser(principal.getName());return user != null ? new UserInfoDTO(user.getUsername(), user.getMaxCaloriesPerDay(), todaysCalories) : null;
}

如果要使用@ResponseBody注释该类的所有方法,则最好使用@RestController注释整个类。

通过添加Jackson JSON库,方法返回值将直接转换为JSON,而无需任何进一步配置。 也可以转换为XML或其他格式,具体取决于客户端指定的Accept HTTP标头的值。

请参见此处的几个配置了错误处理的控制器的示例。

如何使用Spring Security保护REST API

可以使用Spring Security Java配置来保护REST API。 一个好的方法是使用带有回退到HTTP Basic身份验证的表单登录,并包括一些CSRF保护,以及强制所有后端方法只能通过HTTPS访问的可能性。

这意味着后端将为用户提供一个登录表单,并在成功登录到浏览器客户端时分配一个会话cookie,但是通过支持回退到HTTP Basic(通过凭据通过Authorization HTTP标头传递凭据)的非浏览器客户端,它仍然可以很好地工作。 。

遵循OWASP的建议,可以使REST服务达到最小的无状态(唯一的服务器状态是用于身份验证的会话cookie),以避免必须为每个请求通过电线发送凭据。

这是有关如何配置REST API安全性的示例 :

http.authorizeRequests().antMatchers("/resources/public/**").permitAll().anyRequest().authenticated().and().formLogin().defaultSuccessUrl("/resources/calories-tracker.html").loginProcessingUrl("/authenticate").loginPage("/resources/public/login.html").and().httpBasic().and().logout().logoutUrl("/logout");if ("true".equals(System.getProperty("httpsOnly"))) {LOGGER.info("launching the application in HTTPS-only mode");http.requiresChannel().anyRequest().requiresSecure();}

此配置仅涵盖安全性的身份验证方面,选择授权策略取决于API的安全性要求。 如果您需要对授权进行非常细粒度的控制,请检查Spring Security ACL是否适合您的用例。

现在,让我们看一下这种构建Web应用程序的方法与其他常用方法的比较。

将Spring / MVC Angular堆栈与其他常见方法进行比较

这种将Javascript用作前端,将Java用作后端的方法可以简化开发工作流程。

当后端运行时,不需要任何特殊工具或插件即可实现完整的前端热部署功能:只需使用您的IDE将资源发布到服务器即可(例如, Ctrl+F10 IntelliJ中的Ctrl+F10 )并刷新浏览器页面。

仍然可以使用JRebel重新加载后端类,但是对于前端而言 ,不需要任何特殊的操作。 实际上,可以通过使用json-server模拟后端来构建整个前端。 如果需要,这将允许不同的开发人员并行构建前端和后端。

全栈开发的生产力收益?

根据我的经验,能够直接编辑Html和CSS,而无需中间的任何间接层(请参阅此处, 与GWT和JSF进行的高级Angular比较 )有助于减少工作量并使事情简单。 编辑保存刷新开发周期非常快速且可靠,极大地提高了生产率。

当相同的开发人员同时构建Javascript前端和Java后端时,将获得最大的生产率提高,因为大多数功能通常都需要同时更改两者。

这样做的潜在缺点是,开发人员还需要了解Html,CSS和Javascript,但这在最近几年中变得越来越常见。

根据我的经验,与等效的完整Java解决方案相比,使用全栈可以在很短的时间内实现复杂的前端用例(用几天而不是几周的时间),因此提高生产率使学习曲线绝对值得。

结论

Spring MVC和Angular的结合确实为构建表单密集型Web应用程序的新方法打开了大门。 这种方法所带来的生产率提高使其成为值得研究的替代方案。

请求之间不存在任何服务器状态(除了身份验证cookie),通过设计消除了整个类别的错误。

有关更多详细信息,请查看github上的该示例应用程序 ,并让我们知道您对以下注释的想法/问题。

翻译自: https://www.javacodegeeks.com/2015/01/web-app-architecture-the-spring-mvc-angularjs-stack.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/360899.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

antd Datepicker组件报错 ——date.clone is not a function或者date1.isAfter is not a function

问题描述&#xff1a; antd Datepicker组件报错 ——date.clone is not a function或者date1.isAfter is not a function 原因分析&#xff1a; 在From中渲染默认值&#xff0c;一般数据请求拿到返回值存在异步&#xff0c;会晚于渲染&#xff0c;因此日期转换不能放在DatePi…

集成CDI和WebSockets

考虑尝试一个简单的Java EE 7原型应用程序&#xff0c;该应用程序涉及JAX-RS&#xff08;REST&#xff09;&#xff0c;WebSockets和CDI。 注意 &#xff1a;不想让它成为破坏者-但这篇文章主要讨论了我在尝试使用Web套接字和使用CDI作为“胶水”的REST&#xff08;在Java EE应…

Java中连接字符串的最佳方法

最近有人问我这个问题–在Java中使用运算符连接字符串是否对性能不利&#xff1f; 这让我开始思考Java中连接字符串的不同方法&#xff0c;以及它们如何相互对抗。 这些是我要研究的方法&#xff1a; 使用运算符 使用StringBuilder 使用StringBuffer 使用String.concat() …

十大最常见的Java性能问题

Java性能是所有Java应用程序开发人员都关心的问题&#xff0c;因为快速使应用程序与使其正常运行同等重要。 史蒂文海恩斯&#xff08;Steven Haines&#xff09;使用他在Java性能问题上的个人经验得出的结论是&#xff0c; 大多数问题都有共同的根本原因 。 因此&#xff0c;作…

Unity3D 访问Access数据库

Unity3D 访问Access数据库 在开始这个小教程之前呢&#xff0c;其实在网上你已经可以找到相关的资料了&#xff0c;但是我还是要把我自己做练习的一点东西分享出来。写这个教程的主要原因呢&#xff0c;是一个朋友在u3d的官网论坛里&#xff0c;找到了这个demo&#xff0c;但是…

LaTeX 基础笔记。开篇

LaTeX 的起源非常牛逼&#xff0c;有一套书大家可能听说过《计算机程序设计艺术》&#xff0c;写了好几本。当然能在计算机方面写上艺术俩字的书恐怕不是我们一般人能读懂得东西了。他的作者在1976年准备写第二卷的时候发现计算机的排版非常难看&#xff0c;所以&#xff0c;为…

Java旧版不断发展

我最近偶然发现了JDK API的一个非常有趣的警告&#xff0c;即Class.getConstructors()方法。 它的方法签名是这样的&#xff1a; Constructor<?>[] getConstructors()有趣的是&#xff0c; Class.getConstructor(Class...)返回一个Constructor<T> &#xff0c;并…

带Lambda表达式的Apache Wicket

这是怎么回事&#xff1f; :) 我一直在从事一些项目&#xff0c;这些项目值得庆幸的是将Apache Wicket用于表示层。 我自然想到Java的8个lambda表达式如何与Wicket完美匹配。 而不仅仅是我&#xff0c; Wicket团队似乎已经在努力更改API&#xff0c;以为开箱即用的lambda提供支…

装饰者模式如何拯救了我的一天

在工作中&#xff0c;我正在处理庞大的Java代码库&#xff0c;该代码库是由许多不同的开发人员在15年的时间里开发的。 并不是所有的事情都由书来完成&#xff0c;但是同时我通常没有机会重构遇到的每一个奇怪之处。 尽管如此&#xff0c;仍可以每天采取提高代码质量的措施。 …

快速的骆驼和云消息传递

Apache Camel是一个流行的&#xff0c;成熟的开源集成库。 它实现了企业集成模式 &#xff0c;这是在集成分布式系统时经常出现的一组模式。 过去&#xff0c;我写过很多关于Camel的文章&#xff0c; 包括为什么我比Spring Integration更喜欢它 &#xff0c; 路由引擎 如何 工作…

三角形类1

/* 程序的版权和版本声明部分 Copyright (c)2012, 烟台大学计算机学院学生 All rightsreserved. 文件名称&#xff1a; object.cpp 作者&#xff1a;刘清远 完成日期&#xff1a; 2013年3月29日 版本号&#xff1a; v1.0 输入描述&#xff1a;无 问题描述&#xff1a;设计求三…

android 自定义xml属性

Android 自定义组件 Android 提供了非常精致的和非常强大的组件化模型&#xff0c;能够更加方便的构建UI,这些UI组件都是基于基本的layout类:View 和 ViewGroup。 部分能够用的widgets包括&#xff1a;Button&#xff0c;TextView,EditText,ListView,CheckBox&#xff0c;Radio…

LeetCode: Longest Common Prefix

string.erase没掌握好&#xff0c;悲了个剧&#xff0c;2次过 1 class Solution {2 public:3 string longestCommonPrefix(vector<string> &strs) {4 // Start typing your C/C solution below5 // DO NOT write int main() function6 s…

流式传输大数据:Storm,Spark和Samza

有许多分布式计算系统可以实时或近实时处理大数据。 本文将从对三个Apache框架的简短描述开始&#xff0c;并试图对它们之间的某些相似之处和不同之处提供一个快速的高级概述。 阿帕奇风暴 在风暴 &#xff0c;你设计要求的T opology实时计算的图&#xff0c;然后喂到集群&…

uniapp使用阿里云多色图标

下载&#xff0c;然后解压 输入cmd&#xff0c;然后enter 输入 npm install -g iconfont-tools 再输入 iconfont-tools&#xff0c;然后一直enter&#xff0c;直到结束 目录会多了个iconfont-weapp文件&#xff0c;点击去找到 iconfont-weapp-icon.css 导入和使用 t-icon开头 接…

针对Java中的XSD验证XML

有许多工具可用于根据XSD 验证XML文档 。 其中包括操作系统脚本和工具&#xff0c;例如xmllint &#xff0c;XML编辑器和IDE&#xff0c;甚至是在线验证器。 由于前面提到的方法的局限性或问题&#xff0c;我发现拥有自己的易于使用的XML验证工具很有用。 Java使编写这样的工具…

uniapp uni.request GET方式请求,不能直接传数组解决方法

这里写目录标题目录遇到的问题 GET请求方法传数组解决方案目录 遇到的问题 GET请求方法传数组 想传一个数组&#xff0c;但是后台接受到的数据与浏览器中显示的数据和前台代码传的不一样&#xff1b; 前台代码打印 浏览器显示数据 其中HerbalNameList &#xff0c;变成了字…

休眠CascadeType.LOCK陷阱

介绍 引入了Hibernate 显式锁定支持以及Cascade Types之后 &#xff0c;就该分析CascadeType.LOCK行为了。 休眠锁定请求触发内部LockEvent 。 关联的DefaultLockEventListener可以将锁定请求级联到锁定实体子级。 由于CascadeType.ALL也包括CascadeType.LOCK &#xff0c;因…

c++中在堆和栈中申请空间的差别

堆中和栈中申请的空间的比较, 我找到了下面的比较: 栈的情况&#xff1a;栈上分配空间的好处是快&#xff0c;而且对象生存期是自动的&#xff0c;离开当前域之后就自动析构回收。坏处就是栈空间有限&#xff0c;而且不能人为控制对象的生存期&#xff0c;比如你无法将一个函数…

推销自己的海盗猫王运营商

因此&#xff0c;Java没有Elvis运算符&#xff08;或者更正式的名称是null合并运算符或null安全成员选择&#xff09;……虽然我个人不太在意它&#xff0c;但有些人似乎很喜欢它。 当一位同事需要几天后&#xff0c;我坐下来探讨了我们的选择。 而且你知道什么&#xff01; 您…