草庐IT

performance - 为什么Grails推荐使用 Action 作为方法的 Controller 的单例作用域?

coder 2023-06-04 原文

我知道Grails的早期版本使用了 Controller 的原型(prototype)作用域,因为当时所有 Action 都是封闭的。

我知道当前版本的文档建议使用方法作为操作的 Controller 的单例作用域 Controller 。

从下面的帖子中,似乎更希望或推荐使用方法和单例作用域,但尚不清楚原因。

ttp://grails.1312388.n4.nabble.com/Default-scope-for-controllers-doc-td4657986.html

我们有一个大型项目,该项目使用原型(prototype)作用域 Controller 将 Action 作为方法。更改为建议的 Controller 范围会涉及风险和重新测试,以及从现有 Controller 中删除任何非单点友好状态。

我想知道,为什么Grails建议将单例作用域用作方法作为 Action Controller ?仅仅是因为这与Spring MVC更为常见且相似,并且避免了混淆,或者是否有提高性能的机会,还是什么? 如果切换到单例 Controller ,我将获得什么? 如果我不进行切换,费用是多少?

最佳答案

我与Rails的合作并不多,但是(至少在我玩过的版本中,现在可能有所不同) Controller 是模型,其中包含要由 View 呈现的数据。在请求处理期间,您将值存储在 Controller 实例字段中,然后再进行处理以查看渲染器。因此,为每个请求创建一个新的 Controller 实例是有意义的,因此它们是不同的。

Grails受到Rails的启发,并使用了它的几种约定,最著名的是over over configuration。但是,虽然没有充分的文档记录,但我也怀疑很多人使用了它,但是还增加了使用 Controller 作为模型的功能。

使用GSP呈现响应(与转发或重定向,或直接在 Controller 中呈现,例如render foo as JSON相反)时, Controller Action 的典型工作方式是从 Action 中返回一个或多个键/值对,并且通常省略return关键字,因为它在Groovy中是可选的:

class MyController {
   def someAction() {
      def theUser = ...
      def theOtherObject = ...
      [user: theUser, other: theOtherObject]
   }
}

在这里,模型图有两个条目,一个由user键入键,另一个由other键入键,这些将成为GSP中用于访问数据的变量名。

但是大多数人不知道的是,您也可以这样做:
class MyController {

   def user
   def other

   def someAction() {
      user = ...
      other = ...
   }
}

在这种情况下,不会从操作返回模型映射,因此Grails将从 Controller 类的所有属性中填充模型,并且在这种情况下,由于第二种方法中的变量名是与第一个中的 map 键相同。

使 Controller 成为单例的选项是在2.0中添加的(在重命名为2.0之前,在技术上为1.4,请参见this JIRA issue),并且除了保留对闭包的支持之外,我们还添加了对方法作为操作的支持。最初的假设是使用闭包将启用一些有趣的功能,但从未实现。使用方法更为自然,因为您可以在子类中覆盖它们,这与仅属于类作用域的闭包不同。

作为2.0返工的一部分,我们删除了受Rails启发的功能,并假设该功能基本上没有记录,因此对使用该功能的少数奇特应用程序的影响不大。我不记得有人提示失去该功能。

尽管 Controller 类通常很容易被垃圾回收,并且每个请求创建一个实例影响不大,但是很少需要 Controller 中每个请求的状态,因此单例通常更有意义。保留了默认的原型(prototype)范围是为了向后兼容,但是可以通过Config.groovy属性轻松更改默认原型(prototype)(由create-app脚本生成的文件可以做到这一点)。

尽管每个请求的确获得了新的请求和响应,并且如果使用了 session ,则每个用户将拥有自己的 session ,但这不是 Controller 的实例字段。它们看起来像是因为我们可以在 Action 内部访问requestresponsesessionparams等,但实际上它们是getRequest()getResponse()getSession()getParams()方法的属性访问形式,这些形式在编译期间已混合到所有 Controller 中通过AST转换。这些方法不是通过类字段而是通过ThreadLocals访问它们的对象,因此存在每个请求的状态,但未存储在 Controller 实例中。

我不记得在基准测试中使用方法和闭包进行比较的方式是否很多,但是Lari Hotari可能做了一些。如果存在差异,则可能并不重要。您可以通过仅转换一个或几个 Controller 并在测试之前和之后进行操作,来在自己的应用程序中对其进行测试。如果这两种方法之间的性能,扩展性和内存差异不大,则可以放心使用原型(prototype)分数和/或闭包。如果存在差异,并且您的 Controller 中没有实例字段,那么转换为单例和方法可能是值得的。

如果确实有实例字段,则它们可能可以转换为请求属性-request.foo = 42request.setAttribute('foo', 42)的元类快捷方式,因此您可以安全地在其中存储每个请求的数据。

关于performance - 为什么Grails推荐使用 Action 作为方法的 Controller 的单例作用域?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29682673/

有关performance - 为什么Grails推荐使用 Action 作为方法的 Controller 的单例作用域?的更多相关文章

  1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

    我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

  2. ruby - 使用 RubyZip 生成 ZIP 文件时设置压缩级别 - 2

    我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看ruby​​zip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d

  3. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  4. ruby-on-rails - 使用 Ruby on Rails 进行自动化测试 - 最佳实践 - 2

    很好奇,就使用ruby​​onrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提

  5. ruby - 在 Ruby 中使用匿名模块 - 2

    假设我做了一个模块如下:m=Module.newdoclassCendend三个问题:除了对m的引用之外,还有什么方法可以访问C和m中的其他内容?我可以在创建匿名模块后为其命名吗(就像我输入“module...”一样)?如何在使用完匿名模块后将其删除,使其定义的常量不再存在? 最佳答案 三个答案:是的,使用ObjectSpace.此代码使c引用你的类(class)C不引用m:c=nilObjectSpace.each_object{|obj|c=objif(Class===objandobj.name=~/::C$/)}当然这取决于

  6. ruby - 使用 ruby​​ 和 savon 的 SOAP 服务 - 2

    我正在尝试使用ruby​​和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我

  7. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  8. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

    我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

  9. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

    我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

  10. ruby - 使用 ruby​​ 将 HTML 转换为纯文本并维护结构/格式 - 2

    我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h

随机推荐