Web 框架 DuckPHP 发布

2019-12-11 22:01:36 +08:00
 dvaknheo

代码: https://github.com/dvaknheo/duckphp 作者 QQ: 85811616 官方 QQ 群: 714610448

安装

composer require dvaknheo/duckphp # 用 require 
./vendor/bin/duckphp --help     # 查看有什么指令
./vendor/bin/duckphp --create   # --full # 创建工程

为什么叫 DuckPHP ? 取名鸭子类型的典故。 动态语言中经常提到鸭子类型,所谓鸭子类型就是:如果走起路来像鸭子,叫起来也像鸭子,那么它就是鸭子( If it walks like a duck and quacks like a duck, it must be a duck )。鸭子类型是编程语言中动态类型语言中的一种设计风格,一个对象的特征不是由父类决定,而是通过对象的方法决定的。

Web 框架这么多,为什么要用你的框架。 首先,DuckPHP 所有组件都可以替换,这是现代 PHP 框架必备的品质, 出现什么安全问题,替换官方组件就可以。不必等着 composer update。

其次,DuckPHP 本着普通程序员和核心程序员两个层级的使用观点,普通程序员根本不会触及到 DuckPHP 的代码,只要简单的教学就能使用。 核心程序员,可以由浅及深扩展甚至魔改框架。

