完善了Request类、路由作用单一化

This commit is contained in:
TOP糯米 2019-07-08 16:07:59 +08:00
parent 8f540fdb27
commit ba9c63baa3
25 changed files with 546 additions and 294 deletions

View File

@ -0,0 +1 @@
### TOP-Framework

View File

@ -22,7 +22,7 @@ return [
'view' => [
'engine' => 'Top',
'ext' => 'html',
'dir' => '../application/home/view/',
'dir' => APP_PATH . 'home/view/',
'cacheDir' => './runtime/cache/application/home/',
'compileDir' => './runtime/compile/application/home/',
'left' => '{',

View File

@ -2,23 +2,25 @@
namespace app\home\controller;
use app\home\model\Users;
use top\blocks\Json;
class Index extends Common
{
use Json;
public function index()
{
$model = model(Users::class);
$lists = $model->all;
return [
'lists' => $lists
];
return request()->module();
}
public function hello()
{
return 'hello';
}
public function testPage()
{
// return $this->fetch();
return true;
return [];
}
}

View File

@ -3,12 +3,12 @@
<head>
<meta charset="UTF-8">
<title>Title</title>
<script src="https://libs.baidu.com/jquery/1.11.1/jquery.min.js"></script>
<script src="/resource/jquery.min.js"></script>
</head>
<body>
{view name="Index/hello"}
{loop $lists as $value}
{$value->name}
{$value}
{/loop}
</body>
</html>

View File

@ -0,0 +1,13 @@
<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport"
content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
</head>
<body>
</body>
</html>

View File

@ -45,7 +45,7 @@ class Framework
self::frameworkPath();
require 'library/App.php';
App::start(self::$type, self::$defaultModule);
App::run(self::$type, self::$defaultModule);
} else {
echo '请使用Framework::appPath()指定应用目录';
}

19
framework/blocks/Json.php Normal file
View File

@ -0,0 +1,19 @@
<?php
namespace top\blocks;
trait Json {
public function returnJson($msg, $code = 0, $data = [])
{
if (is_array($msg)) {
return json_encode($msg);
} else {
return json_encode([
'msg' => $msg,
'code' => $code,
'data' => $data
]);
}
}
}

View File

@ -21,8 +21,6 @@ class InitDecorator implements DecoratorIfs
*/
public function before()
{
$route = Register::get('Router');
$sessionConfig = Register::get('Config')->get('session');
if (!empty($sessionConfig) && $sessionConfig['open'] === true) {
session_start();
@ -60,7 +58,7 @@ class InitDecorator implements DecoratorIfs
require FRAMEWORK_PATH . 'library/functions/functions.php';
// 加载用户函数库
$funcFile = APP_PATH . $route->module . '/functions.php';
$funcFile = APP_PATH . request()->module() . '/functions.php';
if (file_exists($funcFile)) {
require $funcFile;
}
@ -71,6 +69,5 @@ class InitDecorator implements DecoratorIfs
*/
public function after($data)
{
// TODO Auto-generated method stub
}
}

View File

@ -1,41 +0,0 @@
<?php
namespace top\decorator;
use top\decorator\ifs\DecoratorIfs;
use top\library\Register;
/**
* 辅助控制器的装饰器
*
* @author topnuomi 2018年11月22日
*/
class ReturnDecorator implements DecoratorIfs
{
public function before()
{
// TODO Auto-generated method stub
}
/**
* 布尔或数组则显示视图
* @param array $data
* @throws \Exception
*/
public function after($data)
{
// TODO Auto-generated method stub
if (is_bool($data) && $data === true)
$data = [];
if (is_array($data)) {
if (request()->isAjax()) { // 如果是ajax请求则将数组转jsonecho出去
echo json_encode($data);
} else { // 显示视图
$route = Register::get('Router');
$view = Register::get('View');
echo $view->fetch($route->ctrl . '/' . $route->action, $data);
}
}
}
}

View File

@ -1,31 +0,0 @@
<?php
namespace top\decorator;
use top\decorator\ifs\DecoratorIfs;
/**
* 辅助控制器的装饰器
* @author topnuomi 2018年11月22日
*/
class StringDecorator implements DecoratorIfs
{
public function before()
{
// TODO Auto-generated method stub
}
/**
* 字符串则直接输出
* @param array $data
*/
public function after($data)
{
// TODO Auto-generated method stub
// 如果是字符串直接echo
if (!is_array($data) && !is_bool($data) && !is_object($data)) {
echo $data;
}
}
}

View File

@ -2,18 +2,20 @@
namespace top\library;
use top\library\route\Command;
use top\library\route\Pathinfo;
use top\library\http\Request;
use top\library\error\BaseError;
use top\library\exception\BaseException;
use top\library\http\Response;
class App
{
/**
* start
* 开始执行程序
* @param int $type
* @param string $defaultModule
*/
public static function start($type = 1, $defaultModule = 'home')
public static function run($type = 1, $defaultModule = 'home')
{
// 注册框架自动加载
require 'Loader.php';
@ -29,30 +31,26 @@ class App
}
// 使用whoops美化异常输出
$whoops = new \Whoops\Run;
$whoops->pushHandler(new \Whoops\Handler\PrettyPageHandler);
$whoops->register();
// $whoops = new \Whoops\Run;
// $whoops->pushHandler(new \Whoops\Handler\PrettyPageHandler);
// $whoops->register();
// if (PHP_VERSION > 5.6) {
// set_error_handler([new BaseError(), 'handler']);
// }
// set_exception_handler([new BaseException(), 'handler']);
$routeDriver = '';
if (php_sapi_name() == 'cli') {
// 命令行运行程序
$routeDriver = new Command();
} else {
// 其他方式
switch ($type) {
case 1:
$routeDriver = new Pathinfo();
break;
default:
// 其他
}
if (PHP_VERSION > 5.6) {
set_error_handler([new BaseError(), 'handler']);
}
set_exception_handler([new BaseException(), 'handler']);
$request = Request::instance();
$response = Response::instance();
// 处理请求并得到数据
$responseData = $response->dispatch($request->execute($type, $defaultModule));
// 输出内容
echo $responseData;
if (function_exists('fastcgi_finish_request')) {
fastcgi_finish_request();
}
// 实例化路由
(new Router($routeDriver, $defaultModule))->handler();
}
}

View File

@ -2,6 +2,8 @@
namespace top\library;
use top\library\http\Request;
/**
* 配置类
* @author topnuomi 2018年11月20日
@ -58,7 +60,7 @@ class Config
public function get($name = '')
{
// 加载文件
$module = Register::get('Router')->module;
$module = Request::instance()->module();
$file = APP_PATH . $module . '/config/config.php';
if (!isset(self::$files[$file])) {
if (file_exists($file)) {

View File

@ -2,10 +2,6 @@
namespace top\library;
use top\decorator\ifs\DecoratorIfs;
use top\decorator\InitDecorator;
use top\decorator\ReturnDecorator;
use top\decorator\StringDecorator;
use top\library\exception\RouteException;
use top\library\route\ifs\RouteIfs;
@ -15,12 +11,11 @@ use top\library\route\ifs\RouteIfs;
*/
class Router
{
// 路由实例
private $route;
// 装饰器
private $decorator = [];
/**
* 路由实现
* @var RouteIfs
*/
private $driver;
public $module = '';
@ -30,66 +25,19 @@ class Router
public $action = '';
public $param = [];
public $params = [];
/**
* 实例化时注入具体路由实现和默认位置
* Route constructor.
* @param RouteIfs $route
* Router constructor.
* @param RouteIfs $driver
* @param $default
* @throws RouteException
*/
public function __construct(RouteIfs $route, $default)
public function __construct(RouteIfs $driver, $default)
{
$this->route = $route;
$this->route->default = $default;
$this->route->processing();
$this->module = $this->route->module;
$this->class = $this->route->class;
$this->ctrl = $this->route->ctrl;
$this->action = $this->route->action;
$this->param = $this->route->param;
$this->check();
Register::set('Router', function () {
return $this->route;
});
Register::set('Config', function () {
return Config::instance();
});
}
/**
* 指定装饰器
* @param DecoratorIfs $decorator
*/
private function decorator(DecoratorIfs $decorator)
{
$this->decorator[] = $decorator;
}
/**
* 装饰器前置方法
*/
private function beforeRoute()
{
foreach ($this->decorator as $decorator) {
$decorator->before();
}
}
/**
* 装饰器后置方法
* @param $data
*/
private function afterRoute($data)
{
$this->decorator = array_reverse($this->decorator);
foreach ($this->decorator as $decorator) {
$decorator->after($data);
}
$this->driver = $driver;
$this->driver->default = $default;
$this->driver->processing();
}
/**
@ -113,30 +61,24 @@ class Router
}
/**
* 调用方法并执行程序
* 处理结果返回
* @return $this
* @throws RouteException
*/
public function handler()
{
$userDecorators = Register::get('Config')->get('decorator');
$systemDecorators = [InitDecorator::class, ReturnDecorator::class, StringDecorator::class];
$this->module = $this->driver->module;
$this->class = $this->driver->class;
$this->ctrl = $this->driver->ctrl;
$this->action = $this->driver->action;
$this->params = $this->driver->params;
$decorators = array_merge($systemDecorators, $userDecorators);
foreach ($decorators as $key => $value) {
$this->decorator(new $value());
}
$this->check();
$this->beforeRoute();
Register::set('Config', function () {
return Config::instance();
});
$object = new $this->class();
$reflectionClass = new \ReflectionClass($this->class);
if ($reflectionClass->hasMethod('_init')) {
$data = $object->_init();
}
if (!isset($data) || $data == null) {
$reflectionMethod = new \ReflectionMethod($this->class, $this->action);
$data = $reflectionMethod->invokeArgs($object, $this->param);
}
$this->afterRoute($data);
return $this;
}
}

View File

@ -75,8 +75,7 @@ class View
public function fetch($file = '', $param = [], $cache = false)
{
if (!$file) {
$route = Register::get('Router');
$file = $route->ctrl . '/' . $route->action;
$file = request()->controller() . '/' . request()->method();
}
return $this->template->fetch($file, $param, $cache);
}

View File

@ -244,7 +244,6 @@ class BaseException extends \Exception
EOF;
header("HTTP/1.1 404 Not Found");
echo $content;
exit;
}
public function translateMessage($message)

View File

@ -89,12 +89,69 @@ function get_table_name($classname)
}
/**
* 获取客户端IP
* @return NULL|number|string
* 创建HTTP请求
* @param $url
* @param array $data
* @param array $header
* @return bool|mixed
*/
function get_client_ip()
function create_http_request($url, $data = [], $header = [])
{
return request()->ip();
$curl = curl_init();
curl_setopt($curl, CURLOPT_URL, $url);
if (!empty($data)) {
curl_setopt($curl, CURLOPT_POST, true);
curl_setopt($curl, CURLOPT_POSTFIELDS, $data);
}
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curl, CURLOPT_HEADER, $header);
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, false);
$res = curl_exec($curl);
curl_close($curl);
if ($res) {
return $res;
}
return false;
}
/**
* 获取客户端IP
* @param int $type
* @param bool $client
* @return mixed
*/
function get_client_ip($type = 0, $client = true)
{
$type = $type ? 1 : 0;
static $ip = NULL;
if ($ip !== NULL)
return $ip[$type];
if ($client) {
if (isset($_SERVER['HTTP_X_FORWARDED_FOR'])) {
$arr = explode(',', $_SERVER['HTTP_X_FORWARDED_FOR']);
$pos = array_search('unknown', $arr);
if (false !== $pos) {
unset($arr[$pos]);
}
$ip = trim($arr[0]);
} elseif (isset($_SERVER['HTTP_CLIENT_IP'])) {
$ip = $_SERVER['HTTP_CLIENT_IP'];
} elseif (isset($_SERVER['REMOTE_ADDR'])) {
$ip = $_SERVER['REMOTE_ADDR'];
}
} elseif (isset($_SERVER['REMOTE_ADDR'])) {
$ip = $_SERVER['REMOTE_ADDR'];
}
$long = sprintf("%u", ip2long($ip));
$ip = $long ? [
$ip,
$long
] : [
'0.0.0.0',
0
];
return $ip[$type];
}
/**

View File

@ -2,6 +2,13 @@
namespace top\library\http;
use top\decorator\ifs\DecoratorIfs;
use top\decorator\InitDecorator;
use top\library\Register;
use top\library\route\Command;
use top\library\route\Pathinfo;
use top\library\Router;
/**
* 请求类
* @author topnuomi 2018年11月23日
@ -9,10 +16,57 @@ namespace top\library\http;
class Request
{
/**
* 当前实例
* @var null
*/
private static $instance = null;
/**
* 保存$_SERVER变量
* @var array
*/
private $server = [];
private static $instance;
/**
* 装饰器
* @var array
*/
private $decorator = [];
/**
* 路由实例
* @var null
*/
private $router = null;
/**
* 模块名
* @var string
*/
private $module = '';
/**
* 控制器完整类名
* @var string
*/
private $class = '';
/**
* 控制器名
* @var string
*/
private $ctrl = '';
/**
* 请求参数
* @var array
*/
private $params = [];
/**
* @return null|Request
*/
public static function instance()
{
if (!self::$instance) {
@ -30,84 +84,80 @@ class Request
{
}
public function method()
/**
* 当前请求方式
* @return mixed|string
*/
private function requestMethod()
{
return (isset($this->server['REQUEST_METHOD']) && $this->server['REQUEST_METHOD'] != '') ? $this->server['REQUEST_METHOD'] : '';
}
/**
* POST
*
* @return boolean
*/
public function isPost()
{
return $this->method() == 'POST';
return $this->requestMethod() == 'POST';
}
/**
* GET
*
* @return boolean
*/
public function isGet()
{
return $this->method() == 'GET';
return $this->requestMethod() == 'GET';
}
/**
* PUT
*
* @return boolean
*/
public function isPut()
{
return $this->method() == 'PUT';
return $this->requestMethod() == 'PUT';
}
/**
* DELETE
*
* @return boolean
*/
public function isDelete()
{
return $this->method() == 'DELETE';
return $this->requestMethod() == 'DELETE';
}
/**
* HEAD
*
* @return boolean
*/
public function isHead()
{
return $this->method() == 'HEAD';
return $this->requestMethod() == 'HEAD';
}
/**
* HEAD
*
* @return boolean
*/
public function isPatch()
{
return $this->method() == 'PATCH';
return $this->requestMethod() == 'PATCH';
}
/**
* HEAD
*
* @return boolean
*/
public function isOptions()
{
return $this->method() == 'OPTIONS';
return $this->requestMethod() == 'OPTIONS';
}
/**
* AJAX
*
* @return boolean
*/
public function isAjax()
@ -117,7 +167,6 @@ class Request
/**
* 创建一个请求post或get取决于data是否有值且不为空或空数组
*
* @param string $url
* @param array $data
* @param array $header
@ -125,22 +174,7 @@ class Request
*/
public function create($url, $data = [], $header = [])
{
$curl = curl_init();
curl_setopt($curl, CURLOPT_URL, $url);
if (!empty($data)) {
curl_setopt($curl, CURLOPT_POST, true);
curl_setopt($curl, CURLOPT_POSTFIELDS, $data);
}
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curl, CURLOPT_HEADER, $header);
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, false);
$res = curl_exec($curl);
curl_close($curl);
if ($res) {
return $res;
}
return false;
return create_http_request($url, $data, $header);
}
/**
@ -151,56 +185,158 @@ class Request
*/
public function ip($type = 0, $client = true)
{
$type = $type ? 1 : 0;
static $ip = NULL;
if ($ip !== NULL)
return $ip[$type];
if ($client) {
if (isset($_SERVER['HTTP_X_FORWARDED_FOR'])) {
$arr = explode(',', $_SERVER['HTTP_X_FORWARDED_FOR']);
$pos = array_search('unknown', $arr);
if (false !== $pos)
unset($arr[$pos]);
$ip = trim($arr[0]);
} elseif (isset($_SERVER['HTTP_CLIENT_IP'])) {
$ip = $_SERVER['HTTP_CLIENT_IP'];
} elseif (isset($_SERVER['REMOTE_ADDR'])) {
$ip = $_SERVER['REMOTE_ADDR'];
}
} elseif (isset($_SERVER['REMOTE_ADDR'])) {
$ip = $_SERVER['REMOTE_ADDR'];
return get_client_ip($type, $client);
}
/**
* 模块名称
* @return mixed
*/
public function module()
{
return $this->router->module;
}
/**
* 控制器完整类名
* @return mixed
*/
public function classname()
{
return $this->router->class;
}
/**
* 控制器名称
* @return mixed
*/
public function controller()
{
return $this->router->ctrl;
}
/**
* 方法名称
* @return mixed
*/
public function method()
{
return $this->router->action;
}
/**
* 参数
* @return mixed
*/
public function params()
{
return $this->router->params;
}
/**
* 指定装饰器
* @param DecoratorIfs $decorator
*/
private function decorator(DecoratorIfs $decorator)
{
$this->decorator[] = $decorator;
}
/**
* 装饰器前置方法
*/
private function beforeRoute()
{
foreach ($this->decorator as $decorator) {
$decorator->before();
}
$long = sprintf("%u", ip2long($ip));
$ip = $long ? [
$ip,
$long
] : [
'0.0.0.0',
0
];
return $ip[$type];
}
public function post($name)
/**
* 装饰器后置方法
* @param $data
*/
private function afterRoute($data)
{
$data = (isset($_POST[$name])) ? $_POST[$name] : '';
return $this->checkData($data);
$this->decorator = array_reverse($this->decorator);
foreach ($this->decorator as $decorator) {
$decorator->after($data);
}
}
public function get($name)
/**
* 指定路由驱动
* @param $type
* @return string|Command|Pathinfo
*/
private function routeDriver($type)
{
$data = (isset($_GET[$name])) ? $_GET[$name] : '';
return $this->checkData($data);
}
public function checkData($data)
{
if (is_array($data)) {
foreach ($data as $k => $v)
$data[$k] = filter($v);
$routeDriver = '';
if (php_sapi_name() == 'cli') {
// 命令行运行程序
$routeDriver = new Command();
} else {
$data = filter($data);
// 其他方式
switch ($type) {
case 1:
$routeDriver = new Pathinfo();
break;
default:
// 其他
}
}
return $routeDriver;
}
/**
* 设置路由并执行程序
* @param $type
* @param $defaultModule
* @return mixed
* @throws \top\library\exception\RouteException
*/
public function execute($type, $defaultModule)
{
// 实例化路由,并执行对应方法
$routeDriver = $this->routeDriver($type);
$this->router = (new Router($routeDriver, $defaultModule))->handler();
$data = $this->runAction();
return $data;
}
/**
* 调用对应方法
* @return mixed
* @throws \ReflectionException
*/
private function runAction()
{
$userDecorators = Register::get('Config')->get('decorator');
$systemDecorators = [InitDecorator::class];
$decorators = array_merge($systemDecorators, $userDecorators);
foreach ($decorators as $key => $value) {
$this->decorator(new $value());
}
$this->beforeRoute();
$ctrl = $this->router->class;
$action = $this->router->action;
$params = $this->router->params;
$object = new $ctrl();
$reflectionClass = new \ReflectionClass($ctrl);
if ($reflectionClass->hasMethod('_init')) {
$data = $object->_init();
}
if (!isset($data)) {
$reflectionMethod = new \ReflectionMethod($ctrl, $action);
$data = $reflectionMethod->invokeArgs($object, $params);
}
$this->afterRoute($data);
return $data;
}

View File

@ -0,0 +1,77 @@
<?php
namespace top\library\http;
use top\library\http\response\ResponseData;
/**
* 响应类
* Class Response
* @package top\library\http
*/
class Response
{
/**
* 当前类实例
* @var null
*/
private static $instance = null;
/**
* 响应内容
* @var null
*/
private $content = null;
/**
* 响应头
* @var array
*/
private $header = [];
private function __construct()
{
}
private function __clone()
{
}
/**
* 获取当前类单例
* @return null|Response
*/
public static function instance()
{
if (!self::$instance) {
self::$instance = new self();
}
return self::$instance;
}
/**
* 设置Header
* @param array $header
* @return $this
*/
public function header($header = [])
{
return $this;
}
/**
* 返回内容
* @param $data
* @return false|int|null|string
*/
public function dispatch($data)
{
// 处理响应数据,并返回
$responseData = new ResponseData($data);
$this->content = $responseData->dispatch();
return $this->content;
}
}

View File

@ -0,0 +1,71 @@
<?php
namespace top\library\http\response;
use top\blocks\Json;
use top\library\Register;
/**
* 处理响应数据
* Class FormatResponse
* @package top\library\http
*/
class ResponseData
{
use Json;
/**
* 执行程序后返回的实际数据
* @var false|int|null|string
*/
public $data = null;
public function __construct($data)
{
if (DEBUG === false) {
ob_clean();
}
$this->data = $this->checkData($data);
}
/**
* 检查并处理数据
* @param $data
* @return false|int|null|string
*/
private function checkData($data)
{
$responseData = null;
if (is_array($data)) {
if (request()->isAjax()) {
$responseData = $this->returnJson($data);
} else {
$view = Register::get('View');
$filename = request()->controller() . '/' . request()->method();
$responseData = $view->fetch($filename, $data);
unset($filename);
}
} elseif (is_bool($data)) {
if ($data) {
$responseData = 1;
} else {
$responseData = 0;
}
} else if (is_object($data)) {
$responseData = '[OBJECT]';
} else {
// 否则数据作为字符串处理
$responseData = $data;
}
return $responseData;
}
/**
* 返回处理后的数据
* @return false|int|null|string
*/
public function dispatch()
{
return $this->data;
}
}

View File

@ -36,7 +36,7 @@ class Pathinfo implements RouteIfs
public $action = '';
// 参数
public $param = [];
public $params = [];
// 类名
public $class = '';
@ -84,7 +84,7 @@ class Pathinfo implements RouteIfs
* @return array
* @throws \ReflectionException
*/
public function param()
public function params()
{
unset($this->uriArray[0], $this->uriArray[1], $this->uriArray[2]);
$this->uriArray = array_merge($this->uriArray, []);
@ -94,17 +94,17 @@ class Pathinfo implements RouteIfs
for ($i = 0; $i < count($paramName); $i++) {
$paramNameArray[$paramName[$i]->name] = '';
}
$param = [];
$params = [];
for ($i = 0; $i < count($this->uriArray); $i = $i + 2) {
if (isset($this->uriArray[$i + 1]) && $this->uriArray[$i + 1] != '') {
$_GET[$this->uriArray[$i]] = $this->uriArray[$i + 1];
if (isset($paramNameArray[$this->uriArray[$i]])) {
$param[$this->uriArray[$i]] = $this->uriArray[$i + 1];
$params[$this->uriArray[$i]] = $this->uriArray[$i + 1];
}
}
}
unset($paramName, $paramNameArray);
return $param;
return $params;
}
return [];
}
@ -173,7 +173,7 @@ class Pathinfo implements RouteIfs
$this->ctrl = $this->ctrl();
$this->class = '\\' . APP_NS . '\\' . $this->module . '\\controller\\' . $this->ctrl;
$this->action = $this->action();
$this->param = $this->param();
$this->params = $this->params();
unset($this->uriArray);
}
}

View File

@ -32,5 +32,5 @@ interface RouteIfs
/**
* 解析参数
*/
public function param();
public function params();
}

View File

@ -156,7 +156,7 @@ class Tags
$this->setTags($this->selfTags);
// 加载自定义模板标签
// 文件位置固定
$tagsFile = APP_PATH . Register::get('Router')->module . '/config/tags.php';
$tagsFile = APP_PATH . request()->module() . '/config/tags.php';
if (file_exists($tagsFile)) {
$tags = require $tagsFile;
$this->setTags($tags);

View File

@ -33,6 +33,4 @@ require '../framework/Framework.php';
// Framework::runType(1);
Framework::appPath('../application/');
// 执行程序
Framework::startApp();

View File

@ -3,9 +3,9 @@
<head>
<meta charset="UTF-8">
<title>Title</title>
<script src="https://libs.baidu.com/jquery/1.11.1/jquery.min.js"></script>
<script src="/resource/jquery.min.js"></script>
</head>
<body>
<h1>Hello</h1> <?php $i = 0; foreach ($lists as $value): $i++; echo $value->name; endforeach; ?>
<h1>Hello</h1> <?php $i = 0; foreach ($lists as $value): $i++; echo $value; endforeach; ?>
</body>
</html>

View File

@ -0,0 +1,13 @@
<?php /* topnuomi */ (!defined('APP_PATH')) && exit(0); ?><!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport"
content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
</head>
<body>
</body>
</html>