最新发表

单文件PHP开发框架SinglePHP1.0版发布

单文件PHP框架,羽量级网站开发首选。

采集者烂JJ

SinglePHP是一个单文件PHP框架,适用于简单系统的快速开发,提供了简单的路由方式,抛弃了坑爹的PHP模板,采用原生PHP语法来渲染页面,同时提供了widget功能,简单且实用。采集者烂JJ

协议MIT

github地址:https://github.com/leo108/SinglePHP

文档地址:http://leo108.github.io/SinglePHP/

以前在开发一些极其简单的项目时就比较纠结,不用框架吧代码结构比较乱,还要花时间去写那些数据库连接代码;用框架吧框架代码比业务代码还多,太臃肿了。本文来自http://leo108.com

所以就想自己搞一个简单的框架,满足简单的网站开发需求即可,在开发的过程中也一直坚持简单的原则,可要可不要的功能一律砍掉。

很久之前就开始酝酿这个项目,13年初的时候有了雏形,经过一年多的修修改改,也在几个小网站试验过,感觉差不多了,就在这两天把文档和注释完善了一下,今晚算是正式发布了。

本文来自http://leo108.com

IE JS下一个奇怪的特(bu)性(g)

前一阵子统计一个站点的nginx访问日志时发现了大量的404请求,这些请求的url比较奇怪,都是以/undefined/img/开头的图片文件,这些请求还有一个共同点,那就是ua全部都是ie。

ie

首先find一下图片的文件名,发现都是系统中用到的图片,是第三方登录的图片按钮,用不同浏览器去访问,发现只有确实只有较低版本(<=ie8)的ie浏览器会出现红叉叉。

本文来自http://leo108.com

搜索了下相关的代码,发现这些图片的url是用js拼接出来的,大概代码如下:采集者烂JJ

var static_file_url = window.static_file_url;
html = '<img src="' + static_file_url + '/img/icon_bdshare_weibo.png" width="28" height="28" />';

而window.static_file_url是在页面头部就赋值了的,也就是说,当拼接url的时候,static_file_url并不是window.static_file_url的值,而是未定义undefined,在ie中alert这个变量,确实是undefined。

ie

同时我发现了另一个奇怪的现象,在同一个js文件中有ajax请求,url也是拼接出来的,大概样子如下:

本文来自leo108's blog

var ajax_url = window.server_url;
$.post(ajax_url, param, function(data){}, 'json');

而在这里的ajax请求却是正常的,alert(ajax_url)出来的值也是window.server_url的值。

本文来自http://leo108.com

这两个例子唯一的区别就是变量名,前者选择的变量名和window对象下的属性一致,而后者不一致,所以怀疑是这个地方的问题,于是写了一段测试代码

IE JS下一个奇怪的特(bu)性(g)

window.test_a = 'a';
window.test_b = 'b';
var test_a = window.test_a;
var test_b2 = window.test_b;
alert(test_a); // undefined
alert(test_b2); // b

所以在ie下的这个特(bu)性(g)真是让人爱(i)不(e)释(qu)手(shi)

redmine项目配置tab扩展插件

Redmine提供的插件hook中,并没有扩展项目配置tab的相关hook。本插件可以让其他插件用十分简便的方式,在项目配置中加入一个或多个tab。

采集者烂JJ

使用方式:

采集者烂JJ

在插件的init.rb中增加一行:

redmine

Redmine::Plugin.register :redmine_polls do
[ ... ]

    add_tab :polls, :partial => 'tab/polls'
end

最终效果如图:

未经允许严禁转载

redmine项目配置tab扩展插件http://leo108.com/pid-1977.asp

 

更多用法详见readme

github地址:https://github.com/leo108/redmine-project-setting-tab

http://leo108.com/pid-1977.asp

git@OSC:http://git.oschina.net/leo108/redmine-project-setting-tab

【转】PHP5.3 PHP5.4 PHP5.5 新特性

因为用到PHP新版本,一些新特性必须要了解,且有些可以在开发时就使用,如果不使用,那么何必升级PHP版本呢,显得有些得不偿失了!
所以整理了一下 一些特性,有可能不全,待添加

本文来自leo108's blog

PHP 5.3中的新特性

一.PHP 5.3中的新特性

1. 支持命名空间 (Namespace)
2. 支持延迟静态绑定(Late Static Binding)
3. 支持goto语句
4. 支持闭包、Lambda/Anonymous函数
5. 新增两个魔术方法__callStatic()和__invoke()
6. 新增Nowdoc语法
7. 在类外也可使用const来定义常量
8. 三元运算符增加了一个快捷书写方式:
9. HTTP状态码在200-399范围内均被认为访问成功
10.支持动态调用静态方法
1.支持命名空间 (Namespace)
毫无疑问,命名空间是PHP5.3所带来的最重要的新特性。有了命名空间的概念,在开发大型站点时,就比较容易设计出灵活的结构,同时避免不同包中的类名或变量名产生冲突。