这也涉及到另一个特点,DuckPHP 更自由开放, 不会让你的 工程代码限制于 App 或 app 命名空间。 你写的 DuckPHP 工程 可以作为 插件用于其他 DuckPHP 工程( 旧框架没这个功能。

DuckPHP 支持 composer,但无引用第三方组件。无引用第三方组件对系统的可靠性很重要。比如某框架用了第三方旧组件,或许会引发安全问题不可控。

DuckPHP 的适应性也是很强的,不像某些框架,只能用于全站。DuckPHP 可以在网站子目录里使用,如果不支持 path_info,也可以调整。 一般的框架的配合是很难处理的。 但是 DuckPHP 很容易插入其他框架里提前使用,你也很容易在 DuckPhp 不处理的时候接其他框架。

DuckPHP 的代码层次分明, 用主类把其他所有独立类连在一起。你很容易可以写 DuckPHP 的扩展。你可以很容易读懂。template 目录,附带了好些 demo,但文档还没跟上。 工程方面,做了 phpunit 全覆盖测试。 用了 psr-2 的标准格式化代码。希望有一小撮人能搞起来, 杜绝我现在这种闭门造车的情况。

末了,贴一份完全 demo 源文件在 template/public/demo.php

<?php declare(strict_types=1);
/**
 * DuckPHP
 * From this time, you never be alone~
 */
namespace {
    require_once(__DIR__.'/../../autoload.php');        // @DUCKPHP_HEADFILE
    //头文件可以自行修改。
}
// 以下部分是核心程序员写。
namespace MySpace\Base
{
    use \DuckPhp\Core\View;
    use \DuckPhp\Ext\CallableView;

    // 默认的 View 不支持函数调用,我们这里替换他。
    class App extends \DuckPhp\App
    {
        protected function onInit()
        {
            // 本例特殊,这里演示函数调用的   CallableView 代替系统的 View
            $this->options['callable_view_class'] = 'MySpace\View\Views';
            View::G(CallableView::G());
            
            ////
            return parent::onInit();
        }
    }
    //服务基类, 为了 XXService::G() 可变单例。
    class BaseService
    {
        use \DuckPhp\SingletonEx;
    }
    // 模型基类, 为了 XXModel::G() 可变单例。
    class BaseModel
    {
        use \DuckPhp\SingletonEx;
    }
} // end namespace
// 助手类
namespace MySpace\Base\Helper
{
    class ControllerHelper extends \DuckPhp\Helper\ControllerHelper
    {
        // 一般不需要添加东西,继承就够了
    }
    class ServiceHelper extends \DuckPhp\Helper\ServiceHelper
    {
        // 一般不需要添加东西,继承就够了
    }
    class ModelHelper extends \DuckPhp\Helper\ModelHelper
    {
        // 一般不需要添加东西,继承就够了
    }
    class ViewHelper extends \DuckPhp\Helper\ViewHelper
    {
        // 一般不需要添加东西,继承就够了
    }
} // end namespace
// 以下部分是普通程序员写的。不再和 DuckPhp 的类有任何关系。
namespace MySpace\Controller {

    use MySpace\Base\Helper\ControllerHelper as C;
    use MySpace\Service\MyService;

    class Main
    {
        public function __construct()
        {
            //设置页眉页脚。
            C::setViewWrapper('header', 'footer');
        }
        public function index()
        {
            //获取数据
            $output = "Hello, now time is " . C::H(MyService::G()->getTimeDesc());
            $url_about = C::URL('about/me');
            C::Show(get_defined_vars(), 'main_view'); //显示数据
        }
    }
    class about
    {
        public function me()
        {
            $url_main = C::URL('');
            C::setViewWrapper('header', 'footer');
            C::Show(get_defined_vars());
        }
    }
} // end namespace
namespace MySpace\Service
{
    use MySpace\Base\Helper\ServiceHelper as S;
    use MySpace\Base\BaseService;
    use MySpace\Model\MyModel;

    class MyService extends BaseService
    {
        public function getTimeDesc()
        {
            return "<" . MyModel::G()->getTimeDesc() . ">";
        }
    }

} // end namespace
namespace MySpace\Model
{
    use MySpace\Base\Helper\ModelHelper as M;
    use MySpace\Base\BaseModel;

    class MyModel extends BaseModel
    {
        public function getTimeDesc()
        {
            return date(DATE_ATOM);
        }
    }

}
// 把 PHP 代码去掉看,这是可预览的 HTML 结构
namespace MySpace\View {
    class Views
    {
        public function header($data)
        {
            extract($data); ?>
            <html>
                <head>
                </head>
                <body>
                <header style="border:1px gray solid;">I am Header</header>
    <?php
        }

        public function main_view($data)
        {
            extract($data); ?>
            <h1><?=$output?></h1>
            <a href="<?=$url_about?>">go to "about/me"</a>
    <?php
        }
        public function about_me($data)
        {
            extract($data); ?>
            <h1> OK, go back.</h1>
            <a href="<?=$url_main?>">back</a>
    <?php
        }
        public function footer($data)
        {
            ?>
            <footer style="border:1px gray solid;">I am footer</footer>
        </body>
    </html>
    <?php
        }
    }
} // end namespace
// 以下部分是核心程序员写。
// 这里是入口,单一文件下要等前面类声明
namespace {
    $options = [];
    $options['namespace'] = rtrim('MySpace\\', '\\'); //项目命名空间为 MySpace,  你可以随意命名
    $options['is_debug'] = true;  // 开启调试模式
    
    $options['skip_app_autoload'] = true; // 本例特殊,跳过 app 用的 autoload 免受干扰
    $options['skip_setting_file'] = true; // 本例特殊,跳过设置文件
    
    \DuckPhp\App::RunQuickly($options, function () {
    });
} // end namespace

DuckPHP 的前身是 DNMVCS,因为太拗口,所以改名了。这个名字应该可以吧。

5850 次点击
所在节点    PHP
29 条回复
dvaknheo
2019-12-25 09:18:28 +08:00
@ywisax 无第三方依赖,是解决有这样的场景:A 框架 1.1 用到 symfony/routing 3.4,而 symfony/routing 主版本已经升级到 4.3,有些地方和 3.4 不兼容。 这时候回头发现 3.4 有漏洞。 升级到 3.4.1。
那些部署上去的 A 框架,谁没事去更新?
dvaknheo
2019-12-25 09:51:20 +08:00
@terrywater
yii2 能热更新么? 就是保持 vendor 文件夹的内容不变,更改某个系统类的实现。
我这几天碰到的实力: 装 sdebug 替代 xdebug 之后,phpunit 不认。 我一看 里面代码, 一堆 final class :( 除了自己硬改代码,别无他法。
第三方依赖,我这里补充一下,如果你觉得有更合适的,可以替换之。 比如 Html 编码函数。zend framework3 的比我自己实现的,以及 laravel 实现的,就考虑得更周全。改起来也很容易, 添加 MyProject/Base/App (extends DuckPHP\App) 的_H($str) 方法即可。

其次,yii2 的的初始项目代码里,我看到:login.php <?php $form = ActiveForm::begin([ ... 这,view 里也做计算 ? Laravel 的 view 里也有这个问题。 当然,你可以说,我定规范,view 里的数据只能传进来,不做计算。

use yii\web\Controller; class SiteController extends Controller。yii/web/Controller 多出的是流程无关的助手函数。为什么不抽成助手类更清晰呢。DuckPHP 的原则之一,你个 CURD 程序员,V,C,S,M 助手类 解决不了的,问老大去。不要怀疑系统有问题。不需要学习或折腾其他东西。

配置文件 config/web.php 能精简么,DuckPHP 的理念之一就是默认设置不需要在外面暴露。

最后,我想问各种框架,包括 DuckPHP,为什么框架会有这么多类,这么多文件呢。 这些文件在运行中能用到多少。 不要不行么。

如果现在开始项目,我也不会马上用 DuckPHP。 毕竟公司项目不是实验品。 但如果试验次数够多,我会马上推广使用之。 毕竟我看来是个通用,高性能,优秀的框架。 让 PHP 从 Java 的道路拉回来。
terrywater
2019-12-25 11:14:19 +08:00
@dvaknheo

1.疑问:ii2 能热更新么? 就是保持 vendor 文件夹的内容不变,更改某个系统类的实现?

不是很明白你的意思,你指的是重写框架底层?(也就是不修改框架文件的前提下,重写框架的功能?)如果是这个意思,我回答,是可以的

2.view 里面做计算,并不是不可以,世界不是严格的黑白分明,八卦里面,阳中有阴呢?
工具为了解决具体的问题,这样可以很快的做出来增删改查

如果你感觉 yii2 的 view 这块孬,你可以不用他的 view 封装函数嘛,按照严格的 view 不做计算就行了,yii2 封装的 form 不用就 OK 了,这个没啥可以说事的

yii2 本身很多功能都是基于组件化,yii2 开发了轮子,你感觉不好用,你可以自己开发组件替换掉 yii2 的,这是 yii2 框架的核心所在,就像 yii2 封装的 session,你感觉不好用,配置重写指向一个你自己的 session 就可以了,
譬如 redis session,mongodb session 等组件,替换掉就 OK 了,composer 下载一下别人封装的包

3.use yii\web\Controller; class SiteController extends Controller。yii/web/Controller 多出的是流程无关的助手函数。为什么不抽成助手类更清晰呢

答:看不懂你说的与流程 无关的助手函数是什么意思,show code ?

4.一个框架有多少文件,多少类,并不是一个评价的优劣,对于一个框架的性能来说,主要看初始化部分加载了哪些

yii2 的组件是懒加载模式,配置注入,容器动态生成的,使用的时候才会生成对象,并且是单例模式,
即使 yii2 做了 1 万个组件,初始化并没有加载,这个有什么问题?

5.

本人发现喜欢搞自研框架的,都是在追究一些无关次要的语法洁癖问题,并没有解决太多实质性的问题,把写框架,当成一个自己喜欢的艺术品,而不是一个工具。
terrywater
2019-12-25 11:16:35 +08:00
@ywisax 对,composer 加载的时候,都有 github 对应的版本号,对于一个包无论怎么更新,他的历史版本是不变动的,只要制定了具体的版本号,就不存在这个问题。
terrywater
2019-12-25 11:16:42 +08:00
@ywisax 对,composer 加载的时候,都有 github 对应的版本号,对于一个包无论怎么更新,他的历史版本是不变动的,只要制定了具体的版本号,就不存在这个问题的
terrywater
2019-12-25 11:21:54 +08:00
@dvaknheo 对于您自研框架,本人没有反对,只是发表一下自己的观点

我认为在 yii2 的基础上,进行一些重构,和封装,基本就满足自己的需要了( Yii2 的框架的很多东西,都可以通过配置重写),不需要自己自研框架了

像 yii2,强哥博士文凭,200 几年就开始维护 php 框架,一路走了这么多年,最后沉淀到 yii2 里面,无论水准,经历,视野,还有花费的时间都比我们多的多,又经过了这么多用户的沉淀。
dvaknheo
2019-12-25 12:03:15 +08:00
@terrywater
额,我想了一下,yii2 是可以热更新的

组件确实是框架的粒度问题。组件初始化的时候没加载,为什么要放在 框架的包里,而不是作为附加包呢。
我是从 CodeIgniter 看的,ci 系的代码,本来就落后于时代的,ci4 我更感觉不到有什么存在的意义。
(yii2 的 demo 用了 80/400+ 个 yii 系统 php 文件)
XX 善于解决 XX 发明的问题, 那么,如何避免你这也不是重复这一问题呢,这就是我在写 DuckPHP 的时候一直在考虑的问题。

yii 1 早年我就知道,最 java 化的。 导致了后面 yii2 我也没怎么去看。 我手头有个 yii1 的好产品,自己一直看不下去。

推广方面,我更感兴趣的是为什么 Laravel 而不是 Symfony 会更流行。为什么 smarty 还一直有人用到现在。 推广东西比代码真的难。我寄望于一小撮人能用起来。闭门造车产生的问题,不多做沟通就不会被发现,也没会去解决。

经过你这么介绍,我也想有兴趣折腾几个 yii2 的项目看看。
terrywater
2019-12-25 12:07:27 +08:00
@dvaknheo

1.在包的这个角度上,yii2 是有点重,没有分拆,现在 yii3 在做这个事情了

2.即使没有分拆,但是并不影响性能,组件是懒加载模式,使用的时候才会实例化单例模式对象,不使用不会生成对象,

3.数文件个数,这么没啥意义吧,linux 和 window 的优劣可以用数文件个数对比吗?
你可能说文件个数多,加载慢,不是有 php opchache 扩展吗?个数多也没有影响
dvaknheo
2019-12-30 17:46:38 +08:00
v1.2.2 发布。通过 phpstan level7 的检查,php-cs-fixer 的格式化。phpunit 的 100%覆盖测试。

应用层方面,C::Parameters() 改成 C::getParameters() ,这个只用于自定义路由。

其他都是上层的改动。
用于各种自定义的路由 RouteHookRouteMap 拆分成 route_map 和 route_map_important 两选项, 后者在默认文件路由前钩挂,用于比较重要的自定义路由。 前者用于默认文件路由后 404 处理
把当前站点改成组件化用的 AppPluginTrait 重写。如果没特殊使用,不需要改动。
Logger 类 放入核心目录。
Helper 类从 Core 核心目录移出。 核心框架不再包含。
App 类额外方法都咔嚓。 由 Ext 扩展类自行扩充。
Swoole 的支持方式变更,ext 扩展 + core/app 类方式完成。
随行 demo 更新。

要做的:
错误处理现在复杂化了,需要重新简化以更清晰。
需要添加 init 后 缓存,直接读取到 run 运行的 更快速运行方式。
AppPlugin 组件化的初始化默认应该加上 lazyload 模式,使得多加组件不影响启动效率。
需要添加适配各家框架 nginx 配置的插件 。各家 nginx 配置居然都有不同 :( 。

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/628156

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX