规则重定向到子域名

11

谁定义要重定向到移动版的URL?

我们以Twitter为例:

  • https://twitter.com -> 重定向到移动版
  • https://dev.twitter.com -> 不重定向到移动版

在MVC应用程序中,谁负责规则重定向?
所有位于www中的内容都可以重定向到移动设备吗?


1
你的前端控制器应该负责检测和重定向移动请求。 - Supericy
1
这个问题对我来说还不太清楚。每个控制器都需要为移动设备编写一个解释器吗? - Papa Charlie
4个回答

4
一种方法是使用 .htaccess 或等效工具从服务器(apache 不是唯一的 Web 服务器)进行重定向并强制重定向。这已经在其他答案中涵盖了。但还有另一种不同的方法。一种实际上会利用 MVC 的方法:根本不要重定向。

MVC 设计模式的简要背景

在正确实现的 MVC 中,视图实例将包含所有 UI 逻辑。它们将从模型层获取数据(“如何”构成 Model2、MVP 和 MVVM 之间的大部分差异),决定从哪些模板组装响应,甚至是否需要更多内容的响应仅使用 HTTP 位置标头。

移动版和桌面版之间的差异将包含在视图实例中,由控制器协助,在正确的 MVC 结构中,控制器只会更改模型层和当前视图的状态。它所做的更改应取决于用户输入。

设置所有内容

以下代码将成为 bootstrap.php 或 init.php 的一部分:

// the request instance acts like abstraction  for all the user input
$request = new Request;
$request->setUri();

// instantiate the routing mechanism
$router = new Router( new RouteBuilder );
$router->import('/path/to/config.file');

// apply rules from router so that request instance now 
// contains the parsed values from URI
$router->route( $request );

// handling of model layer
$serviceFactory = new ServiceFactory;

// since in MVC the controllers are closely tied to views 
// (1 controller for 1 view), usually it is convenient to use same class names
$resource = $request->getParameter('resource');
    
// instantiation of view and controller
$class = '\\View\\' . $resource;
$view = new {$class}( $serviceFactory );

$class = '\\Controller\\' . $resource;
$controller = new {$class}( $serviceFactory, $view);

// i find it convenient to have controller's action be made from 
// both REQUEST_METHOD and command name
$action = $request->getMethod() . $request->getParameter('command');

// run it all
$controller->{$action}( $request );
echo $view->render();

当执行命中控制器的操作时,它会提供一个完全准备好的Request实例。该实例确定了关于用户设备的详细信息,并提供了一个简单的接口来读取这些详细信息。
控制器还可以访问模型层和当前视图,这两者都是通过构造函数注入的。
决定显示什么内容。
最直接的方法是让控制器改变当前视图的状态。
namespace Conroller;

class SomeThing
{
    public function getUserDetails( $request )
    {
        if ( $request->isFromMobile() )
        {
            $this->view->adjustFor( $request->getDeviceType() );
        }
       
        $community = $this->serviceFactory->create('Community');
        $community->loadUser( $request->getParameter('id'));
    }
}

在这种情况下,adjustFor()方法通知当前视图实例,它需要使用为某些非默认设备准备的模板。
但是这种方法有一个非常重要的缺点:它违反了SOLID原则中的OCP(对于懒人来说:简短版本),因为如果您决定为现有项目添加移动版,则必须重新编写每个控制器方法。
有更好的方法
尽管以下代码相对容易理解,但存在一些缺陷:
$resource = $request->getParameter('resource');

// instantiation of view and controller
$class = '\\View\\' . $resource;
$view = new {$class}( $serviceFactory );

当您只需要提供HTML和JSON/XML响应时,它甚至开始崩溃。视图开始在整个代码中累积相同重复的IF。这是一个明确的迹象,表明您应该使用多态性,而这些行将是执行它的位置
与上面显示的代码不同,您可以使用类似以下的内容:
$resource = $request->getParameter('resource');

$class = '\\View\\' . $request->getDeviceType . $resource;
$view = new {$class}( $serviceFactory );

现在,当你拥有移动端/桌面应用时,你有两个类:\View\DekstopSomething\View\MobileSomething。它们各自可以具有不同的逻辑,并从模型层请求完全不同的数据。
同时,您的其余代码与输出形式完全解耦。
好处是什么?
以下是使用此方法而不是使用服务器重定向的几个原因:
1. 您的应用程序变得独立于服务器软件
并非所有地方都有Apache(高负载站点经常使用Nginx或Lighttpd),即使您有Apache,使用mod_rewrite的能力也取决于服务器的配置。
2. 网站中所有链接的统一方案
查看某个新闻项目的链接始终相同,无论您使用什么设备。这样更容易共享和收藏URL。
3. 更改的单一点
这种方法使您可以最初为桌面用户创建站点,然后添加移动/平板支持,而无需重写任何现有代码。

您可能也会对阅读有关MVC上下文中模型层访问控制实现的两篇旧文章感兴趣。


2

要检测移动用户,取决于您的方式。可以通过重写规则由Web服务器完成。也可以通过PHP代码中的用户代理脚本进行检测,在这种情况下,它将由控制器完成。由于有很多选择,因此可能会令人困惑。Google在这方面有很多具体信息。

重写示例:

RewriteCond %{HTTP_USER_AGENT}  ^.*iPhone.*
RewriteRule ^(.*)$         m.site.com/$1

如果我使用.htaccess重定向,任何子域都会被包括在内,甚至是超出规则范围的子域。 - Papa Charlie
htaccess在旧的手机时代可能更有用,因为您可以通过MIME(即WAP)进行检测。话虽如此,Apache mod_rewrite将允许使用用户代理进行重定向。请参见上面的示例。 - BlueRebel

0