【转】PHP5.3 PHP5.4 PHP5.5 新特性

在PHP5.3之前,惯例的划分Package的办法是通过目录名来分隔代码文件,代码中的类名则用下划线_来表示目录。例如本文来自http://leo108.com

<?php
class Zend_Db_Table_Select {}
// 表示当前这个类的文件位于Zend/Db/Table/Select目录下
?>

这样的命名方式被PEAR、Zend Framework及各种PHP项目广泛采用。虽然该方法可以避免不同包或类库中的类名产生冲突,但在书写代码的时候显得较为麻烦和笨拙。
在PHP5.3中,则只需要指定不同的命名空间即可,命名空间的分隔符为反斜杆\。
select.php未经允许严禁转载

<?php
namespace Zend\Db\Table;
class Select {}
?>

这样即使其它命名空间下存在名为Select的类,程序在调用时也不会产生冲突。代码的可读性也有所增加。
调用方法
call.php

本文来自http://leo108.com

<?php
//namespace Zend\Db;
include('select.php');
$s = new Zend\Db\Table\Select();
$s->test();
?>

 2.支持延迟静态绑定(Late Static Binding)
在PHP5中,我们可以在类中通过self关键字或者__CLASS__来判断或调用当前类。但有一个问题,如果我们是在子类中调用,得到的结果将是父类。因为在继承父类的时候,静态成员就已经被绑定了。 例如:

本文来自http://leo108.com

<?php
class A {
    public static function who() {
        echo __CLASS__;
    }
    public static function test() {
        self::who();
    }
}
class B extends A {
    public static function who() {
         echo __CLASS__;
    }
}
B::test();
?>

以上代码输出的结果是:
A
这和我们的预期不同,我们原来想得到子类的相应结果。
PHP 5.3.0中增加了一个static关键字来引用当前类,即实现了延迟静态绑定:

<?php
class A {
    public static function who() {
        echo __CLASS__;
    }
    public static function test() {
        static::who(); // 这里实现了延迟的静态绑定
    }
}
class B extends A {
    public static function who() {
         echo __CLASS__;
    }
}  

B::test();
?>

以上代码输出的结果是:
B
3.支持goto语句
多数计算机程序设计语言中都支持无条件转向语句goto,当程序执行到goto语句时,即转向由goto语句中的标号指出的程序位置继续执行。尽管goto语句有可能会导致程序流程不清晰,可读性减弱,但在某些情况下具有其独特的方便之处,例如中断深度嵌套的循环和 if 语句。

本文来自leo108's blog

<?php
goto a;
echo 'Foo';
a:
echo 'Bar';
for($i=0,$j=50; $i<100; $i++) {
  while($j--) {
    if($j==17) goto end;
  }
}
echo "i = $i";
end:
echo 'j hit 17';
?>

4.支持闭包、Lambda/Anonymous函数
闭包(Closure)函数和Lambda函数的概念来自于函数编程领域。例如JavaScript 是支持闭包和 lambda 函数的最常见语言之一。

PHP

在PHP中,我们也可以通过create_function()在代码运行时创建函数。但有一个问题:创建的函数仅在运行时才被编译,而不与其它代码同时被编译成执行码,因此我们无法使用类似APC这样的执行码缓存来提高代码执行效率。本文来自http://leo108.com

在PHP5.3中,我们可以使用Lambda/匿名函数来定义一些临时使用(即用即弃型)的函数,以作为array_map()/array_walk()等函数的回调函数。采集者烂JJ

<?php
echo preg_replace_callback('~-([a-z])~', function ($match) {
    return strtoupper($match[1]);
}, 'hello-world');
// 输出 helloWorld
$greet = function($name)
{
    printf("Hello %s\r\n", $name);
};
$greet('World');
$greet('PHP');
//...在某个类中
$callback =      function ($quantity, $product) use ($tax, &#038;$total)         {
   $pricePerItem = constant(__CLASS__ . "::PRICE_" .  strtoupper($product));
   $total += ($pricePerItem * $quantity) * ($tax + 1.0);
 };
array_walk($products, $callback);
?>

 5. 新增两个魔术方法__callStatic()和__invoke()
PHP中原本有一个魔术方法__call(),当代码调用对象的某个不存在的方法时该魔术方法会被自动调用。新增的__callStatic()方法则只用于静态类方法。当尝试调用类中不存在的静态方法时,__callStatic()魔术方法将被自动调用。本文来自http://leo108.com

<?php
class MethodTest {
    public function __call($name, $arguments) {
        // 参数 $name 大小写敏感
        echo "调用对象方法 '$name' "
             . implode(' -- ', $arguments). "\n";
    }  

    /**  PHP 5.3.0 以上版本中本类方法有效  */
    public static function __callStatic($name, $arguments) {
        // 参数 $name 大小写敏感
        echo "调用静态方法 '$name' "
             . implode(' -- ', $arguments). "\n";
    }
}  

