`
kenby
  • 浏览: 724579 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Yaf源码阅读之请求的处理(二)

    博客分类:
  • web
 
阅读更多

上一篇文章 我们分析了Yaf框架的启动,包括配置文件的初始化以及框架类的加载。本文将分析Yaf

处理一次Web请求的详细过程,这是一个Web框架最核心的部分,理解了这点,就理解了Yaf的实现原理。

 

(一)框架类的实例化

在上一篇文章提到了Yaf框架类的加载,类加载完成后,还要实例化,

我们逐一分析Yaf各个框架类的构造函数。

 

不急于分析各种调用关系,我们先看下Application启动后,Yaf的对象关系如图1-1所示。


   

                                                                图1-1 Yaf对象关系图

 

一切构造从 new Yaf_Application() 开始。

 

所有类的实例化都从Yaf_Applicaion的构造函数开始,它主要做了这几件事:

  • 实例化Config
  • 实例化Request
  • 实例化Dispatcher
  • 实例化Loader

yaf_application.c

PHP_METHOD(yaf_application, __construct) {
	//实例化Config,用于读取配置信息
	zconfig = yaf_config_instance(NULL, config, section TSRMLS_CC);

	//解析application的配置信息,将配置参数保存到全局变量中
	yaf_application_parse_option(
				zend_read_property(yaf_config_ce,
								   zconfig, 
								   ZEND_STRL(YAF_CONFIG_PROPERT_NAME),
				1 TSRMLS_CC) TSRMLS_CC)

	//实例化Request,保存请求的信息
	request = yaf_request_instance(NULL, YAF_G(base_uri) TSRMLS_CC);

	//实例化分发器Dispatcher,这是Yaf的核心控制类
	zdispatcher = yaf_dispatcher_instance(NULL TSRMLS_CC);
	yaf_dispatcher_set_request(zdispatcher, request TSRMLS_CC);

	//把config实例设置成Application的属性
	zend_update_property(yaf_application_ce, self, 
						 ZEND_STRL(YAF_APPLICATION_PROPERTY_NAME_CONFIG), 
						 zconfig TSRMLS_CC);

	//把request实例设置成Application的属性
	zend_update_property(yaf_application_ce, self, 
						 ZEND_STRL(YAF_APPLICATION_PROPERTY_NAME_DISPATCHER), 
						 zdispatcher TSRMLS_CC);

	//创建loader,用于自动加载类
	loader = yaf_loader_instance(NULL, local_library,
					strlen(YAF_G(global_library))? YAF_G(global_library) : NULL TSRMLS_CC);

	//把dispatcher设置成Application的属性
	zend_update_property(yaf_application_ce, self, 
						 ZEND_STRL(YAF_APPLICATION_PROPERTY_NAME_DISPATCHER), 
						 zdispatcher TSRMLS_CC);
}

 

在Yaf_Application的构造函数中,调用其他类的构造函数时,又会继续创建它依赖的其它对象,

最终所有框架类都完成实例化,形成图1-1所示的关系图,然后就可以开始处理用户的请求了。

 

(二)处理请求

几乎所有MVC的Web框架处理一个请求都会走过3个流程,如图2-1所示。


                                                    图2-1 MVC框架处理请求的流程图

 

Yaf也是按照这个流程来处理请求的,更细化一点,Yaf提供了完善的API,

并支持Bootstrap和插件机制,我们引用Yaf官方文档的流程图来分析,如图2-2所示。



                                 

                                  图2-2 Yaf流程图

 

从Yaf的流程图可以看到,一切请求的处理从 run() 方法开始。

yaf_application.c

PHP_METHOD(yaf_application, run) {
	zval *running;
	yaf_dispatcher_t  *dispatcher;
	yaf_response_t	  *response;
	yaf_application_t *self = getThis();

	//确保application的状态是非running的
	running = zend_read_property(yaf_application_ce, 
								 self, 
								 ZEND_STRL(YAF_APPLICATION_PROPERTY_NAME_RUN), 
								 1 TSRMLS_CC);

	if (IS_BOOL == Z_TYPE_P(running)
			&& Z_BVAL_P(running)) {
		yaf_trigger_error(YAF_ERR_STARTUP_FAILED TSRMLS_CC, 
						  "An application instance already run");
		RETURN_TRUE;
	}

	//把application的状态置为running
	ZVAL_BOOL(running, 1);
	zend_update_property(yaf_application_ce, 
						 self, 
						 ZEND_STRL(YAF_APPLICATION_PROPERTY_NAME_RUN),
						 running TSRMLS_CC);

	//调用dispatcher来处理这次请求
	dispatcher = zend_read_property(yaf_application_ce, 
									self, 
									ZEND_STRL(YAF_APPLICATION_PROPERTY_NAME_DISPATCHER), 
									1 TSRMLS_CC);
	if ((response = yaf_dispatcher_dispatch(dispatcher TSRMLS_CC))) {
		RETURN_ZVAL(response, 1, 1);
	}

	RETURN_FALSE;
}

 

可以看出,run() 函数调用了 yaf_dispatcher_dispatch 来处理请求。继续探索,等等,先剧透下,

我们来看看接下来的函数调用关系图,如图2-3所示。


 
                                                     图2-3 处理请求的函数调用关系图

 

继续分析yaf_dispatcher_dispatch,它是处理请求的中心函数,它调用路由器完成路由,

然后根据路由结果,找到对应的action执行,最后把结果发送给客户端。

yaf_dispatcher.c

yaf_response_t * yaf_dispatcher_dispatch(yaf_dispatcher_t *dispatcher TSRMLS_DC) {
	zval *return_response, *plugins, *view;
	yaf_response_t *response;
	yaf_request_t *request;
	uint nesting = YAF_G(forward_limit);

	//实例化一个Response,用来返回的数据
	response = yaf_response_instance(NULL, sapi_module.name TSRMLS_CC);
	
	//取得request对象
	request	 = zend_read_property(yaf_dispatcher_ce, 
								  dispatcher, 
								  ZEND_STRL(YAF_DISPATCHER_PROPERTY_NAME_REQUEST), 
								  1 TSRMLS_CC);

	//取得所有插件
	plugins	 = zend_read_property(yaf_dispatcher_ce, dispatcher, ZEND_STRL(YAF_DISPATCHER_PROPERTY_NAME_PLUGINS), 1 TSRMLS_CC);

	if (IS_OBJECT != Z_TYPE_P(request)) {
		yaf_trigger_error(YAF_ERR_TYPE_ERROR TSRMLS_CC, "Expect a %s instance", yaf_request_ce->name);
		zval_ptr_dtor(&response);
		return NULL;
	}

	//路由此次请求
	if (!yaf_request_is_routed(request TSRMLS_CC)) {
		//在路由开始前,调用各个插件的route_startup方法
		YAF_PLUGIN_HANDLE(plugins, YAF_PLUGIN_HOOK_ROUTESTARTUP, request, response);
		YAF_EXCEPTION_HANDLE(dispatcher, request, response);
		
		//开始路由,路由的结果是找到处理此次请求的module、controller和action
		//这几个结果分别保存在Request对象的$module、$controller和$action变量中
		if (!yaf_dispatcher_route(dispatcher, request TSRMLS_CC)) {
			yaf_trigger_error(YAF_ERR_ROUTE_FAILED TSRMLS_CC, "Routing request failed");
			YAF_EXCEPTION_HANDLE_NORET(dispatcher, request, response);
			zval_ptr_dtor(&response);
			return NULL;
		}
		
		yaf_dispatcher_fix_default(dispatcher, request TSRMLS_CC);
		
		//在路由完成后,调用各个插件的route_shutdown方法
		YAF_PLUGIN_HANDLE(plugins, YAF_PLUGIN_HOOK_ROUTESHUTDOWN, request, response);
		YAF_EXCEPTION_HANDLE(dispatcher, request, response);
		(void)yaf_request_set_routed(request, 1 TSRMLS_CC);
	} else {
		yaf_dispatcher_fix_default(dispatcher, request TSRMLS_CC);
	}

	//在路由结束后,分发循环之前调用各个插件的loop_startup函数
	YAF_PLUGIN_HANDLE(plugins, YAF_PLUGIN_HOOK_LOOPSTARTUP, request, response);
	YAF_EXCEPTION_HANDLE(dispatcher, request, response);

	//初始化view
	view = yaf_dispatcher_init_view(dispatcher, NULL, NULL TSRMLS_CC);
	if (!view) {
		return NULL;
	}

	//进入分发循环,最多循环 application.forward_limit 次
	do {
		//在请求分发之前执行各个插件的pre_dispatch方法
		YAF_PLUGIN_HANDLE(plugins, YAF_PLUGIN_HOOK_PREDISPATCH, request, response);
		
		//调用yaf_dispatcher_handle处理请求
		if (!yaf_dispatcher_handle(dispatcher, request, response, view TSRMLS_CC)) {
			YAF_EXCEPTION_HANDLE(dispatcher, request, response);
			zval_ptr_dtor(&response);
			return NULL;
		}
		yaf_dispatcher_fix_default(dispatcher, request TSRMLS_CC);
		
		//在请求分发结束后,执行各个插件的post_dispatch方法
		YAF_PLUGIN_HANDLE(plugins, YAF_PLUGIN_HOOK_POSTDISPATCH, request, response);
		YAF_EXCEPTION_HANDLE(dispatcher, request, response);
	} while (--nesting > 0 && !yaf_request_is_dispatched(request TSRMLS_CC));

	//在分发循环结束后,执行各个插件的loop_shutdown方法
	YAF_PLUGIN_HANDLE(plugins, YAF_PLUGIN_HOOK_LOOPSHUTDOWN, request, response);
	YAF_EXCEPTION_HANDLE(dispatcher, request, response);

	if (0 == nesting && !yaf_request_is_dispatched(request TSRMLS_CC)) {
		yaf_trigger_error(YAF_ERR_DISPATCH_FAILED TSRMLS_CC, "The max dispatch nesting %ld was reached", YAF_G(forward_limit));
		YAF_EXCEPTION_HANDLE_NORET(dispatcher, request, response);
		zval_ptr_dtor(&response);
		return NULL;
	}

	//取出返回给客户端的数据
	return_response = zend_read_property(yaf_dispatcher_ce, dispatcher, ZEND_STRL(YAF_DISPATCHER_PROPERTY_NAME_RETURN), 1 TSRMLS_CC);

	if (!Z_BVAL_P(return_response)) {
		//将数据发送给客户端
		(void)yaf_response_send(response TSRMLS_CC);
		yaf_response_clear_body(response, NULL, 0 TSRMLS_CC);
	}

	return response;
}

 

 路由的工作交给 yaf_dispatcher_route,这里不分析路由的详细过程,只需了解,路由的目的是找出

处理此次请求的module、controller和action,并把这些信息保存在Request对象中。接下来,由

yaf_dispatcher_handle函数去调用action的方法来处理请求。我们继续分析 yaf_dispatcher_handle。

yaf_dispatcher.c

int yaf_dispatcher_handle(yaf_dispatcher_t *dispatcher, yaf_request_t *request,  yaf_response_t *response, yaf_view_t *view TSRMLS_DC) {
	zend_class_entry *request_ce;
	char *app_dir = YAF_G(directory);

	request_ce = Z_OBJCE_P(request);

	yaf_request_set_dispatched(request, 1 TSRMLS_CC);
	if (!app_dir) {
		yaf_trigger_error(YAF_ERR_STARTUP_FAILED TSRMLS_CC, "%s requires %s(which set the application.directory) to be initialized first",
				yaf_dispatcher_ce->name, yaf_application_ce->name);
		return 0;
	} else {
		int	is_def_module = 0;
		/* int is_def_ctr = 0; */
		zval *module, *controller, *dmodule, *dcontroller, *instantly_flush;
		zend_class_entry *ce;
		yaf_controller_t *executor;
		zend_function    *fptr;

		//从Request对象中读取路由结果(module、controller)
		module		= zend_read_property(request_ce, request, ZEND_STRL(YAF_REQUEST_PROPERTY_NAME_MODULE), 1 TSRMLS_CC);
		controller	= zend_read_property(request_ce, request, ZEND_STRL(YAF_REQUEST_PROPERTY_NAME_CONTROLLER), 1 TSRMLS_CC);

		dmodule		= zend_read_property(yaf_dispatcher_ce, dispatcher, ZEND_STRL(YAF_DISPATCHER_PROPERTY_NAME_MODULE), 1 TSRMLS_CC);
		dcontroller = zend_read_property(yaf_dispatcher_ce, dispatcher, ZEND_STRL(YAF_DISPATCHER_PROPERTY_NAME_CONTROLLER), 1 TSRMLS_CC);

		if (Z_TYPE_P(module) != IS_STRING
				|| !Z_STRLEN_P(module)) {
			yaf_trigger_error(YAF_ERR_DISPATCH_FAILED TSRMLS_CC, "Unexcepted a empty module name");
			return 0;
		} else if (!yaf_application_is_module_name(Z_STRVAL_P(module), Z_STRLEN_P(module) TSRMLS_CC)) {
			yaf_trigger_error(YAF_ERR_NOTFOUND_MODULE TSRMLS_CC, "There is no module %s", Z_STRVAL_P(module));
			return 0;
		}

		if (Z_TYPE_P(controller) != IS_STRING
				|| !Z_STRLEN_P(controller)) {
			yaf_trigger_error(YAF_ERR_DISPATCH_FAILED TSRMLS_CC, "Unexcepted a empty controller name");
			return 0;
		}

		if(strncasecmp(Z_STRVAL_P(dmodule), Z_STRVAL_P(module), Z_STRLEN_P(module)) == 0) {
			is_def_module = 1;
		}

		/* if (strncasecmp(Z_STRVAL_P(dcontroller), Z_STRVAL_P(controller), Z_STRLEN_P(controller)) == 0) {
			is_def_ctr = 1;
		} */

		//实例化Controller对象
		ce = yaf_dispatcher_get_controller(app_dir, Z_STRVAL_P(module), Z_STRVAL_P(controller), Z_STRLEN_P(controller), is_def_module TSRMLS_CC);
		if (!ce) {
			return 0;
		} else {
			zval  *action, *render, *ret = NULL;
			char  *action_lower, *func_name, *view_dir;
			uint  func_name_len;

			yaf_controller_t *icontroller;

			MAKE_STD_ZVAL(icontroller);
			object_init_ex(icontroller, ce);

			/* cause controller's constructor is a final method, so it must be a internal function
			   do {
			   zend_function *constructor = NULL;
			   constructor = Z_OBJ_HT_P(exec_ctr)->get_constructor(exec_ctr TSRMLS_CC);
			   if (constructor != NULL) {
			   if (zend_call_method_with_2_params(&exec_ctr, *ce
			   , &constructor, NULL, &ret, request, response) == NULL) {
			   yaf_trigger_error(YAF_ERR_CALL_FAILED, "function call for %s::__construct failed", (*ce)->name);
			   return 0;
			   }
			   }
			   } while(0);
			   */
			yaf_controller_construct(ce, icontroller, request, response, view, NULL TSRMLS_CC);
			if (EG(exception)) {
				zval_ptr_dtor(&icontroller);
				return 0;
			}
		
			/* view template directory for application, please notice that view engine's directory has high priority */
			if (is_def_module) {
				spprintf(&view_dir, 0, "%s%c%s", app_dir, DEFAULT_SLASH, "views");
			} else {
				spprintf(&view_dir, 0, "%s%c%s%c%s%c%s", app_dir, DEFAULT_SLASH, "modules", DEFAULT_SLASH, Z_STRVAL_P(module), DEFAULT_SLASH, "views");
			}

			if (YAF_G(view_directory)) {
				efree(YAF_G(view_directory));
			}
			YAF_G(view_directory) = view_dir;

			zend_update_property(ce, icontroller, ZEND_STRL(YAF_CONTROLLER_PROPERTY_NAME_NAME),	controller TSRMLS_CC);

			//从Request对象中读取路由结果(action)
			action		 = zend_read_property(request_ce, request, ZEND_STRL(YAF_REQUEST_PROPERTY_NAME_ACTION), 1 TSRMLS_CC);
			action_lower = zend_str_tolower_dup(Z_STRVAL_P(action), Z_STRLEN_P(action));

			php_printf("dispatcher action_lower = %s\n", action_lower);

			/* because the action might call the forward to override the old action */
			Z_ADDREF_P(action);

			func_name_len = spprintf(&func_name,  0, "%s%s", action_lower, "action");
			efree(action_lower);

			php_printf("dispatcher func_name = %s\n", func_name);

			if (zend_hash_find(&((ce)->function_table), func_name, func_name_len + 1, (void **)&fptr) == SUCCESS) {
				//Action是一个定义在Controller类中的成员方法,
				//就直接调用Controller定义的action方法
				uint count = 0;
				zval ***call_args = NULL;

				ret = NULL;

				executor = icontroller;
				if (fptr->common.num_args) {
					zval *method_name;

					yaf_dispatcher_get_call_parameters(request_ce, request, fptr, &call_args, &count TSRMLS_CC);
					MAKE_STD_ZVAL(method_name);
					ZVAL_STRINGL(method_name, func_name, func_name_len, 0);

					call_user_function_ex(&(ce)->function_table, &icontroller, method_name, &ret, count, call_args, 1, NULL TSRMLS_CC);

					efree(method_name);
					efree(call_args);
				} else {
					zend_call_method(&icontroller, ce, NULL, func_name, func_name_len, &ret, 0, NULL, NULL TSRMLS_CC);
				}

				efree(func_name);

				if (!ret) {
					zval_ptr_dtor(&action);
					zval_ptr_dtor(&icontroller);
					return 0;
				}

				if ((Z_TYPE_P(ret) == IS_BOOL
							&& !Z_BVAL_P(ret))) {
					/* no auto-render */
					zval_ptr_dtor(&ret);
					zval_ptr_dtor(&action);
					zval_ptr_dtor(&icontroller);
					return 1;
				}
				zval_ptr_dtor(&ret);
			} else if ((ce = yaf_dispatcher_get_action(app_dir, icontroller,
							Z_STRVAL_P(module), is_def_module, Z_STRVAL_P(action), Z_STRLEN_P(action) TSRMLS_CC))
					&& (zend_hash_find(&(ce)->function_table, YAF_ACTION_EXECUTOR_NAME,
							sizeof(YAF_ACTION_EXECUTOR_NAME), (void **)&fptr) == SUCCESS)) {
				//Action是一个独立的类,就实例化这个类,调用其execute方法
				zval ***call_args;
				yaf_action_t *iaction;
				uint count = 0;

				efree(func_name);

				MAKE_STD_ZVAL(iaction);
				object_init_ex(iaction, ce);

				yaf_controller_construct(ce, iaction, request, response, view, NULL TSRMLS_CC);
				executor = iaction;

				zend_update_property(ce, iaction, ZEND_STRL(YAF_CONTROLLER_PROPERTY_NAME_NAME), controller TSRMLS_CC);
				zend_update_property(ce, iaction, ZEND_STRL(YAF_ACTION_PROPERTY_NAME_CTRL), icontroller TSRMLS_CC);
				zval_ptr_dtor(&icontroller);

				if (fptr->common.num_args) {
					zval *method_name = NULL;

					yaf_dispatcher_get_call_parameters(request_ce, request, fptr, &call_args, &count TSRMLS_CC);
					MAKE_STD_ZVAL(method_name);
					ZVAL_STRINGL(method_name, YAF_ACTION_EXECUTOR_NAME, sizeof(YAF_ACTION_EXECUTOR_NAME) - 1, 0);

					call_user_function_ex(&(ce)->function_table, &iaction, method_name, &ret, count, call_args, 1, NULL TSRMLS_CC);

					efree(method_name);
					efree(call_args);
				} else {
					zend_call_method_with_0_params(&iaction, ce, NULL, "execute", &ret);
				}

				if (!ret) {
					zval_ptr_dtor(&action);
					zval_ptr_dtor(&iaction);
					return 0;
				}

				if (( Z_TYPE_P(ret) == IS_BOOL
							&& !Z_BVAL_P(ret))) {
					/* no auto-render */
					zval_ptr_dtor(&ret);
					zval_ptr_dtor(&action);
					zval_ptr_dtor(&iaction);
					return 1;
				}
				zval_ptr_dtor(&ret);
			} else {
				efree(func_name);
				zval_ptr_dtor(&icontroller);
				return 0;
			}

			if (executor) {
				/* controller's property can override the Dispatcher's */
				int auto_render = 1;
				render = zend_read_property(ce, executor, ZEND_STRL(YAF_CONTROLLER_PROPERTY_NAME_RENDER), 1 TSRMLS_CC);
				instantly_flush	= zend_read_property(yaf_dispatcher_ce, dispatcher, ZEND_STRL(YAF_DISPATCHER_PROPERTY_NAME_FLUSH), 1 TSRMLS_CC);
				if (render == EG(uninitialized_zval_ptr)) {
					render = zend_read_property(yaf_dispatcher_ce, dispatcher, ZEND_STRL(YAF_DISPATCHER_PROPERTY_NAME_RENDER), 1 TSRMLS_CC);
					auto_render = Z_BVAL_P(render);
				} else if (Z_TYPE_P(render) <= IS_BOOL && !Z_BVAL_P(render)) {
					auto_render = 0;
				}

				if (auto_render) {
					ret = NULL;
					if (!Z_BVAL_P(instantly_flush)) {
						zend_call_method_with_1_params(&executor, ce, NULL, "render", &ret, action);
						zval_ptr_dtor(&executor);

						if (!ret) {
							zval_ptr_dtor(&action);
							return 0;
						} else if (IS_BOOL == Z_TYPE_P(ret) && !Z_BVAL_P(ret)) {
							zval_ptr_dtor(&ret);
							zval_ptr_dtor(&action);
							return 0;
						}

						if (Z_TYPE_P(ret) == IS_STRING && Z_STRLEN_P(ret)) {
							yaf_response_alter_body(response, NULL, 0, Z_STRVAL_P(ret), Z_STRLEN_P(ret), YAF_RESPONSE_APPEND  TSRMLS_CC);
						} 

						zval_ptr_dtor(&ret);
					} else {
						zend_call_method_with_1_params(&executor, ce, NULL, "display", &ret, action);
						zval_ptr_dtor(&executor);

						if (!ret) {
							zval_ptr_dtor(&action);
							return 0;
						}

						if ((Z_TYPE_P(ret) == IS_BOOL && !Z_BVAL_P(ret))) {
							zval_ptr_dtor(&ret);
							zval_ptr_dtor(&action);
							return 0;
						}
						zval_ptr_dtor(&ret);
					}
				} else {
					zval_ptr_dtor(&executor);
				}
			}
			zval_ptr_dtor(&action);
		}
		return 1;
	}
	return 0;
}

 

处理请求之前,yaf_dispatcher_handle函数首先从Request对象读取路由结果

(module、controller和action)

然后实创建controller对象,执行action定义的逻辑处理请求。需要注意的是,

Yaf编写Action有两种方式:

(1)把action方法写在Controller类中

 

class Controller_Index extends Yaf_Controller_Abstract {
    public function indexAction () {
        echo "Hello, World!";
    }
}

 

(2)在Controller对象的$actions保存所有action序对,序对的键必须与路由结果的action名字一样,

         而且是小写,序对的值是action类文件的相对路径,然后在action类的execute方法中写处理逻辑。

 

    controllers/Index.php

class Controller_Index extends Yaf_Controller_Abstract {
    public $actions = array (
        'index' => 'actions/Index.php',
    );
}

   actions/Index.php

class Action_Index extends Yaf_Action_Abstract {
    public function execute () {
        echo "Hello, World!";
    }
}

 

从源码中可以看出,Yaf优先从Controller中寻找action方法,如果找到,就执行之,

没有找到再从Controller中定义的$actions数组中寻找Action类,实例化这个Action类,

并执行其execute方法。

 

至此,Yaf的请求处理分析差不多了,下一篇文章将会分析Yaf的路由机制。

 

  • 大小: 26.3 KB
  • 大小: 31.3 KB
  • 大小: 31 KB
  • 大小: 30.5 KB
  • 大小: 17.9 KB
  • 大小: 52.2 KB
  • 大小: 43.2 KB
分享到:
评论
2 楼 kenby 2014-11-15  
zmfyea 写道
写的太赞了!!!
怎么没有下篇了?


好久没时间更新了,其实明白了请求的处理,剩下的可以自己去读源码啦
1 楼 zmfyea 2014-11-14  
写的太赞了!!!
怎么没有下篇了?

相关推荐

    PHP框架YAF的sample

    4. **阅读源码**:查看YAF框架的源码,了解其内部实现机制,这对于深入学习非常有帮助。 5. **结合其他服务**:结合Memcached或Redis等缓存服务,学习如何在YAF中使用它们提升性能。 总之,通过这个YAF Sample,你...

    Yaf demo(基于上一次优化)

    通常,Yaf被用于构建高性能的Web应用,尤其适用于需要处理大量并发请求的场合。博文链接指向了作者"Alfred-long"在ITEYE博客上的一篇文章,这可能是对Yaf优化过程或优化结果的详细记录,包括可能的性能提升、代码...

    yaf项目开发

    1. 安装Yaf扩展:通过PECL或者源码编译安装Yaf扩展。 2. 配置Yaf:编写`conf.yaf.ini`配置文件,定义应用路径、错误处理、路由规则等。 3. 创建项目目录结构:遵循Yaf的标准目录结构,如`app`、`public`、`conf`等。...

    yaf的login示例全套代码--带php nginx环境-最新

    【描述】"yaf框架的login示例全套代码--带php nginx环境,需要自己安装下mysql和redis"表明这个项目不仅包含了登录功能的源码,还依赖于MySQL数据库用于存储用户信息,以及Redis作为缓存服务。开发者在使用此示例...

    基于PHP YAF的CMS系统.zip

    5. Web开发流程:通过项目源码,了解Web应用从请求到响应的处理过程,以及错误处理和日志记录等常见实践。 6. PHP安全:学习如何防止SQL注入、XSS攻击等常见的Web安全问题。 7. 版本控制:如果项目使用了版本控制...

    毕设&课设&项目&实训-基于Yaf框架的听课数据收集反馈系统.zip

    10. **异常处理(Exception Handling)**:Yaf支持自定义异常处理,当程序出错时,可以优雅地捕获并处理异常,避免程序崩溃。 在“听课数据收集反馈系统”项目中,可能涉及的功能包括用户登录注册、课程信息管理、...

    MySql网络验证登陆注册源码 带MD5写法

    MySQL是世界上最流行的开源关系型数据库管理系统之一,广泛应用于各种规模的企业和项目中。在这个源码中,"MySql网络验证登陆注册源码 带MD5写法" 提供了一种实现用户登录注册功能的方式,它涉及到的主要知识点包括...

    yaf-cms:yaf创建的内容管理系统

    例如,可以通过调整Yaf的配置参数来优化内存使用和请求处理速度,或者利用Yaf的缓存机制来提升数据读取效率。同时,系统的模块化设计使得添加新功能或集成其他服务变得相对容易,例如接入第三方API,实现社交登录或...

    管理系统系列-- 基于Yaf的Cmf管理系统.zip

    - **实现**:通过Yaf的路由机制,可以根据URL动态解析请求,调用对应的控制器和视图来处理内容管理任务。 - **扩展性**:Yaf的插件系统和Cmf的模块化设计,允许开发者轻松添加新的功能和组件。 【压缩包子文件的...

    yaf框架 zend_DB 简单后台实例

    3. **路由灵活**:Yaf的路由系统强大且灵活,支持多种路由规则,可以根据URL动态分发请求。 4. **自动加载**:Yaf支持自动加载类库,减少手动引入文件的麻烦。 接下来,我们看看如何结合Zend_Db进行数据库操作。...

    lnmp: php7.1+nginx1.11+mysql5.6+yaf+redis

    Nginx1.11是其稳定版本之一,提供负载均衡、静态文件处理和HTTP缓存等功能。在 LNMP架构中,Nginx作为前端服务器,负责接收来自用户的请求,根据配置转发到PHP处理器,并将处理结果返回给用户。 **MySQL5.6**: ...

    Thinkphp5内核宠物领养平台H5源码

    - **Controller**:接收用户请求,调用Model处理数据,并将结果传递给View显示。 4. **宠物信息发布系统** 平台包含宠物信息发布功能,用户可以上传照片、填写宠物信息,包括品种、年龄、性别、健康状况等。后台...

    php学习资源

    - [Yaf源码](https://github.com/laruence/yaf) #### 结语 通过本文的介绍,我们深入了解了Ap框架与Yaf框架的基本概念、工作原理以及核心功能。这两种框架不仅在PHP开发社区中占据重要地位,也为开发者提供了强大...

    yaf-extensions:php 框架 YAF 的一些扩展(https

    "yaf-extensions-master" 这个文件名可能表示这是项目的主分支或源代码仓库,其中包含了所有扩展的源码。通常,这样的命名意味着这是一份 Git 仓库的克隆,里面包含了项目的所有文件和目录,包括类库、配置文件、...

    基于PHP的淘宝客系统源码.zip

    在本场景中,PHP被用作开发淘宝客系统的主要工具,它能处理HTTP请求,与数据库进行交互,生成动态网页内容,还能与其他Web服务API(如淘宝API)进行通信。 【压缩包子文件的文件名称列表】:由于只给出了一个文件名...

    yafrk-lib:Yafrk通用类库

    1. **路由管理**:受到Yaf项目的影响,yafrk-lib可能包含了一个强大且灵活的路由系统,允许开发者定义自定义的路由规则,将HTTP请求映射到相应的处理函数或控制器,提高了应用程序的可扩展性。 2. **依赖注入**:...

Global site tag (gtag.js) - Google Analytics