我在php中开发了一个基本的MVC框架作为学习项目——这实际上是它的第二个版本,我正在努力改进第一个版本不足的两个方面:
我能够接收请求并将其解析为多个部分, 例如controller, action, args 等。这些映射到相应的 Controller 类/文件,例如"/foo/bar"-> FooController::bar() - 所有这些都在我的 RequestRouter 类中完成并封装在 请求对象。
Controller::methods() 可以很好地呈现正确的 View 。
然后是模块,它们的组织就像核心的结构一样 (/root/raspberry/vendors/core/module)
我认为我目前遇到的问题是与模块相关的路由/请求处理的组合:
经过一些研究,我认为我可以创建一个Decorator,它实现了一个Front Controller 模式并包装了一个给定的Controller。装饰器可以重新解析请求以创建/editor Controller 并重新映射剩余的段 (/editor/action/args)。
所有这些看起来都可以正常工作,但我觉得我在流程的早期缺少一些基本的东西 (RequestRouter)。我在 SO 中研究了其他类似的问题,并阅读了 HMVC,原则上它似乎可以回答我的问题,但它似乎比框架驱动更多的是接口(interface)驱动(如果这有意义的话?)我已经还研究了 Kohana 等其他框架,但我不太了解它们的模块系统和路由到同一模块中多个 Controller 的工作原理。
任何有关如何在不引入前端 Controller 或重新解析请求的情况下有效实现模块系统的见解或建议,将不胜感激。或者,如果我应该以不同的方式重新构造我的模块,我想了解如何做到这一点。
我的 RequestRouter 维护了一个我预定义的路由列表(包括它们的默认方法)。使用这些预定义的路由,我可以访问/admin/editor 并获取 EditorController::index(),但我必须为每个 Controller 定义一个路由,并请求转到模块。我不认为这是好的设计。这是我的路线示例:
Array
(
[/foo] => Array
(
[controller] => FooController
[method] => bar
[path] => /core
)
[/admin] => Array
(
[controller] => AdminController
[method] => index
[path] => /vendors/admin
)
[/admin/editor] => Array
(
[controller] => EditorController
[method] => index
[path] => /vendors/admin
)
)
这是我的请求对象的样子:
Request Object
(
[properties:Request:private] => Array
(
[url] => /admin/editor
[query] =>
[uri] => /admin/editor
[controller] => admin
[action] => editor
[args] =>
[referrer] => Array
(
[HTTP_REFERER] =>
[REMOTE_ADDR] => 127.0.0.1
[HTTP_VIA] =>
[HTTP_X_FORWARDED_FOR] =>
)
[get] =>
)
[request_status:Request:private] => 200
)
这是我的 list 示例:
[controller] => Array
(
[icontroller] => /htdocs/raspberry/raspberry/core/controller/icontroller.class.php
[index] => /htdocs/raspberry/raspberry/core/controller/index.php
[serviceerror] => /htdocs/raspberry/raspberry/core/controller/serviceerror.controller.php
[admin] => /htdocs/raspberry/raspberry/vendors/core/admin/controller/admin.controller.ph
[composer] => /htdocs/raspberry/raspberry/vendors/core/admin/controller/composer.controller.php
)
这是应用程序文件系统:
http://s11.postimage.org/pujb2g9v7/Screen_shot_2012_10_09_at_8_45_27_PM.png
最佳答案
这个问题似乎是由过度简化的路由机制引起的。我得到的印象是您使用的是简单的 explode()从 URL 收集参数。虽然这仅适用于基本示例,但当您尝试使用更高级的路由方案时,设置将失败。
而不是在 / 上拆分字符串,您应该将其与正则表达式模式进行匹配。基本上,您定义要匹配的模式列表,第一个匹配项用于填充 Request实例。
在您的情况下,将定义两种模式:
'#admin/(:?(:?/(?P<controller>[^/\.,;?\n]+))?/(?P<action>[^/\.,;?\n]+))?#' '#(:?(:?/(?P<controller>[^/\.,;?\n]+))?/(?P<action>[^/\.,;?\n]+))?#' 如果第一个失败,第二个会匹配。
P.S. 您应该知道 Controller 不应该渲染输出。响应的生成是 View 实例的责任。 View 应该是功能齐全的对象,其中包含表示逻辑并可以从多个模板中组合响应。
关于php - 如何在 MVC 框架中有效地实现模块,并在单个模块中处理到多个 Controller 的路由?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12810899/
假设我做了一个模块如下:m=Module.newdoclassCendend三个问题:除了对m的引用之外,还有什么方法可以访问C和m中的其他内容?我可以在创建匿名模块后为其命名吗(就像我输入“module...”一样)?如何在使用完匿名模块后将其删除,使其定义的常量不再存在? 最佳答案 三个答案:是的,使用ObjectSpace.此代码使c引用你的类(class)C不引用m:c=nilObjectSpace.each_object{|obj|c=objif(Class===objandobj.name=~/::C$/)}当然这取决于
出于纯粹的兴趣,我很好奇如何按顺序创建PI,而不是在过程结果之后生成数字,而是让数字在过程本身生成时显示。如果是这种情况,那么数字可以自行产生,我可以对以前看到的数字实现垃圾收集,从而创建一个无限系列。结果只是在Pi系列之后每秒生成一个数字。这是我通过互联网筛选的结果:这是流行的计算机友好算法,类机器算法:defarccot(x,unity)xpow=unity/xn=1sign=1sum=0loopdoterm=xpow/nbreakifterm==0sum+=sign*(xpow/n)xpow/=x*xn+=2sign=-signendsumenddefcalc_pi(digits
作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代
Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题
我有多个ActiveRecord子类Item的实例数组,我需要根据最早的事件循环打印。在这种情况下,我需要打印付款和维护日期,如下所示:ItemAmaintenancerequiredin5daysItemBpaymentrequiredin6daysItemApaymentrequiredin7daysItemBmaintenancerequiredin8days我目前有两个查询,用于查找maintenance和payment项目(非排他性查询),并输出如下内容:paymentrequiredin...maintenancerequiredin...有什么方法可以改善上述(丑陋的)代
如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby
我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%
Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack
我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何
我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>