$obj = new MethodTest;
$obj->runTest('通过对象调用');  

MethodTest::runTest('静态调用');  // As of PHP 5.3.0
?>

以上代码执行后输出如下:
调用对象方法’runTest’ –- 通过对象调用调用静态方法’runTest’ –- 静态调用
以函数形式来调用对象时,__invoke()方法将被自动调用。(注:原文此处代码不正确)

【转】PHP5.3 PHP5.4 PHP5.5 新特性

<?php
class CallableClass 
{
    function __invoke($x) {
        var_dump($x);
    }
}
$obj = new CallableClass;
$obj(5); //int(5)
var_dump(is_callable($obj)); //bool(true)

6.新增Nowdoc语法
用法和Heredoc类似,但使用单引号。Heredoc则需要通过使用双引号来声明。
Nowdoc中不会做任何变量解析,非常适合于传递一段PHP代码。

【转】PHP5.3 PHP5.4 PHP5.5 新特性

<?php
// Nowdoc 单引号 PHP 5.3之后支持
$name = 'MyName';
echo <<<'EOT'
My name is "$name".
EOT;
//上面代码输出 My name is "$name". ((其中变量不被解析)
// Heredoc不加引号
echo <<<FOOBAR
Hello World!
FOOBAR;
//或者 双引号 PHP 5.3之后支持
echo <<<"FOOBAR"
Hello World!
FOOBAR;
?>

支持通过Heredoc来初始化静态变量、类成员和类常量。

【转】PHP5.3 PHP5.4 PHP5.5 新特性

<?php
// 静态变量
function foo()
{
    static $bar = <<<LABEL
Nothing in here...
LABEL;
}  

// 类成员、常量
class foo
{
    const BAR = <<<FOOBAR
Constant example
FOOBAR;  

    public $baz = <<<FOOBAR
Property example
FOOBAR;
}
?>

 7. 在类外也可使用const来定义常量
PHP中定义常量通常是用这种方式:

<?php
define("CONSTANT", "Hello world.");
?>

并且新增了一种常量定义方式:

<!--
<?php
const CONSTANT = 'Hello World';
?>
-->

8. 三元运算符增加了一个快捷书写方式

原本格式为是(expr1) ? (expr2) : (expr3)
如果expr1结果为True,则返回expr2的结果。【转】PHP5.3 PHP5.4 PHP5.5 新特性

PHP5.3新增一种书写方式,可以省略中间部分,书写为expr1 ?: expr3
如果expr1结果为True,则返回expr1的结果

【转】PHP5.3 PHP5.4 PHP5.5 新特性

9. HTTP状态码在200-399范围内均被认为访问成功
10.支持动态调用静态方法http://leo108.com/pid-1969.asp

<!--
<?php
class Test{
    public static function testgo()
    {
         echo "gogo!";
    }
}
$class = 'Test';
$action = 'testgo';
$class::$action();  //输出 "gogo!"
?>
-->

 

11. 支持嵌套处理异常(Exception)
12. 新的垃圾收集器(GC),并默认启用

本文来自http://leo108.com

二.PHP5.3中其它值得注意的改变

