高质量PHP代码的50个技巧(3)

Python032

高质量PHP代码的50个技巧(3),第1张

42

43

44

45

/**

Method to execute a command in the terminal

Uses :

1. system

2. passthru

3. exec

4. shell_exec

*/

function terminal($command)

{

//system

if(function_exists('system'))

{

ob_start()

system($command , $return_var)

$output = ob_get_contents()

ob_end_clean()

}

//passthru

else if(function_exists('passthru'))

{

ob_start()

passthru($command , $return_var)

$output = ob_get_contents()

ob_end_clean()

}

//exec

else if(function_exists('exec'))

{

exec($command , $output , $return_var)

$output = implode("\n" , $output)

}

//shell_exec

else if(function_exists('shell_exec'))

{

$output = shell_exec($command)

}

else

{

$output = 'Command execution not possible on this system'

$return_var = 1

}

return array('output' =>$output , 'status' =>$return_var)

}

terminal('ls')

上面的函数将运行shell命令, 只要有一个系统函数可用, 这保持了代码的一致性.

5. 灵活编写函数

?

1

2

3

4

5

6

function add_to_cart($item_id , $qty)

{

$_SESSION['cart']['item_id'] = $qty

}

add_to_cart( 'IPHONE3' , 2 )

使用上面的函数添加单个项目. 而当添加项列表的时候,你要创建另一个函数吗? 不用, 只要稍加留意不同类型的参数, 就会更灵活. 如:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

function add_to_cart($item_id , $qty)

{

if(!is_array($item_id))

{

$_SESSION['cart']['item_id'] = $qty

}

else

{

foreach($item_id as $i_id =>$qty)

{

$_SESSION['cart']['i_id'] = $qty

}

}

}

add_to_cart( 'IPHONE3' , 2 )

add_to_cart( array('IPHONE3' =>2 , 'IPAD' =>5) )

现在, 同个函数可以处理不同类型的输入参数了. 可以参照上面的例子重构你的多处代码, 使其更智能.

6. 有意忽略php关闭标签

我很想知道为什么这么多关于php建议的博客文章都没提到这点.

?

1

2

3

<?php

echo "Hello"

//Now dont close this tag

这将节约你很多时间. 我们举个例子:

一个 super_class.php 文件

?

1

2

3

4

5

6

7

8

9

<?php

class super_class

{

function super_function()

{

//super code

}

}

?>

//super extra character after the closing tag

index.php

?

1

2

require_once('super_class.php')

//echo an image or pdf , or set the cookies or session data

这样, 你将会得到一个 Headers already send error. 为什么? 因为 “super extra character” 已经被输出了. 现在你得开始调试啦. 这会花费大量时间寻找 super extra 的位置。因此, 养成省略关闭符的习惯:

?

1

2

3

4

5

6

7

8

9

<?php

class super_class

{

function super_function()

{

//super code

}

}

//No closing tag

这会更好.

7. 在某地方收集所有输入, 一次输出给浏览器

这称为输出缓冲, 假如说你已在不同的函数输出内容:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

function print_header()

{

echo "<p id='header'>Site Log and Login links</p>"

}

function print_footer()

{

echo "<p id='footer'>Site was made by me</p>"

}

print_header()

for($i = 0 $i <100$i++)

{

echo "I is : $i '

}

print_footer()

替代方案, 在某地方集中收集输出. 你可以存储在函数的局部变量中, 也可以使用ob_start和ob_end_clean. 如下:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

function print_header()

{

$o = "<p id='header'>Site Log and Login links</p>"

return $o

}

function print_footer()

{

$o = "<p id='footer'>Site was made by me</p>"

return $o

}

echo print_header()

for($i = 0 $i <100$i++)

{

echo "I is : $i '

}

echo print_footer()

为什么需要输出缓冲:

>>可以在发送给浏览器前更改输出. 如 str_replaces 函数或可能是 preg_replaces 或添加些监控/调试的html内容.

>>输出给浏览器的同时又做php的处理很糟糕. 你应该看到过有些站点的侧边栏或中间出现错误信息. 知道为什么会发生吗? 因为处理和输出混合了.

8. 发送正确的mime类型头信息, 如果输出非html内容的话.

输出一些xml.

?

1

2

3

4

5

6

$xml = '<?xml version="1.0" encoding="utf-8" standalone="yes"?>'

$xml = "<response>

<code>0</code>

</response>"

//Send xml data

echo $xml

工作得不错. 但需要一些改进.

?

1

2

3

4

5

6

7

$xml = '<?xml version="1.0" encoding="utf-8" standalone="yes"?>'

$xml = "<response>

<code>0</code>

一、无需任何PHP扩展的加密

此类加密的代表有 威盾PHP加密专家、PHP在线加密平台、PHP神盾 等。

此类加密都是以eval函数为核心,辅以各式各样的字符串混淆和各种小技巧,来达到加密目的(更准确的说,应该算是混淆)。下面以一个简单的hello world为例来说明此类加密的大体过程。

<?php

echo "hello world"

首先 ,我们把这段代码变为通过eval执行的

<?php

eval('echo "hello world"')

然后 ,我们再进行一些转换,比如说base64编码

<?php

eval(base64_decode('ZWNobyAiaGVsbG8gd29ybGQiOw=='))

就这样子,我们的第一个加密过的php代码新鲜出炉了。。。

上面这个例子非常非常简单,基本上任何有一点php语言基础甚至别的语言基础的人都能轻松的看懂并解密。因此,我们需要一些方法让这个加密至少看上去不是那么简单。

二、同时采用多种编码函数

除了刚才提到的base64,php还有许多内置的编码函数,例如urlencode、gzcompress等。把这些函数混合使用可以提高解密的复杂度(不是难度),此外还可以使用strtr来制定自己的编码规则。 使用变量来代替函数名 使用特定字符来命名变量

这儿所说的特定字符是一些极其相似的字符,如I和1,0和O。试想一下满屏都是O和0组成的变量,并且每一个的名字长度都在10个字符以上。。。 判断文件自身是否被修改

这个功能看似容易,对文件做一下摘要再进行下对比即可知道是否被修改了,但是如何才能在文件内把摘要嵌入进去呢?我没有找到完美的方案,但一个变通的方案还是很容易的。。。

<?php

$code = substr(file_get_contents(__FILE__), 0, -32)

$hash = substr(file_get_contents(__FILE__), -32)

if (md5($code) !== $hash) {

exit('file edited')

}

ACBC41F727E00F85BEB3440D751BB4E3

当然,你可以把这个校验字符串放在别的位置来提高破解的难度。有了这个,别人想破解你的程序可就得多费一点功夫了。。。

既然知道了原理,那解密自然也就非常简单了,总体来说就三步:

把eval替换为输出,比如echo 根据编码规则把字符串还原 如果文件未解密完全,从第一步开始继续

当然,实际上的解密过程并没有这么简单,比如说如果加密的时候使用了gzcompress,那得到的数据将会包含一些二进制数据,而采用一般的文本编辑器打开时这些数据都会显示为乱码,并且在保存时丢失部分数据。解决方法很简单也很麻烦,那就是使用二进制(16进制)方式打开、修改和保存。

求一PHP算法,字典生成。时间一到再加100分。如:字符:0-9,长度:1,

那就生成0,1,2,3,4,5,6,7,8,9

长度:2,就会生成00-99

现在要求字符可以包括a-z,或者其他特殊符号,求一高效的生成算法。

参考答案一

function get_string($strlen){

$source='0123456789'//任意字符

$len = strlen($source)//长度

$return = array()

for($i = 0 $i <$len$i++){

for($j = 0$j <$strlen$j++){

$return[$i] .= $i

}

}

return implode(',', $return)

}

如果输入长度2: 输出结果就是:

00,11,22,33,44,55,66,77,88,99

参考答案二

优化了进位算法:

PHP code =0$no--){ $word=$source{$series[$no]}.$word$series[$no]+=$tonext_valueif($no>0){ if($series[$no]==$len){ $series[$no]=0$tonext_value=1}else{ $tonext_value=0} } } echo "$word "} } gene_dic(2)?>

简单的说,我会把这个理解为0-9(十进制)下十个数字生成两位数字、可重复的排列问题。

排列算法我自己建立过的就是简单的N进制下的+1算法,保证可以遍历。

即:

初始化到0,

1. +1

2. 是否超过要生成的位数?否,则回到1

3. 输出

参考答案三

PHP code =0$no--){//循环遍历数组每次从源字串中取一个字符,为便于进位运算,取字符是从后往前取 $word=$source{$series[$no]}.$word//先取出一个字符 //取出一个字符后就要判断当前数组元素如何如果改变值,为下一次“大循环”做准备 if($no==$n-1){//末位的判断,它比较特殊,每次大循环都要增值 if($series[$no]==$len-1){ $series[$no]=0$tonext_value=1//归零时就进位 }else{ $series[$no]+=1$tonext_value=0//未归零就增值,不进位 } }elseif($no<$n-1){//中间位的进位判断 $series[$no]+=$tonext_value//先取得上一位的进位值 if($series[$no]==$len){ $series[$no]=0$tonext_value=1//归零了就继续进位 }else{ $tonext_value=0//不归零就不进位 } }else{ $series[$no]+=$tonext_value//大循环次数决定了“老大”是只进不出的。 } } echo "$word "//输入单词 } } gene_dic(2)//测试,结果OK。

参考答案四

PHP code =0$no--){//循环遍历数组每次从源字串中取一个字符,为便于进位运算,取字符是从后往前取 $word=$source{$series[$no]}.$word//先取出一个字符 //取出一个字符后就要判断当前数组元素如何如果改变值,为下一次“大循环”做准备 if($no==$n-1){//末位的判断,它比较特殊,每次大循环都要增值 if($series[$no]==$len-1){ $series[$no]=0$tonext_value=1//归零时就进位 }else{ $series[$no]+=1$tonext_value=0//未归零就增值,不进位 } }elseif($no<$n-1){//中间位的进位判断 $series[$no]+=$tonext_value//先取得上一位的进位值 if($series[$no]==$len){ $series[$no]=0$tonext_value=1//归零了就继续进位 }else{ $tonext_value=0//不归零就不进位 } }else{ $series[$no]+=$tonext_value//大循环次数决定了“老大”是只进不出的。 } } echo "$word "//输入单词 } } gene_dic(2)//测试,结果OK。

参考答案五

应该是:

function get_string($strlen){

$source='0123456789'

$len = strlen($source)

$return = array()

for($i = 0 $i <$len$i++){

for($j = 1$j <= $strlen$j++){

$return[$i] .= substr($source,$i,1)

}

}

return implode(',', $return)

}

【拓展阅读】如何开始一门语言的学习

一门语言从发明到演进必有原因。

现在还有很多人推荐学习不同的语言。通过比较,了解它的发展史,

创始人的初心等因素都需要留意。多个思考,这个语言在5年,在10年后还是否保持活力?

当有几个类似的语言被选择时,我们不妨对它们做一个Swat分析。

列出这些语言的共同点,还有它们之间的规则差异。

了解语言的发展史

开发语言从汇编开始,如最早的计算机ENIAC,使用的就是它来编程。

再到Fortarin,再到C语言,Cobol,Basic。每一个语言都与当时发展的阶段有点密切关联。

人类的每个发明都与懒惰有关,语言也是为便捷性而生。有的语言

C是除汇编外最重视效率的语言,扩展的C++也继承了此特性。Perl是做文本处理效率最佳的语言,虽然它的发展有点慢。PHP做Web开发,是“世界上最好的.语言”,Python的阅读性和大数据处理都做得样样俱佳。

当了解语言的历史沿革后,会让我们对其创始人有很强烈的兴趣,成为忠实的脑残粉,学习该语言的兴趣会更浓烈。

人们常常说某个语言比哪个好,这其实没有必要。不必要为其它人的语言所惑,需要你自己做出选择。

语言的共通点

这个星球的人都是一个鼻子两双只水汪汪的大眼睛,与人们的模样一般,编程语言也有一个大致相同的长相。

语法:这是开发此语言定义的规则“套路”:

运算符顺序,变量常量定义/作用域,表达式定义,字符串定义,行尾结束符等。

流程控制:循环控制

这些语法都是成对的,如if,for,while,foreach,有的语言还提供goto这样类似汇编语言的语法。

函数与方法

一些能够复用的高质量代码组合。函数执行后有返回,有递归,有嵌套,还有干完活就完事的简单任务。有静态函数和动态函数区分。

容器

数组,哈希表(也叫散列),字典等用来保存数据的容器。

错误/例外处理

现代编程语言基本都支持出错的抛出,除了C语言之外。

比如硬盘不足,网络出错,黑客攻击等情形。就像购物中心里出现煤气泄露时,监测设备,物联网设备能够及时记录与传递给指挥中心。

没有错误抛出的语言,需要自己考虑尽可能出错的场景并处理,比如:

if(is_overfllow)

//处理

if(network_error)

//处理

可以还有不少需要关注的维度,这会让代码变得艰涩难懂,也难以维护。

我们可以用这样的方式,让其更简洁:

on error goto ERROR

ERROR:

..//

但这总是会需要我们照顾很多情形。于是C++推出了一个语法:

try{

//可能会出错的代码

//可能会出错的代码

}catch{

//处理出错的逻辑

//处理出错的逻辑

}finally{

//出不出错都要执行的代码

}

最后一句是微软公司给业界提供贡献的finally代码块。

以上这些成为语言处理异常机制的基础。

容器

容器是很重要的一节,所以我们单独再提出来。很多逻辑处理,使用容器保存数据,该语言会提供便捷的方法来提供存取。

比如C、Perl、PHP、Ruby中均提供的数组和关联数组,LISP提供的列表,Java、Python提供的元组、链表等。

虽然名字相同,但是实现方式却是完全不同,使用方法当然也不一样。

没有万能的容器,只有最合适的。可以从节省内存,节约时间还是编码效率等综合考虑。

字符串与字符编码

是否支持unicode编码。从摩斯码到ASCII到统一的Unicode编码支持。

并发处理

有的语言在设计时并无此方面的考虑,或者天生设计存在缺陷。

即多线程,多进程的概念。包括共享,锁,事备等特性。

面向对象

支持类,继承,模块,包,命名空间,闭包等。有这些特性才会让人们的工作变得更便利、更有效率。

小结

学习一门语言的关键,需要我们在平静地心绪下,带着浓厚的兴趣去学习,在比较中学习,在历史中学习。

有时候感觉还是不够通畅,先做知识的搬运工也是不错。另外,不断的实践会让我们的信心更足。