当前位置:Gxlcms > PHP教程 > ThinkPHP框架入门知识_PHP教程

ThinkPHP框架入门知识_PHP教程

时间:2021-07-01 10:21:17 帮助过:2人阅读

ThinkPHP框架入门知识


ThinkPHP 是一个免费开源的,快速、简单的面向对象的 轻量级PHP开发框架 ,创立于2006年初,遵循Apache2开源协议发布,是为了敏捷WEB应用开发和简化企业应用开发而诞生的。ThinkPHP从诞生以来一直秉承简洁 实用的设计原则,在保持出色的性能和至简的代码的同时,也注重易用性。并且拥有众多的原创功能和特性,在社区团队的积极参与下,在易用性、扩展性和性能方 面不断优化和改进,已经成长为国内最领先和最具影响力的WEB应用开发框架,众多的典型案例确保可以稳定用于商业以及门户级的开发。

ThinkPHP基于MVC的PHP框架
M – Model 模型 工作:负责数据的操作
V – View 视图(模板) 工作:负责前台页面显示
C – Controller 控制器(模块) 工作:描述功能

ThinkPHP核心文件介绍
├─ThinkPHP.php 框架入口文件
├─Common 框架公共文件
├─Conf 框架配置文件
├─Extend 框架扩展目录
├─Lang 核心语言包目录
├─Lib 核心类库目录
│ ├─Behavior 核心行为类库
│ ├─Core 核心基类库
│ ├─Driver 内置驱动
│ │ ├─Cache 内置缓存驱动
│ │ ├─Db 内置数据库驱动
│ │ ├─TagLib 内置标签驱动
│ │ └─Template 内置模板引擎驱动
│ └─Template 内置模板引擎
└─Tpl 系统模板目录

#项目目录结构及说明:
Home 前台应用文件夹
├─Common 项目公共文件目录
├─Conf 项目配置目录
├─Lang 项目语言目录
├─Lib 项目类库目录
│ ├─Action Action类库目录
│ ├─Behavior 行为类库目录
│ ├─Model 模型类库目录
│ └─Widget Widget类库目录
├─Runtime 项目运行时目录
│ ├─Cache 模板缓存目录
│ ├─Data 数据缓存目录
│ ├─Logs 日志文件目录
│ └─Temp 临时缓存目录
└─Tpl 项目模板目录

ThinkPHP 3 MVC模式和URL访问
什么是MVC
M -Model 编写model类 对数据进行操作
V -View 编写html文件,页面呈现
C -Controller 编写类文件(UserAction.class.php)
ThinkPHP的MVC特点
编写非常灵活,只有view都可以执行

ThinkPHP的MVC对应的目录
M 项目目录/应用目录/Lib/Model
V 项目目录/应用目录/Tpl
C 项目目录/应用目录/Lib/Action
url访问C
url的4种访问方式
1.PATHINFO 模式

http://域名/项目名/入口文件/模块名/方法名/键1/值1/键2/值2

2.普通模式

http://域名/项目名/入口文件?m=模块名&a=方法名&键1=值1&键2=值2

3.REWRITE模式

http://域名/项目名/模块名/方法名/键1/值1/键2/值2

4.兼容模式

http://域名/项目名/入口文件?s=模块名/方法名/键1/值1/键2/值2

ThinkPHP 3.1.2 输出和模型使用
ThinkPHP 3 的输出
a、通过 echo 等PHP原生的输出方式在页面中输出
b、通过display方法输出
想分配变量可以使用assign方法
c、修改左右定界符
休要修改配置文件中的配置项
‘TMPL_L_DELIM’=>'<{‘, //修改左定界符
‘TMPL_R_DELIM’=>’}>’, //修改右定界符

ThinkPHP 3 的模型使用
需要在方法中通过new Model(表名)的形式操作数据库
$m=new Model(‘User’);
$arr=$m->select();
‘DB_TYPE’=>’mysql’, //设置数据库类型
‘DB_HOST’=>’localhost’,//设置主机
‘DB_NAME’=>’thinkphp’,//设置数据库名
‘DB_USER’=>’root’, //设置用户名
‘DB_PWD’=>”, //设置密码
‘DB_PORT’=>’3306′, //设置端口号
‘DB_PREFIX’=>’tp_’, //设置表前缀
也可以使用DSN方法进行配置
‘DB_DSN’=>’mysql://root:@localhost:3306/thinkphp’,//使用DSN方式配置数据库信息
如果两种方式同时存在,以DSN方式为优先

还有一种简单实用模型的方式
M() 等效为 new Model();
$m=M(‘User’);
$arr=$m->select();

使用模型的实例可以对数据进行操作,操作的工作一般就是对数据库进行 增删改查 CURD
增 -C Create $m->add()
删 -D Delete $m->delete()
改 -U Update $m->save()
查 -R Read $m->select()

a、模板可以遍历数组


1
2
3

<{$vo.id}>----<{$vo.username}>-----<{$vo.sex}>


b、我们可以开启调试功能中的page_trace
1.开启调试功能
//3.开启调试模式(在主入口文件配置index.php)
define(‘APP_DEBUG’,true);
2.我们需要设置配置文件,开启页面trace
‘SHOW_PAGE_TRACE’=>true,//开启页面Trace,需要有$this->display()才显示

CURD特性
读取数据
对数据的读取 Read
$m=new Model(‘User’);
$m=M(‘User’);
select
$m->select();//获取所有数据,以数组形式返回
find
$m->find(2);//获取单条数据
getField(字段名)//获取一个具体的字段值
$arr=$m->where(‘id=2′)->getField(‘username’);
创建数据
对数据的添加 Create
$m=new Model(‘User’);
$m=M(‘User’);
$m->字段名=值
$m->add();
返回值是新增的id号
删除数据
$m=M(‘User’);
$m->delete(2); //删除id为2的数据
$m->where(‘id=2′)->delete(); //与上面效果相同,也是删除id为2的数据
返回值是受影响行数
更新数据
$m=M(‘User’);
$data['id']=1;
$data['username']=’ztz2′;
$m->save($data);
返回值是受影响行数