1. 修复了大量bug
2. PHP性能提高
3. php.ini中可使用变量
4. mysqlnd进入核心扩展 理论上说该扩展访问mysql速度会较之前的MySQL 和 MySQLi 扩展快(参见http://dev.mysql.com/downloads/connector/php-mysqlnd/)
5. ext/phar、ext/intl、ext/fileinfo、ext/sqlite3和ext/enchant等扩展默认随PHP绑定发布。其中Phar可用于打包PHP程序,类似于Java中的jar机制。
6. ereg 正则表达式函数 不再默认可用,请使用速度更快的PCRE 正则表达式函数

 本文来自http://leo108.com

PHP 5.4中的新特性

1. Buid-in web server内置了一个简单的Web服务器
把当前目录作为Root Document只需要这条命令即可:
# php -S localhost:3300
也可以指定其它路径:
# php -S localhost:3300 -t /path/to/root
还可以指定路由:
# php -S localhost:3300 router.php

【转】PHP5.3 PHP5.4 PHP5.5 新特性

2.Traits
Traits提供了一种灵活的代码重用机制,即不像interface一样只能定义方法但不能实现,又不能像class一样只能单继承。至于在实践中怎样使用,还需要深入思考。
魔术常量为__TRAIT__PHP

3. Short array syntax 数组简短语法本文来自http://leo108.com

<?php
$arr = [1,'tsing', 'tsingpost.com'];
$array = [
  "foo" => "bar",
  "bar" => "foo"
  ];

 4. Array dereferencing 数组值

<?php
function myfunc() {
    return array(1,'tsing', 'tsingpost.com');
}
我认为比数组简短语法更方便的是dereferencing,以前我们需要这样:
$arr = myfunc();
echo $arr[1];
在PHP5.4中这样就行了:
echo myfunc()[1];
其他:
$name = explode(",", "tsings,male")[0];
explode(",", "tsings,male")[3] = "phper";

5. Upload progress

Session提供了上传进度支持,通过$_SESSION["upload_progress_name"]就可以获得当前文件上传的进度信息,结合Ajax就能很容易实现上传进度条了。

http://leo108.com/pid-1969.asp

6. JsonSerializable Interface
实现了JsonSerializable接口的类的实例在json_encode序列化的之前会调用jsonSerialize方法,而不是直接序列化对象的属性。

7. Use mysqlnd by default

本文来自leo108's blog

现在mysql, mysqli, pdo_mysql默认使用mysqlnd本地库,在PHP5.4以前需要:
$./configure --with-mysqli=mysqlnd
现在:
$./configure --with-mysqli

 8.实例化类

采集者烂JJ

<?php
class test{
    function show(){
    return 'test';
    }
}
echo (new test())->show();

 9.支持 Class::{expr}() 语法

http://leo108.com/pid-1969.asp

<?php
foreach ([new Human("Gonzalo"), new Human("Peter")] as $human) {
    echo $human->{'hello'}();
}

 10.Callable typehint

<?php
function foo(callable $callback) {
}
  则:
  foo("false"); //错误,因为false不是callable类型
  foo("printf"); //正确
  foo(function(){}); //正确
class A {
  static function show() {
    }
}
  foo(array("A", "show")); //正确

 11.函数类型提示的增强

<!--
由于php是弱类型的语言,因此在php 5.0后,引入了函数类型提示的功能,其含义为对于传入函数中的参数都进行类型检查,举个例子,有如下的类:
class bar {
function foo(bar $foo) {
}
//其中函数foo中的参数规定了传入的参数必须为bar类的实例,否则系统会判断出错。同样对于数组来说,也可以进行判断,比如:
function foo(array $foo) {
}
}
  foo(array(1, 2, 3)); // 正确,因为传入的是数组
  foo(123); // 不正确,传入的不是数组
-->

12.新增加了$_SERVER["REQUEST_TIME_FLOAT"],这个是用来统计服务请求时间的,并用ms来表示

<?php
echo "脚本执行时间 ", round(microtime(true) - $_SERVER["REQUEST_TIME_FLOAT"], 2), "s";

 13. 让Json更懂中文(JSON_UNESCAPED_UNICODE)本文来自leo108's blog

<?php
echo json_encode("中文", JSON_UNESCAPED_UNICODE);
//"中文"

 14. 二进制直接量(binary number format)【转】PHP5.3 PHP5.4 PHP5.5 新特性

<?php
$bin  = 0b1101;
echo $bin;
//13

PHP 5.4.0 性能大幅提升, 修复超过100个bug.
废除了register_globals, magic_quotes以及安全模式。
另外值得一提的是多字节支持已经默认启用了,
default_charset从ISO-8859-1已经变为UTF-8.
默认发送“Content-Type: text/html; charset=utf-8”,
你再也不需要在HTML里写meta tag,也无需为UTF-8兼容而传送额外的header了。PHP

删除的特性
最后,我们集中整理了几年来标记为已弃用的多个特性。这些特性包括 allow_call_time_pass_reference、define_syslog_variables、highlight.bg、register_globals、register_long_arrays、magic_quotes、safe_mode、zend.ze1_compatibility_mode、session.bug_compat42、session.bug_compat_warn 以及 y2k_compliance。
除了这些特性之外,magic_quotes 可能是最大的危险。在早期版本中,未考虑因 magic_quotes 出错导致的后果,简单编写且未采取任何举措使自身免受 SQL 注入攻击的应用程序都通过 magic_quotes 来保护。如果在升级到 PHP 5.4 时未验证已采取正确的 SQLi 保护措施,则可能导致安全漏洞。

其他改动和特性
有一种新的“可调用的”类型提示,用于某方法采用回调作为参数的情况。
htmlspecialchars() 和 htmlentities() 现在可更好地支持亚洲字符,如果未在 php.ini 文件中显式设置 PHP default_charset,这两个函数默认使用 UTF-8 而不是 ISO-8859-1。
会话 ID 现在默认通过 /dev/urandom(或等效文件)中的熵生成,而不是与早期版本一样成为必须显式启用的一个选项。
mysqlnd 这一捆绑的 MySQL 原生驱动程序库现在默认用于与 MySQL 通信的各种扩展,除非在编译时通过 ./configure 被显式覆盖。
可能还有 100 个小的改动和特性。从 PHP 5.3 升级到 5.4 应该极为顺畅,但请阅读迁移指南加以确保。如果您从早期版本升级,执行的操作可能稍多一些。请查看以前的迁移指南再开始升级。未经允许严禁转载

PHP 5.5中的新特性

新特性及提案列表 都相当大,而且不是按重要性排序。所以,如果你不想通读一遍的话,这里有四个特点我个人是最兴奋的:
:一个简单的密码散列API
:标量类型提示
:Getter和Setter
:生成器
现在,让我们来看看PHP5.5 可能会新增的功能:

http://leo108.com/pid-1969.asp

1、放弃对Windows XP和2003 的支持

【转】PHP5.3 PHP5.4 PHP5.5 新特性

2.弃用e修饰符
e修饰符是指示preg_replace函数用来评估替换字符串作为PHP代码,而不只是仅仅做一个简单的字符串替换。不出所料,这种行为会源源不断的出现安全问题。这就是为什么在PHP5.5 中使用这个修饰符将抛出一个弃用警告。作为替代,你应该使用preg_replace_callback函数。你可以从RFC找到更多关于这个变化相应的信息。未经允许严禁转载

3.新增函数和类未经允许严禁转载

boolval()
PHP已经实现了strval、intval和floatval的函数。为了达到一致性将添加boolval函数。它完全可以作为一个布尔值计算,也可以作为一个回调函数。

hash_pbkdf2()
PBKDF2全称“Password-Based Key Derivation Function 2”,正如它的名字一样,是一种从密码派生出加密密钥的算法。这就需要加密算法,也可以用于对密码哈希。更广泛的说明和用法示例本文来自leo108's blog

array_column()
$userNames = array_column($users, ‘name’);
// is the same as
$userNames = [];
foreach ($users as $user) {
$userNames[] = $user['name'];
}【转】PHP5.3 PHP5.4 PHP5.5 新特性

intl 扩展
将有许多改进 intl的扩展。例如,将会有新的IntlCalendar,IntlGregorianCalendar,IntlTimeZone,IntlBreakIterator,IntlRuleBasedBreakIterator,IntlCodePointBreakIterator类。之前,我竟然不知道有这么多关于intl扩展,如果你想知道更多,我建议你去最新公告里找 Calendar和 BreakIterator。

【转】PHP5.3 PHP5.4 PHP5.5 新特性

4.一个简单的密码散列API

未经允许严禁转载

<?php
$password = "foo";
// creating the hash
$hash = password_hash($password, PASSWORD_BCRYPT);
// verifying a password
if (password_verify($password, $hash)) {
    // password correct!
} else {
    // password wrong!
}

5.新的语言特性和增强功能。

PHP

 常量引用

PHP

“常量引用”意味着数组可以直接操作字符串和数组字面值。举两个例子:

<?php
function randomHexString($length) {
    $str = '';
    for ($i = 0; $i < $length; ++$i) {
        $str .= "0123456789abcdef"[mt_rand(0, 15)]; // direct dereference of string
    }
}
function randomBool() {
    return [false, true][mt_rand(0, 1)]; // direct dereference of array
}

我不认为在实践中会使用此功能,但它使语言更加一致。请参阅 RFC。

6.调用empty()函数(和其他表达式)一起工作
目前,empty()语言构造只能用在变量,而不能在其他表达式。
在特定的代码像empty($this->getFriends())将会抛出一个错误。作为PHP5.5 这将成为有效的代码未经允许严禁转载

7.获取完整类别名称【转】PHP5.3 PHP5.4 PHP5.5 新特性

PHP5.3 中引入命名空间的别名类和命名空间短版本的功能。虽然这并不适用于字符串类名称

<?php
use Some\Deeply\Nested\Namespace\FooBar;
// does not work, because this will try to use the global FooBar class
$reflection = new ReflectionClass('FooBar');
echo FooBar::class;

为了解决这个问题采用新的FooBar::class语法,它返回类的完整类别名称

PHP

8.参数跳跃http://leo108.com/pid-1969.asp

如果你有一个函数接受多个可选的参数,目前没有办法只改变最后一个参数,而让其他所有参数为默认值。
RFC上的例子,如果你有一个函数如下:

function create_query($where, $order_by, $join_type='', $execute = false, $report_errors = true) { ... }

那么有没有办法设置$report_errors=false,而其他两个为默认值。为了解决这个跳跃参数的问题而提出:

create_query("deleted=0", "name", default, default, false);

我个人不是特别喜欢这个提议。在我的眼睛里,代码需要这个功能,只是设计不当。函数不应该有12个可选参数。

9.标量类型提示

【转】PHP5.3 PHP5.4 PHP5.5 新特性

标量类型提示原本计划进入5.4,但由于缺乏共识而没有做。获取更多关于为什么标量类型提示没有做进PHP的信息,请参阅: 标量类型提示比你认为的更难。
对于PHP5.5 而言,针对标量类型提示讨论又一次出现,我认为这是一个相当不错的 提议。
它需要通过输入值来指定类型。例如:123,123.0,“123”都是一个有效的int参数输入,但“hello world”就不是。这与内部函数的行为一致。本文来自http://leo108.com

<?php
function foo(int $i) { ... }
foo(1); // $i = 1
foo(1.0); // $i = 1
foo("1"); // $i = 1
foo("1abc"); // not yet clear, maybe $i = 1 with notice
foo(1.5); // not yet clear, maybe $i = 1 with notice
foo([]); // error
foo("abc"); // error

10.Getter 和 Setter

如果你从不喜欢写这些getXYZ()和setXYZ($value)方法,那么这应该是你最受欢迎的改变。提议添加一个新的语法来定义一个属性的设置/读取:

<?php
class TimePeriod {
public $seconds;
public $hours {
get { return $this->seconds / 3600; }
set { $this->seconds = $value * 3600; }
}
}
$timePeriod = new TimePeriod;
$timePeriod->hours = 10;
var_dump($timePeriod->seconds); // int(36000)
var_dump($timePeriod->hours); // int(10)

当然还有更多的功能,比如只读属性。如果你想要知道更多,请参阅 RFC。本文来自http://leo108.com

11.生成器【转】PHP5.3 PHP5.4 PHP5.5 新特性

目前,自定义迭代器很少使用,因为它们的实现,需要大量的样板代码。生成器解决这个问题,并提供了一种简单的样板代码来创建迭代器。
例如,你可以定义一个范围函数作为迭代器:

http://leo108.com/pid-1969.asp

<?php
function *xrange($start, $end, $step = 1) {
for ($i = $start; $i < $end; $i += $step) {
yield $i;
}
}
foreach (xrange(10, 20) as $i) {
// ...
}

上述xrange函数具有与内建函数相同的行为,但有一点区别:不是返回一个数组的所有值,而是返回一个迭代器动态生成的值。

http://leo108.com/pid-1969.asp

12.列表解析和生成器表达式

列表解析提供一个简单的方法对数组进行小规模操作:

$firstNames = [foreach ($users as $user) yield $user->firstName];

上述列表解析相等于下面的代码:

本文来自http://leo108.com

$firstNames = [];
foreach ($users as $user) {
$firstNames[] = $user->firstName;
}

也可以这样过滤数组:

$underageUsers = [foreach ($users as $user) if ($user->age < 18) yield $user];

生成器表达式也很类似,但是返回一个迭代器(用于动态生成值)而不是一个数组。PHP

13.finally关键字

本文来自leo108's blog

这个和java中的finally一样,经典的try … catch … finally 三段式异常处理。

未经允许严禁转载

14.foreach 支持list()

对于“数组的数组”进行迭代,之前需要使用两个foreach,现在只需要使用foreach + list了,但是这个数组的数组中的每个数组的个数需要一样。看文档的例子一看就明白了。

<?php
$array = [
[1, 2],
[3, 4],
];
foreach ($array as list($a, $b)) {
echo "A: $a; B: $b\n";
}

 

15.增加了opcache扩展
使用opcache会提高php的性能,你可以和其他扩展一样静态编译(–enable-opcache)或者动态扩展(zend_extension)加入这个优化项。

PHP

16.非变量array和string也能支持下标获取了http://leo108.com/pid-1969.asp

<?php
echo array(1, 2, 3)[0];
echo [1, 2, 3][0];
echo "foobar"[2];

 

原文地址:

本文来自http://leo108.com

http://www.tsingpost.com/articles/201312/261.html

【转】PHP5.3 PHP5.4 PHP5.5 新特性

http://www.tsingpost.com/articles/201312/264.html未经允许严禁转载

http://www.tsingpost.com/articles/201312/268.html

搭建http协议的git服务器

使用了一段时间的git,完爆svn,所以打算自己搞一个git服务器玩玩。Apache

服务器用的是centos 5.8 64位。

首先要做的是安装git,比较偷懒,直接用yum,但是yum源里没有git,可以使用第三方源。

rpm -Uvh http://repo.webtatic.com/yum/centos/5/latest.rpm
yum install --enablerepo=webtatic git-all

然后是http服务器,选的apache,本来我是想用nginx的,后来试了一下,发现nginx自带的dav不支持git的push,不能提交代码,如果要完整支持则需要引入第三方模块https://github.com/arut/nginx-dav-ext-module,嫌麻烦就用apache吧,反正对性能没要求。

本文来自leo108's blog

apache的编译命令如下:

./configure --prefix=/path/to/apache/ --enable-rewrite --enable-deflate --disable-userdir --enable-so --enable-expires --enable-headers --with-included-apr --with-apr=/usr --with-apr-util=/usr --enable-ssl --with-ssl=/usr --enable-dav --enable-dav-fs --enable-dav-lock
make && make install

创建文件夹/path/to/apache/conf/vhost,然后在/path/to/apache/conf/httpd.conf最后面加上一行

include conf/vhost/*.confgit

其他apache的基础配置,例如设置启动账户之类的本文就不说了,网上教程很多。

接下来是创建git根目录,我选的是/home/leo/git/,在这个目录下创建目录leo_git_test,并且初始化git目录采集者烂JJ

mkdir -p /home/leo/git/leo_git_test
cd /home/leo/git/leo_git_test
git init --bare
git update-server-info

接下来是设置apache的访问控制,在/path/to/apache/conf/vhost下执行

搭建http协议的git服务器

/path/to/apache/bin/htpasswd -c git.passwd leo

按照提示输入两次密码。采集者烂JJ

创建git.conf文件,内容如下:本文来自leo108's blog

<VirtualHost *:80>
    SetEnv GIT_PROJECT_ROOT /home/leo/git/   #git目录
    SetEnv GIT_HTTP_EXPORT_ALL
    ScriptAlias / /usr/libexec/git-core/git-http-backend/ #将/上的请求转发给git

    <Location "/">
        DAV on           #开启dav扩展
        Order allow,deny
        Allow from all
        AuthType Basic   #开启密码验证
        AuthName "Git"
        AuthUserFile /path/to/apache/conf/vhost/git.passwd
        Require valid-user
    </Location>
</VirtualHost>

然后启动apache即可。

在本地clone测试一下Apache

git clone http://127.0.0.1/leo_git_test

输入之前设置的http账号密码即可。

未经允许严禁转载

【转】git flow开发流程

原文链接http://ihower.tw/blog/archives/5140

大家都知道 Git 開 branch 很方便,非常鼓勵 topic branch,但有沒有一套模型流程告訴我們應該怎麼管理 branch 呢? 有人便整理出一套最佳實踐慣例 A successful Git branching model,我們團隊就採用了這套流程。簡單來說,他將 branch 分成兩個主要分支,三種支援性分支:

git branch最佳实践

git branch最佳实践

  • 主要分支
    • master: 永遠處在 production-ready 狀態
    • develop: 最新的下次發佈開發狀態
  • 支援性分支
    • Feature branches: 開發新功能都從 develop 分支出來,完成後 merge 回 develop
    • Release branches: 準備要 release 的版本,只修 bugs。從 develop 分支出來,完成後 merge 回 master 和 develop
    • Hotfix branches: 等不及 release 版本就必須馬上修 master 趕上線的情況。會從 master 分支出來,完成後 merge 回 master 和 develop

作者還提供了 git-flow 指令工具幫助我們很容易的實踐,用法如下:http://leo108.com/pid-1955.asp

首先是初始化動作:git-flow

git flow init

采集者烂JJ

初始化動作會問你一些問題,大抵是命名慣例:【转】git flow开发流程

No branches exist yet. Base branches must be created now.
Branch name for production releases: [master]
Branch name for “next release” development: [develop]
How to name your supporting branch prefixes?
Feature branches? [feature/]
Release branches? [release/]
Hotfix branches? [hotfix/]
Support branches? [support/]
Version tag prefix? []

設定完之後,預設的 branch 就變成 develop 了。有任何開發,一律都先開 branch:

【转】git flow开发流程

git flow feature start some_awesome_feature
(以此類推 git flow release 和 git flow hotfix)

【转】git flow开发流程

完成之後輸入【转】git flow开发流程

git flow feature finish some_awesome_featurehttp://leo108.com/pid-1955.asp

就會合併回 develop 並幫你刪除這個 (local) branch。

本文来自leo108's blog

關於 remote branch

push 一個 feature branch 到遠端:未经允许严禁转载

git flow feature publish some_awesome_feature
或 git push origin feature/some_awesome_feature本文来自leo108's blog

追蹤一個遠端的 branch:

git flow feature track some_awesome_feature
或 git checkout -b feature/some_awesome_feature -t origin/feature/some_awesome_feature

刪除遠端的 branch:本文来自leo108's blog

git push origin :feature/some_awesome_feature

本文来自http://leo108.com

我們還碰到一個問題是輸入 git flow feature finish 時出現以下錯誤:

【转】git flow开发流程

warning: not deleting branch ‘feature/some_awesome_feature’ that is not yet merged to
‘refs/remotes/origin/feature/some_awesome_feature’, even though it is merged to HEAD.
error: The branch ‘feature/some_awesome_feature’ is not fully merged.
If you are sure you want to delete it, run ‘git branch -D feature/some_awesome_feature’.采集者烂JJ

原因是這個 feature branch 一開始是從遠端 checkout 出來的,以及這個 feature branch 有 commit 沒有 push 回去 ,所以 git flow 不敢幫你刪除 local branch,這時候其實 merge 動作已經完成了,所以你可以手動輸入 git branch -D feature/some_awesome_feature 強制刪除 local branch 即可。(小結論:git-flow 只是個輔助工具,了解 git 還是必要的)git-flow

關於 feature branch 的合併

如果是開發時間比較久的 feature branch,很可能會因為 1. 不定時的 merge develop 與新版同步 2. 實驗性質的修改 3. 需求的變更 等等因素,而讓這個 feature branch 的 commit 記錄變成髒髒的,這時候我們會用以下的方式來做 merge 動作:git-flow

1. 先對 feature branch 做 `git rebase develop`。會很苦,但是弄完會很有成就感,整個 branch commit history 會變成很乾淨。請學 interactive mode,可以讓你拿掉一些 commit、合併或修改,你也可以 rebase 多次直到滿意為止。
2. 在從 develop bracnh 做 `git merge feature/some_awesome_feature –no-ff`,–no-ff 的意思是會強制留一個 merge commit log 記錄,這可以讓 commit tree 看清楚發生了 merge 動作。(因為我們剛做了 rebase,而 git 預設的合併模式是 fast-forward,所以如果不加 –no-ff 是不會有 merge commit 的) 這個 merge commit 的另一個額外方便之處是,如果想要 reset/revert 整個 branch 只要 reset/revert 這個 commit 就可以了。
3. 如果此 feature branch 有 remote branch,要先砍掉 `git push origin :feature/some_awesome_feature` 再 `git push origin develop` (這是因為 rebase 一個已經 push 出去的 repository,然後又把修改的 history push 出去,會造成超級大災難啊~)

git

先 rebase 再 merge –no-ff 這樣做的好處到底是什麼? 看圖體會一下吧:

【转】git flow开发流程

本文来自leo108's blog

每一次的 merge 就代表了一個 feature 完成,也可以很清楚看到這個 feature branch 底下包含哪些 commit。git

對了,如果有用 Github 的話,請記得務必用一用它的 pull request 功能,我們會在 branch 完成後發一個 pull request,好讓大家可以對一整個 branch 做 code review 留言。【转】git flow开发流程

mac删文件时提示Operation not permitted解决方案

删除文件时会提示Operation not permitted,一开始以为是权限问题,但是sudo之后删除也不行,搜索之后找到解决方案。http://leo108.com/pid-1953.asp

这应该是FreeBSD系统一个特性,文件有一些属性,称为flag,通过chflags命令可以修改这些属性。

arch, archived
set the archived flag (super-user only)

opaque set the opaque flag (owner or super-user only). [Directory is opaque when viewed through a union mount]采集者烂JJ

nodump set the nodump flag (owner or super-user only)本文来自leo108's blog

sappnd, sappend
set the system append-only flag (super-user only)

采集者烂JJ

schg, schange, simmutable
set the system immutable flag (super-user only)http://leo108.com/pid-1953.asp

uappnd, uappend
set the user append-only flag (owner or super-user only)

采集者烂JJ

uchg, uchange, uimmutable
set the user immutable flag (owner or super-user only)

hidden set the hidden flag [Hide item from GUI]本文来自leo108's blog

如果一个文件拥有uchg属性,则不可以修改、删除或者重命名该文件,所以只需执行chflags nouchg filename去除uchg属性,然后rm即可。

本文来自http://leo108.com

浅析vqmod原理及优化思路

前几天有个朋友要我帮忙给opencart增加一个小功能,需求很简单,分分钟就搞定了。未经允许严禁转载

PS:我看过的第一个PHP开源系统的代码应该就是opercart,大学时代还用opencart的引擎(框架)写过一个小网站。

在opencart的后台,看到有opencart的文档链接,好奇点了进去。vQmod

然后我就发现了一个有趣的东西:vqmod,http://docs.opencart.com/display/opencart/vQmod

根据文档的描述,vqmod技术可以在不修改系统文件的情况下,对原系统的功能做任意更改,这样做的一个很大的好处是新系统不会因为二次开发而不能把系统升级到最新版本。未经允许严禁转载

vqmod在官方wiki上有简单介绍原理https://code.google.com/p/vqmod/wiki/About未经允许严禁转载

大概原理就是把原系统中的所有include(_once)、require(_once)中的文件路径替换成VQMod::modCheck(),参数就是该文件路径,该函数会根据用户定义的规则,把相关文件修改之后存一份缓存,然后返回缓存文件的路径,这样就可以实现对原系统的修改。

从代码层面来看,首先是执行了VQMod::bootup(),这个函数会扫描vqmod/xml文件夹下所有的xml文件,解析xml文件后放入静态属性$_mods中。而VQMod::modCheck这个函数根据文件路径,生成一个缓存文件的路径,如果该缓存文件存在并且未过期就直接返回缓存文件路径,否则检查静态属性$_mods中是否存在对该文件修改的规则,存在就根据规则生成新文件存入缓存文件,并返回缓存文件路径,否则返回原文件路径。

采集者烂JJ

vqmod需要dom扩展的支持,用于解析xml文件,如果一个php运行环境不支持dom扩展,则vqmod就不能使用。个人认为这个完全可以用php或json来替代,使用php可以直接用一个return array()来返回需要的信息;而使用json的唯一问题是引号,会导致规则编写不方便。这个也比较容易解决,可以做一个vqmod规则生成页面,用工具来生成对应的规则文件。