我自己添加的注释本文讲述了HTTP请求处理阶段的内容,非常棒的一篇文章。部分重点内容翻译如下,请大家阅读。
Nginx processes HTTP requests in multiple phases. In each of the phases there might be 0 or more handlers called. In the Nginx source code phases have specific constants associated with them. Here is a list of all phases:
Nginx以多个阶段(phases)处理HTTP请求。对于每个阶段可能有0个多个handler被调用。在Nginx源码中都有常量名称标识各个阶段,下面是所有阶段的列表:
On every phase you can register any number of your handlers.Exceptions are following phases:
在每个阶段你都可以注册多个自定义的hanlders。除了下面几个阶段。
- NGX_HTTP_FIND_CONFIG_PHASE. On this phase no handlers are called, instead a search for configuration location is performed and “Location” request header is filled.
-
NGX_HTTP_POST_ACCESS_PHASE. On this phase no handlers are called,
only the result of access checks is interpreted and applied. The phase is required to implement directive satisfy all/any. -
NGX_HTTP_POST_REWRITE_PHASE. On this phase no handlers are called,
instead request URI transformation post-processing is performed; -
NGX_HTTP_TRY_FILES_PHASE. On this phase no handlers are called,
instead Nginx processes the try_files directive.
Each phase has a list of handlers associated with it. Once registered on a phase, handler can return one of the following values:
每个阶段都有一个与之相关的handler的列表。一旦把handler注册到对应的阶段,那么handler就会返回某个下面的值:
NGX_OK:请求已经成功处理,请求将会传到下一个阶段。
NGX_DECLINED:请求需要被转发到本阶段的下一个handler
NGX_AGAIN,NGX_DONE:请求已经被正确处理,同时请求被挂起,直到某个事件(子请求结束、socket可写或超时等)到来,handler才会再次被调用。
- NGX_OK — the request has been successfully processed, request must be routed to the next phase;
- NGX_DECLINED — request must be routed to the next handler;
- NGX_AGAIN, NGX_DONE — the request has been successfully processed, the request must be suspended until some event (e.g., subrequest finishes, socket becomes writeable or timeout occurs) and handler must be called again;
- NGX_ERROR, NGX_HTTP_… — an error has occurred while processing the request.
In order to register a handler you need to find the configuration of ngx_http_core_module and add the handler to one of elements of phases vector. Example:
为了在某个阶段注册自定义的handler,需要找到ngx_http_core_module,并添加handler到指定的phases 向量中。
static ngx_int_t ngx_http_sample_module_init(ngx_conf_t *cf) { ngx_http_handler_pt *h; ngx_http_core_main_conf_t *cmcf; cmcf = ngx_http_conf_get_module_main_conf(cf, ngx_http_core_module); h = ngx_array_push(&cmcf->phases[NGX_HTTP_CONTENT_PHASE].handlers); if (h == NULL) { return NGX_ERROR; } *h = ngx_http_sample_module_handler; return NGX_OK; }
The vector phases has one entry for each phase. Each entry contains a field handlers which is a vector of handlers that are registered on this phase.
Handlers are called in reverse order. Therefore the last handler registered at configuration time will be called first at runtime.
As you can see, the order of actions in request processing has nothing to do with the order of directives in configuration file. Phase handlers are called regardless of what configuration the user has specified. Hence a phase handler must be able to determine when it is applicable and return NGX_DECLINED when it is not and do it as fast as possible to avoid performance penalties.
从上面可以看出,请求处理的顺序和配置文件中的配置指令的先后顺序无关,无论配置文件中指令的顺序如何,各个阶段的处理函数都会按照预先的顺序执行。因此一个处理函数需要知道自己什么时候会被调用,何时需要返回NGX_DECLINED,而且保证减少性能损耗。
The phase NGX_HTTP_ACCESS_PHASE calls handlers that restrict access to resources. In this phase the order in which handlers are called is determined by directive satisfy. The values, that handlers return, have additional meaning:
- NGX_OK — handler allows to access the resource specified by request URI;
- NGX_HTTP_FORBIDDEN, NGX_HTTP_UNAUTHORIZED — handler does not allow to request the resource specified by request URI.
In case of satisfy all all handlers must return NGX_OK in order to proceed with the next phase.
In case of satisfy any at least one handler must return NGX_OK in order to proceed with the next phase.
Nginx uses the phase NGX_HTTP_CONTENT_PHASE to generate a response. Whenever a location configuration of ngx_http_core_module has handler field initialized, all requests on content phase are routed to this handler. The handler that is specified by handler field is hence called content handler. When content handler is not set, request is routed to handlers of content phase in main configuration.
Nginx使用阶段NGX_HTTP_CONTENT_PHASE产生响应。无论何时,ngx_http_core_module的location配置中都有handler字段的初始化,所有请求在content phase处理时都会转发到这个handler。这个handler是由handler字段指定。因此这个handler字段又称为content handler。当content handler不指定时,请求就会转发到主配置中的content phase的各个handler中。
这儿我经过验证,果真如此,即:如果不指定ngx_http_core_loc_conf_t ->handler,那么请求转发到默认的content phase中的handler(如ngx_http_index_handler,ngx_http_static_handler);但是如果指定了ngx_http_core_loc_conf_t ->handler,那么请求就会转发到该指定的handler中处理,生成内容。hello_world模块是后者,http://localhost输出welcome to Nginx!是前者。可以在函数内部输入printf,看输出,就可以更加明确这一点。
How to override content handler? Here is an example:
如何指定content handler呢?如下所示:
static char * ngx_http_sample_module_command(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) { ngx_http_core_loc_conf_t *clcf; clcf = ngx_http_conf_get_module_loc_conf(cf, ngx_http_core_module); clcf->handler = ngx_http_sample_handler; return NGX_CONF_OK; }
The content handler has following specific properties:
Content handler 有下面特殊的属性:
1)不能再被调用。就说说,当返回NGX_AGAIN与NGX_DONE时,content handler不会再被调用。取而代之的是handler必须改变请求的读或写handler
2)每个location都有他自己的content handler。
3)如果content handler 返回NGX_DECLINED,Nginx转发请求道content phase handlers。
- It is not resumable. That is, it’s not going to be called again if it returns NGX_AGAIN or NGX_DONE. Instead the handler must change read and/or write handlers of the request;
- Each location can have it’s own dedicated content handler;
- If content handler returns NGX_DECLINED, Nginx routes request to content phase handlers.
How do you find out which phase to put your handler on?
Although this blog is not to criticize Nginx, here seems to be a bit of a problem. According to my feelings and to Igor Sysoev comments, phases are legacy from Apache. They are not as flexible for a module developer as you would expect. Clearly this point could be improved (and it will be), but for the moment here are suggestions, that you can use in order to figure out what phase you need to put your handler on:
- Your handler needs to be manageable by satisfy directive (e.g. access restriction checks) — put it on access phase;
- Your handler does resource limitations — put it on pre-access phase;
- Your handler changes URI or manipulates variables than need to be accessible in set directive or rewrites — put it on rewrite phase;
- Your handler generates content — put it on content phase, take care of the handler registration order;
- Your handler does logging — put it on logging phase;
When you should use content handlers?
What is the difference between content phase handler and content handler?
- the content phase handler is promiscuous: it is called for every request that reaches the content phase and this particular handler. The content handler is called only for those requests that reach a location with configured content handler;
- more than one content phase handler can be called per location. Only one content handler can be called per location.
A combination of these two types of handlers are employed by nginx mogilefs module for doing PUT requests:
Main location is handled by a content phase handler. This handler has 3 stages that correspond to create_open command, storing the resource on storage node and create_close command. On each stage content phase handler does a subrequest. When subrequest finishes, it wakes up the main content phase handler. In case of create_open and create_close commands subrequest is routed to a hidden location that has a content handler set up in it. This handler implements communication with MogileFS tracker using upstream module.
做到效果如下的Tab组件的使用:
首先布局组件,在activity_tab_main.xml中的代码如下:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" > <ImageView android:id="@+id/image1" android:layout_width="wrap_content" android:layout_height="wrap_content" android:src="/blog_article/@drawable/star1/index.html"/> <ImageView android:id="@+id/image2" android:layout_width="wrap_content" android:layout_height="wrap_content" android:src="/blog_article/@drawable/star2/index.html"/> <ImageView android:id="@+id/image3" android:layout_width="wrap_content" android:layout_height="wrap_content" android:src="/blog_article/@drawable/star3/index.html"/> </RelativeLayout>
再在主要的Activity(TabMainActivity.java)中编写如下的代码:
package com.bzu.tab.activity; import android.os.Bundle; import android.app.Activity; import android.app.TabActivity; import android.content.res.Resources; import android.view.LayoutInflater; import android.view.Menu; import android.view.MenuItem; import android.widget.TabHost; import android.support.v4.app.NavUtils; //第一步:extends TabActivity public class TabMainActivity extends TabActivity { @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // 第二步:获取选项卡组 TabHost tabHost = getTabHost(); // 第三步:引用布局文件 LayoutInflater inflater = LayoutInflater.from(this); inflater.inflate(R.layout.activity_tab_main, tabHost.getTabContentView()); // 第四步:创建对象 Resources r = getResources(); TabHost.TabSpec tab01 = tabHost.newTabSpec("tab01").setIndicator("", r.getDrawable(R.drawable.star01)).setContent(R.id.image1); TabHost.TabSpec tab02 = tabHost.newTabSpec("tab02").setIndicator("", r.getDrawable(R.drawable.star02)).setContent(R.id.image2); TabHost.TabSpec tab03 = tabHost.newTabSpec("tab03").setIndicator("", r.getDrawable(R.drawable.star03)).setContent(R.id.image3); //第五步:把创建好的对象放入tabHost中 tabHost.addTab(tab01); tabHost.addTab(tab02); tabHost.addTab(tab03); } @Override public boolean onCreateOptionsMenu(Menu menu) { getMenuInflater().inflate(R.menu.activity_tab_main, menu); return true; } }
这里只说我在写程序时遇到段错误的原因,如何解决的
总之,涉及到指针的,你不能在未定义,未初始化,未 “new” 之前使用,否则等待你的肯定是 segment fault,然后程序直接退出
1. 类A在执行中可能会几次用到用到类B,也可能只有一次用到,也可能一次都不用,这种情况下我的习惯就是在类A的构造函数中初始化B类对象为NULL,然后在用的地方先判断B是否实例化,未实例化,实例化,然后使用,delete B类对象时也要先判断下B是否实例化
因为这里已经初始化了类B的对象,使用时容易出的错误是在判断是否实例化时,如果在判断类本身是否实例化之前,判断了(或者说使用了)类B的成员函数/成员变量,出现段错误,因为这个类不存在,其成员函数/变量也不存在
举个我用qt做播放器的例子,其中调用到mplayer播放器:
发生错误的时候,我在mainwindow.h 里 有定义了个widget,用于将视频播放的位置固定在这个widget上。
private:
QWidget *widget;
然后在mainwindow.cpp里,我调用这个窗口:
QString common = "mplayer -slave -quiet -ac mad -zoom movie/" + file_name + " -wid " + QString::number(widget->winId());
程序运行后果然提示段错误!后来了发现是在调用widget的时候未初始化,即要在之前:
widget = new QWidget(this);
QString common = "mplayer -slave -quiet -ac mad -zoom movie/" + file_name + " -wid " + QString::number(widget->winId());
这样就可以正常运行了!