查询方式
普通查询方式
a、字符串
$arr=$m->where(“sex=0 and username=’gege'”)->find();
b、数组
$data['sex']=0;
$data['username']=’gege';
$arr=$m->where($data)->find();
注意:这种方式默认是and的关系,如果使用or关系,需要添加数组值
$data['sex']=0;
$data['username']=’gege';
$data['_logic']=’or';
表达式查询方式
$data['id']=array(‘lt’,6);
$arr=$m->where($data)->select();
EQ 等于
NEQ不等于
GT 大于
EGT大于等于
LT 小于
ELT小于等于
LIKE 模糊查询
$data['username']=array(‘like’,’%ge’);
$arr=$m->where($data)->select();
NOTLIKE
$data['username']=array(‘notlike’,’%ge%’); //notlike中间没有空格
$arr=$m->where($data)->select();

注意:如果一个字段要匹配多个通配符
$data['username']=array(‘like’,array(‘%ge%’,’%2%’,’%五%’),’and’);//如果没有第三个值,默认关系是or关系
$arr=$m->where($data)->select();
BETWEEN
$data['id']=array(‘between’,array(5,7));
$arr=$m->where($data)->select();
//SELECT * FROM tp_user WHERE ( (id BETWEEN 5 AND 7 ) )
$data['id']=array(‘not between’,array(5,7));//注意,not 和 between中间一定要有空格
$arr=$m->where($data)->select();
IN
$data['id']=array(‘in’,array(4,6,7));
$arr=$m->where($data)->select();
//SELECT * FROM tp_user WHERE ( id IN (4,6,7) )

$data['id']=array(‘not in’,array(4,6,7));
$arr=$m->where($data)->select();
//SELECT * FROM tp_user WHERE ( id NOT IN (4,6,7) )
区间查询
$data['id']=array(array(‘gt’,4),array(‘lt’,10));//默认关系是 and 的关系
//SELECT * FROM tp_user WHERE ( (id > 4) AND (id < 10) )

$data['id']=array(array(‘gt’,4),array(‘lt’,10),’or’) //关系就是or的关系

$data['name']=array(array(‘like’,’%2%’),array(‘like’,’%五%’),’gege’,’or’);
统计查询
count //获取个数
max //获取最大数
min //获取最小数
avg //获取平均数
sum //获取总和
SQL直接查询
a、query 主要数处理读取数据的
成功返回数据的结果集
失败返回boolean false
$m=M();
$result=$m->query(“select * from t_user where id >50″);
var_dump($result);
b、execute 用于更新个写入操作
成功返回影响行数
失败返回boolean false
$m=M();
$result=$m->execute(“insert into t_user(username) values(‘ztz3′)”);
var_dump($result);

连贯操作
常用连贯操作
1.where
帮助我们设置查询条件
2.order
对结果进行排序
$arr=$m->order(‘id desc’)->select();
$arr=$m->order(array(‘id’=>’desc’,’sex’=>’asc’))->select();
3.limit
限制结果
limit(2,5)
limit(‘2,5′)
limit(10)//limit(0,10)
4.field
设置查询字段
field(‘username as name,id’)
field(array(‘username’=>’name’,’id’)
field(‘id’,true) //获取除了id以外的所有字段
5.table
设置表名
6.group
分组
7.having

alias 用于给当前数据表定义别名 字符串
page 用于查询分页(内部会转换成limit) 字符串和数字
join* 用于对查询的join支持 字符串和数组
union* 用于对查询的union支持 字符串、数组和对象
distinct 用于查询的distinct支持 布尔值
lock 用于数据库的锁机制 布尔值
cache 用于查询缓存 支持多个参数(以后在缓存部分再详细描述)
relation 用于关联查询(需要关联模型扩展支持) 字符串
validate 用于数据自动验证 数组
auto 用于数据自动完成 数组
filter 用于数据过滤 字符串
scope* 用于命名范围 字符串、数组

视图
模板的使用
a、规则
模板文件夹下[TPL]/[分组文件夹/][模板主题文件夹/]和模块名同名的文件夹[Index]/和方法名同名的文件[index].html(.tpl)
更换模板文件的后缀名(修改配置文件)
‘TMPL_TEMPLATE_SUFFIX’=>’.html’,//更改模板文件后缀名
b、修改模板文件目录层次
‘TMPL_FILE_DEPR’=>’_’,//修改模板文件目录层次
c、模板主题
‘DEFAULT_THEME’=>’your’,//设置默认模板主题
需要在TPL下面新建一个your文件夹作为模板主题文件夹
如何动态修改模板主题?
1、在后台准备一个功能,修改config.php文件中的默认模板项
2、通过url传递 t=主题 参数可以修改不同的模板
‘DEFAULT_THEME’=>’your’,//设置默认模板主题
‘TMPL_DETECT_THEME’=>true,//自动侦测模板主题
‘THEME_LIST’=>’your,my’,//支持的模板主题列表

输出模板内容
a、display
1.display中没有参数
$this->display();
2.可以带参数
$this->display(本模块文件夹下的其他模板文件);
$this->display(‘index2′);

$this->display(其他文件夹下的模板文件);
$this->display(‘Public:error’);//注意,仅仅需要在Tpl下有Public文件夹以及其中的error.html即可,不需要一定有Public模块

$this->display(其他主题下的 文件夹下的 模板文件);//需要开启主题支持
$this->display(‘my:Index:index’);

$this->display(一个url路径);
$this->display(‘./Public/error.html’);

$this->display(‘./Public/error.html’,’utf-8′,’text/xml’);

$this->show($content);
3.fetch方法
获得模板文件中的内容,以字符串形式返回
$content=$this->fetch(‘Public:error’);
4.show方法
不需要模板文件,可以直接输出模板内容
$content=$this->fetch(‘Public:error’);
dump($content);
$content=str_replace(‘h1′,’i’,$content);
$this->show($content);
模板中的赋值
//$this->assign(‘name’,’赵桐正’);
$this->name=’赵桐正2′;
$this->display();
模板替换
__PUBLIC__:会被替换成当前网站的公共目录 通常是 /Public/
__ROOT__: 会替换成当前网站的地址(不含域名)
__APP__: 会替换成当前项目的URL地址 (不含域名)
__GROUP__:会替换成当前分组的URL地址 (不含域名)
__URL__: 会替换成当前模块的URL地址(不含域名)
__ACTION__:会替换成当前操作的URL地址 (不含域名)
__SELF__: 会替换成当前的页面URL

更换模板变量规则,修改配置项
‘TMPL_PARSE_STRING’=>array( //添加自己的模板变量规则
‘__CSS__’=>__ROOT__.’/Public/Css’,
‘__JS__’=>__ROOT__.’/Public/Js’,
),

模板中的变量
变量输出
1.标量输出
2.数组输出
{$name[1]} //索引数组
{$name['k2']} //关联数组
{$name.k1}
3.对象输出
{$name:k}
{$name->k}
系统变量
{$Think.get.id}
使用函数
{$name|strtoupper} 生成的编译后文件是
{$name|date=’Y m d H:i:s’,###}
默认值
{$name|default=’这里是默认值’}
运算符
+ – * / % ++ —
{$name++}

模板中的基本语法
导入CSS和JS文件
1、css link
js scr


2.import
//导入Public文件夹下面的Js目录中的test.js文件,import标签可以省略type属性,默认就是js的

//可以更改默认文件夹 设置basepath属性

3.load
//方法可以自动检测导入的文件类型

分支结构
1、if

男人是泥巴做的

女人是水做的


未成年

青春年少

成年

> gt
< lt
== eq
<= elt
>= egt
!= neq
=== heq
!== nheq


一个和尚挑水吃
两个和尚台水吃
三个和尚没水吃
这里是默认值

循环结构
1.for





{$j}abc

2.volist

{$v.username}


3.foreach

{$k}——-{$v}


特殊标签
1、比较标签
eq或者 equal 等于
neq 或者notequal 不等于
gt 大于
egt 大于等于
lt 小于
elt 小于等于
heq 恒等于
nheq 不恒等于

2.范围标签
in
在这些数字里面不在这些数字的范围内
在这些数字里面不在这些数字的范围内
between
{$n}在1-10之间{$n}不在1到10之间
3.present
标签来判断模板变量是否已经赋值,
m有赋值m没有赋值
4.Empty
empty标签判断模板变量是否为空,
n为空赋值n有值
5.Defined
判断常量是否已经定义
6.Define
在模板中定义常量
7.Assing
模板中变量赋值

其他标签使用
1、在模板中直接使用PHP代码
echo “我是赵桐正”
2、建议更改左右定界符
在配置文件中改变
‘TMPL_L_DELIM’=>'<{‘, //修改左定界符
‘TMPL_R_DELIM’=>’}>’, //修改右定界符

模板的使用技巧
模板包含







在模板中变量用[变量]接受

模板渲染
1、自动开启模板渲染 设置配置文件
‘LAYOUT_ON’=>true,//开启模板渲染
准备一个模板渲染页面,在页面中使用{__CONTENT__}接受具体模板页面的内容
如果在摸一个具体模板中不希望使用渲染模板,可以在页首添加{__NOCONTENT__}
2、不开启自动模板渲染可以在每一个具体页面的页首添加

3.使用技巧
在渲染模板文件中也可以使用其他模板文件的内容


这里是渲染页面!!!


{__CONTENT__}


模板的继承

控制器的模块和操作
空模块和空操作
1、空操作
function _empty($name){
$this->show(“$name 不存在 返回首页”);
}
2.空模块
class EmptyAction extends Action{
function index(){
$city=M(‘City’);
$arr=$city->select();
$this->assign(‘list’,$arr);
$name=MODULE_NAME;
$this->display(“City:$name”);
}
}

前置操作和后置操作
1、前置操作: _before_操作名
2、后置操作: _after_操作名

URL
URL规则
1、默认是区分大小写的
2、如果我们不想区分大小写可以改配置文件
‘URL_CASE_INSENSITIVE’=>true,//url不区分大小写
3、如果模块名为 UserGroupAction
那么url找模块就必要要写成

http://localhost/thinkphp4/index.php/user_group/index

4、如果’URL_CASE_INSENSITIVE’=>false
那么url也可以写为

http://localhost/thinkphp4/index.php/UserGroup/index

URL伪静态
‘URL_HTML_SUFFIX’=>’html|shtml|xml’,//限制伪静态的后缀
URL路由
1、启动路由
要在配置文件中开启路由支持
2、使用路由
1.规则表达式配置路由
‘my’=>’Index/index’,//静态地址路由
‘:id/:num’=>’Index/index’,//动态地址路由
‘year/:year/:month/:date’=>’Index/index’,//动态和静态混合地址路由
‘year/:yeard/:monthd/:dated’=>’Index/index’,//动态和静态混合地址路由
加上 d代表类型只能是数字
‘my/:id$’=>’Index/index’,// 加上$说明地址中只能是 my/1000 后面不能有其他内容了
2.正则表达式配置路由
‘/^year/(d{4})/(d{2})/(d{2})/’=>’Index/index?year=:1&month=:2&date=:3′
3、注意事项:
1.越复杂的路由越往前面放
‘URL_ROUTE_RULES’=>array(
‘my/:year/:month:/:day’=>’Index/day’,
‘my/:idd’=>’Index/index’,
‘my/:name’=>’Index/index’,
)
2.可以使用$作为完全匹配的路由规则
‘URL_ROUTE_RULES’=>array(
‘my/:idd$’=>’Index/index’,
‘my/:name$’=>’Index/index’,
‘my/:year/:month:/:day$’=>’Index/day’,
),
3.用正则匹配的方式
‘URL_ROUTE_RULES’=>array(
‘/^my/(d+)$/’=>’Index/index?id=:1′,
‘/^my/(w+)$/’=>’Index/index?name=:1′,
‘/^my/(d{4})/(d{2})/(d{2})$/’=>’Index/day?year=:1&month=:2&day=:3′,
),

URL重写
URL生成

分组、页面跳转与ajax
多应用配置技巧
使用分组
页面跳转
$this->success(‘查询成功’,U(‘User/test’));
$this->redirect(‘User/test’,”,5,’页面正在跳’);
Ajax技巧
大C方法 获取配置文件中数组名称和值
echo C(‘DB_USER’);

大F方法 文件处理
写:F(‘文件名’,’数组’,’目录’);
读:F(‘文件名’,”,’目录’);

大U方法 URL处理
在PHP里
U(‘方法名’)
在模板中
当前函数{:U(‘方法名’)}
其他函数{:U(‘函数/方法名’)}

*文件引入

*表单处理
方法1 $this->_post(”);
获取提交表单,会使用函数htmlspecialchars()过滤
$username=$this->_post(‘username’);

方法2 I(‘username’); [3.1.3新功能]
大I,自动判断post和get
$username=I(‘username’);
echo I(‘username’,’不存在值时的默认值’,’使用函数’);

查看是否有数据提交:
print_r(I(‘post.’));

禁止表单处理函数的访问,提升安全性
方法1
if(!$this->isPost()) _404(‘页面不存在’,U(‘index’));
echo ‘正常提交';

方法2 (推荐) halt 页面可以定制错误页面:
if(!IS_POST) halt(‘页面不存在’);
echo ‘正常提交';

制定方法:在Conf/config.php 添加:’TMPL_EXCEPTION_FILE’=>’./Public/Tpl/error.html’,
文件接受错误内容:./Public/Tpl/error.html,只能写原生php,支持常量,如__APP__

//返回插入的ID值,数据库从1开始
if(M(‘user’)->data($data)->add()){
$this->success(‘添加成功’,’index’);
}else{
$this->error(‘添加失败’);
}

*输出到模板

1,数据准备
方法1
$this->assign(‘变量名’,’变量值’)

方法2
$this->变量名=’变量值';

方法3(新版本,缩短了代码)
$this->assign(‘变量名’,’变量值’)->display();

方法4(一行搞定)
$this->assign(‘data’,M(‘User’)->select())->display();

2,模板输出
方法1(.语法会判断是对象还是数组,配置参数:TMP_VAR_IDENTIFY=>’array’, 这样就会认为是数组,以提高速度)

{$v.username}—{$v.password}—{$v.time|date=’y-m-d H:i:s,###’}

使用函数 {:phpinfo()}

方法2
volist

*分组应用(应用组)前后台只用一个入口文件
idnex.php (默认)
define(‘APP_NAME’,’App’);
define(‘APP_PATH’,’./App/’);
define(‘APP_DEBUG’,’TRUE’);
require ‘./ThinkPHP/ThinkPHP.php';
?>

App/Conf/config.php
return array(
//开启分组
‘APP_GROUP_LIST’=>’Index,Admin’,
‘DEFAULT_GROUP’=>’Index’,

‘TMPL_FILE_DEPR’=>’_’, //默认模板分隔符为_而非文件夹形式
);

自定义控制器
1,在Action文件夹删除原来的默认控制器
2,建立两个文件夹,分别为前台和后台如Index Admin
访问: 前台host/index.php/Index 后台host/index.php/Admin

自定义配置文件(互相不能访问,但公共配置可以互相访问)
1,在host/App/Conf/建立两个文件夹,分别为前台和后台如Index Admin
配置: 前台host/App/Conf/Index/config.php
后台host/App/Conf/Admin/config.php

自定义函数
1,在host/App/Common/建立两个文件夹,分别为前台和后台如Index Admin
配置: 前台host/App/Common/Index/function.php
后台host/App/Common/Admin/function.php

注意:分组应用不支持{:U(“form_save”)}直接使用,即当前控制器当前方法,这样写:
{:U(“/Index/Index/form_save”)}
*分组应用[完毕]

**独立分组配置
开启:
1,先配置应用分组
2,添加参数
‘APP_GROUP_MODE’=>1, //开启独立分组
‘APP_GROUP_PATH’=>’Modules’, //默认分组路径,默认:Modules

常用路径变量:__ROOT__.’/’.APP_NAME.’/’.C(‘APP_GROUP_PATH’).’/’.GROUP_NAME.’/’
新的目录结构:
ThinkPHP //系统目录
App //项目目录
Public //静态文件目录
index.php //所有项目的入口文件

App:
Modules //项目模块 ==>Admin/Action,Tpl,Model,Conf,Common; Index/Action,Tpl,Model,Conf,Common;
Common //公共Common
Conf //公共Conf
Lib //公共Lib
Tpl //公共Tpl
Runtimes //运行时

Public:
css js img

**JQuery 异步提交
$(‘#send-btn’).click(
function(){
var username=$(‘input[username=username]‘);
var password=$(‘input[password=password]‘);

if(username.val()==”){
alert(‘用户名不能为空’);
username.focus();
//让用户名获取焦点
return;
}

if(password.val()==”){
alert(‘密码不能为空’);
password.focus();
//让用户名获取焦点
return;
}

//开始异步传输(要在模板内才可以解析php,不可以在单独js文件操作)
var sendUrl='{:U(“Index/Index/form_save”,”,”)}';

$.post(sendUrl,{username:username.val(),password:password.val()},function(data){},’json’);
}
);

**php异步表单处理:
Public function form_save(){
//var_dump($this->isAjax()); //判断是否有数据提交
//var_dump(IS_AJAX); //[新版本]判断是否有数据提交
if(!$this->isAjax()) halt (‘页面不存在’);
$data=array(
‘username’=>$this->_post(‘username’),
‘password’=>$this->_post(‘password’),
‘time’=>time()
);
//p($data);

}

11课开始没记录

*引入模块
1,验证码模块,核心包里没有,需要手动添加,放在ThinkPHP/Extend/Library/ORG/Util/Image.class.php
import(‘ORG.Util.Image’);
Image::BuildImageVerify(4,5,’png’);
访问:该方法地址即可
换验证码,使用一个JS随机数函数:Math.random();
注意:如果出现伪静态后缀名,需要在{:U(‘Admin/Login/index’),”,”} 增加2个空参数
混合模式区分大小写

验证码后台验证
if(md5($_POST[code])!= session('verify')){ $this->error('验证码错误');}
if(md5($this->_post[code])!= session('verify')){ $this->error('验证码错误');}
if(I('code','','md5')!= session('verify')){ $this->error('验证码错误');}

2,分页类
import('ORG.Util.Page'); //导入类
$max=M('User')->count(); //查询总条数
$page=new Page($max,5,'','Index/Index/index/p/');
//实例化对象,参数:1总条数,2每页条数,3分页跳转的参数,4设置URL路径,在分组时有用,因为少个应用组名
$limit=$page->firstRow.','.$page->listRows; //获取第一页和总页数并设置为limit值
$m=M('User')->order('time DESC')->limit($limit)->select(); //查出第一页的数据
$this->User=$m; //把数据分配给前台模板,用于foreach遍历
$this->page=$page->show(); //把分页内容分配给前台模板

模板里分页数据调用:{$page}

**SESSION 保存到数据库
1, 配置文件添加: 'SESSION_TYPE'=>'Db',

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/878455.htmlTechArticleThinkPHP框架入门知识 ThinkPHP 是一个免费开源的,快速、简单的面向对象的 轻量级PHP开发框架 ,创立于2006年初,遵循Apache2开源协议发布,是...

人气教程排行