正如@Supericy所说,在大多数情况下,前端(视图)负责检测,因为它的名字就是它所显示给用户的内容。

例如,如果您使用移动检测类,只需在您的index.php或类似文件的顶部编写,如果您打算使用header()函数。

<?php
include 'Mobile_Detect.php';
$detect = new Mobile_Detect();

if ($detect->isMobile()) {
    // Any mobile device. Do stuff
} else {

}

?>

在移动设备检测类页面https://github.com/serbanghita/Mobile-Detect上有更多的示例。您可以针对设备进行非常具体的检测。

还有很多其他方法可以实现这一点,但在我看来,这是最简单的方法。


0
检测移动设备的用户最好交给您的服务器处理,因为它应该控制所有流量。如果您的代码留在不同的Web根目录中,或者您不需要向使用桌面和移动设备的用户显示相同的页面,则以下是一段Apache配置代码,可以解决您的问题。请务必访问http://detectmobilebrowsers.com/以获取移动设备检测正则表达式的新版本。
# if already mobile - skip this (usually there are will be another config for mobile)
RewriteCond %{HTTP_HOST} !^mobile\.
# put all sites you don't want to redirect here, or compose a regex that will match sites
RewriteCond %{HTTP_HOST} !(dev.twitter.com) 

# taken from http://detectmobilebrowsers.com/
RewriteCond %{HTTP_USER_AGENT} (android|bb\d+|meego).+mobile|avantgo|bada\/|blackberry|blazer|compal|elaine|fennec|hiptop|iemobile|ip(hone|od|ad)|iris|kindle|lge\ |maemo|midp|mmp|netfront|opera\ m(ob|in)i|palm(\ os)?|phone|p(ixi|re)\/|plucker|pocket|psp|series(4|6)0|symbian|treo|up\.(browser|link)|vodafone|wap|windows\ (ce|phone)|xda|xiino [NC,OR]
RewriteCond %{HTTP_USER_AGENT} ^(1207|6310|6590|3gso|4thp|50[1-6]i|770s|802s|a\ wa|abac|ac(er|oo|s\-)|ai(ko|rn)|al(av|ca|co)|amoi|an(ex|ny|yw)|aptu|ar(ch|go)|as(te|us)|attw|au(di|\-m|r\ |s\ )|avan|be(ck|ll|nq)|bi(lb|rd)|bl(ac|az)|br(e|v)w|bumb|bw\-(n|u)|c55\/|capi|ccwa|cdm\-|cell|chtm|cldc|cmd\-|co(mp|nd)|craw|da(it|ll|ng)|dbte|dc\-s|devi|dica|dmob|do(c|p)o|ds(12|\-d)|el(49|ai)|em(l2|ul)|er(ic|k0)|esl8|ez([4-7]0|os|wa|ze)|fetc|fly(\-|_)|g1\ u|g560|gene|gf\-5|g\-mo|go(\.w|od)|gr(ad|un)|haie|hcit|hd\-(m|p|t)|hei\-|hi(pt|ta)|hp(\ i|ip)|hs\-c|ht(c(\-|\ |_|a|g|p|s|t)|tp)|hu(aw|tc)|i\-(20|go|ma)|i230|iac(\ |\-|\/)|ibro|idea|ig01|ikom|im1k|inno|ipaq|iris|ja(t|v)a|jbro|jemu|jigs|kddi|keji|kgt(\ |\/)|klon|kpt\ |kwc\-|kyo(c|k)|le(no|xi)|lg(\ g|\/(k|l|u)|50|54|\-[a-w])|libw|lynx|m1\-w|m3ga|m50\/|ma(te|ui|xo)|mc(01|21|ca)|m\-cr|me(rc|ri)|mi(o8|oa|ts)|mmef|mo(01|02|bi|de|do|t(\-|\ |o|v)|zz)|mt(50|p1|v\ )|mwbp|mywa|n10[0-2]|n20[2-3]|n30(0|2)|n50(0|2|5)|n7(0(0|1)|10)|ne((c|m)\-|on|tf|wf|wg|wt)|nok(6|i)|nzph|o2im|op(ti|wv)|oran|owg1|p800|pan(a|d|t)|pdxg|pg(13|\-([1-8]|c))|phil|pire|pl(ay|uc)|pn\-2|po(ck|rt|se)|prox|psio|pt\-g|qa\-a|qc(07|12|21|32|60|\-[2-7]|i\-)|qtek|r380|r600|raks|rim9|ro(ve|zo)|s55\/|sa(ge|ma|mm|ms|ny|va)|sc(01|h\-|oo|p\-)|sdk\/|se(c(\-|0|1)|47|mc|nd|ri)|sgh\-|shar|sie(\-|m)|sk\-0|sl(45|id)|sm(al|ar|b3|it|t5)|so(ft|ny)|sp(01|h\-|v\-|v\ )|sy(01|mb)|t2(18|50)|t6(00|10|18)|ta(gt|lk)|tcl\-|tdg\-|tel(i|m)|tim\-|t\-mo|to(pl|sh)|ts(70|m\-|m3|m5)|tx\-9|up(\.b|g1|si)|utst|v400|v750|veri|vi(rg|te)|vk(40|5[0-3]|\-v)|vm40|voda|vulc|vx(52|53|60|61|70|80|81|83|85|98)|w3c(\-|\ )|webc|whit|wi(g\ |nc|nw)|wmlb|wonu|x700|yas\-|your|zeto|zte\-) [NC]

# rewrite only sites.
RewriteCond %{HTTP_HOST} (\.|)([^\.]+\.[^\.0-9]+)$

#redirect to mobile
RewriteRule ^(.*)$ http://mobile.twitter.com$1 [L,NS]

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接