PHP-RSA加密跨域通讯实战

本文涉及的产品
密钥管理服务KMS,1000个密钥,100个凭据,1个月
简介:

PHP-RSA加密跨域通讯实战

AUTH:PHILO EMAIL:lijianying12 at gmail.com

基于POST GET 的http通讯虽然非常成熟,但是很容易被人监听。 并且如果使用跨域jsonp的通讯很容易在历史记录中发现通讯网址以及参数。为了克服这些问题, 并且降低服务器成本,我们没有使用SSL而使用 RSA加密。文章中的php加密解密 JS的加密解密 互相加密解密 都能验证通过。

其中PHP依赖常见的OPENSSL LIB 。 JS依赖 jsencrypt。

我们使用jsonp get RSA加密通讯好处如下:

  1. 前后分离适合cdn加速。
  2. 安全跨域更适合松散结构的网站。
  3. 不用去买ssl证书了。

首先要生成密匙对


  
  
  1. openssl genrsa 1024 > private.key
  2. openssl rsa -in private.key -pubout > public.key

JS的RSA加密流程

下载最新版本请移步到github:jsencrypt 代码在目录BIN下面是否用压缩的根据情况决定。

生成KEY


  
  
  1. var keySize = 1024; //加密强度
  2. var crypt = new JSEncrypt({default_key_size: keySize}); //RSA 操作对象
  3. //方法1 (async)
  4. crypt.getKey(function () {
  5. crypt.getPrivateKey();
  6. crypt.getPublicKey();
  7. });
  8. //方法2:
  9. crypt.getKey();
  10. crypt.getPrivateKey();
  11. crypt.getPublicKey();

客户端加密场景:


  
  
  1. var crypt1 = new JSEncrypt(); //新建rsa对象
  2. var publickey = '\
  3. -----BEGIN PUBLIC KEY-----\
  4. MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQC3N8LJFqlsa6loCgFpgZVMr/Sx\
  5. DMQY7pr0euNQfh2g+UVPbB0MGhoc7nWL0FQhCgDedbjQw/nGFStFx7W1+0o1oRTY\
  6. u5ebNVivZSobraUv7LJvwT8O66Zs8cxbKLqQ/nE/WwJvXomSIckH6R8iOUO8/QT9\
  7. kv6/L0Uma3qA07pmDQIDAQAB\
  8. -----END PUBLIC KEY-----\
  9. ';
  10. crypt1.setPublicKey(publickey );//添加来自服务端的publickey
  11. crypt1.encrypt('abc'); //返回值为加密后的结果

客户端解密场景:


  
  
  1. var privatekey = '-----BEGIN RSA PRIVATE KEY-----\
  2. MIICXQIBAAKBgQC3N8LJFqlsa6loCgFpgZVMr/SxDMQY7pr0euNQfh2g+UVPbB0M\
  3. Ghoc7nWL0FQhCgDedbjQw/nGFStFx7W1+0o1oRTYu5ebNVivZSobraUv7LJvwT8O\
  4. 66Zs8cxbKLqQ/nE/WwJvXomSIckH6R8iOUO8/QT9kv6/L0Uma3qA07pmDQIDAQAB\
  5. AoGBAKba3UWModbfZXQeSJLxNCqWw9zJp3ydL/keQQ35DLqgyIJAD2QKEWXvtJUT\
  6. sMo19fyicSGOmFXQyYvPCKkmpLkOMAj1XaNpSMtSrcMx+gC01PO6Ey9rsUxW1g3u\
  7. fpqbEk9E3a5AtCS0I61nbUpRL6rqMtR5o2wcNR3TLtJt7pjxAkEA7hlFJKU1zWGp\
  8. OvvkJDnHc2NOCEJoGjqCR9wwv96+/xAykl2laI6WvEbbhjoO0+8+d17oigjhneS5\
  9. 2UKFcfqw7wJBAMT+MCQ5TYLQlvjrBaDMqOdLsqtaDE6CpkrgwV820QMvHOo3R4Xd\
  10. uSbrA2tOr9t2/x+FzF971lRGdPFIch9UYMMCQQCZtO6SDaWCBP3++gX57OL5dq41\
  11. XsldxU+9nERMWTvr5UUAgDv8F7Dvsr6dFHXmE5i77yUnlzwvdi0UOIF1Z2U5AkBV\
  12. wyRKYPgx34Ya0JcerntKV1Zt60I4XADx0G/feAn/DN/VyENHMISPQPm4GgXN0jy4\
  13. CJQ1bcCd6B65fQTSRvXpAkA2Vv5yXzeKDls/AyxHEoros/VYftVc1HOFC++q13Rw\
  14. NH2rnlRT8FMTFEqL9MYRqvvYAFf5VmH0M1Nx5t4LRN+l\
  15. -----END RSA PRIVATE KEY-----\
  16. ';
  17. var crypt2 = new JSEncrypt();//新建加密对象
  18. crypt2.setPrivateKey(privatekey);//给加密对象设置privatekey
  19. crypt2.getPublicKey();//Tip 我们是不需要存储publickey的直接用private能得到publickey
  20. crypt2.decrypt("MeUqWB5LwTh8crzPqbZtEtKuZxYvPWH9CTCChK1qoBzIgIXGPCdzNMbiH0cCYHl5qWSERIDOgDIgv4dXsIMjEJ5q0cp/qNQYHM5va0iw0UvKvQB1E8aWtY2nFEPy4F+ArQ0Mj/ijr/CntEP1jHKC3WU9nu2kYrBIBnbj14Bs+kI=");//调用解密方法

但是虽然写到了这里,加密方面还是不够用,因为1024长度的RSA加密最多只能加密长度为117的字符串。而URL长度最多为4k因此这里我们要让加密长度达到2691以达到能用的程度。

那么这种加密长度大概能容纳多少数据呢? 我们借助json-generator来帮忙生成JSON


  
  
  1. sdata =[
  2. {
  3. "_id": "542f9ac2359c7d881bc0298e",
  4. "index": 0,
  5. "guid": "db1dacc1-b870-4e3c-bc1a-80dfd9506610",
  6. "isActive": false,
  7. "balance": "$1,570.15",
  8. "picture": "http://placehold.it/32x32",
  9. "age": 36,
  10. "eyeColor": "blue",
  11. "name": "Effie Barr",
  12. "gender": "female",
  13. "company": "ZORK",
  14. "email": "effiebarr@zork.com",
  15. "phone": "+1 (802) 574-3379",
  16. "address": "951 Cortelyou Road, Wikieup, Colorado, 4694",
  17. "about": "Sunt reprehenderit do laboris velit qui elit duis velit qui. Nostrud sit eiusmod cillum exercitation veniam ad sint irure cupidatat sunt consectetur magna. Amet nisi velit laboris amet officia et velit nisi nostrud ipsum. Cupidatat et fugiat esse minim occaecat cillum enim exercitation laboris velit nisi est enim aute. Enim do pariatur\r\n",
  18. "registered": "2014-05-08T15:26:35 -08:00",
  19. "latitude": 48.576424,
  20. "longitude": 146.634137,
  21. "tags": [
  22. "esse",
  23. "proident",
  24. "quis",
  25. "consectetur",
  26. "magna",
  27. "tempor",
  28. "anim"
  29. ],
  30. "friends": [
  31. {
  32. "id": 0,
  33. "name": "Trisha Cannon"
  34. },
  35. {
  36. "id": 1,
  37. "name": "Todd Bullock"
  38. },
  39. {
  40. "id": 2,
  41. "name": "Eileen Drake"
  42. },
  43. {
  44. "id": 3,
  45. "name": "Ferrell Kelly"
  46. },
  47. {
  48. "id": 4,
  49. "name": "Fischer Blankenship"
  50. },
  51. {
  52. "id": 5,
  53. "name": "Morales Mann"
  54. },
  55. {
  56. "id": 6,
  57. "name": "Brandie Pittman"
  58. },
  59. {
  60. "id": 7,
  61. "name": "Virgie Kerr"
  62. }
  63. ],
  64. "greeting": "Hello, Effie Barr! You have 1 unread messages.",
  65. "favoriteFruit": "apple"
  66. },
  67. {
  68. "_id": "542f9ac21c260d03e763a4f2",
  69. "index": 1,
  70. "guid": "9e3a3d8a-26f8-46b7-aca0-336a194808b1",
  71. "isActive": true,
  72. "balance": "$3,617.89",
  73. "picture": "http://placehold.it/32x32",
  74. "age": 31,
  75. "eyeColor": "brown",
  76. "name": "Butler Best",
  77. "gender": "male",
  78. "company": "SPORTAN",
  79. "email": "butlerbest@sportan.com",
  80. "phone": "+1 (905) 428-3046",
  81. "address": "798 Joval Court, Wanship, Delaware, 8974",
  82. "about": "Nostrud occaecat id sunt pariatur ad nisi do veniam sit officia non consequat amet fugiat. Est eiusmod labore ut cillum qui eu elit ut eiusmod exercitation. Ut anim nostrud eiusmod voluptate tempor proident id do pariatur. In Lorem ullamco ea irure adipisicing. Quis est dolor ex commodo aliqua nisi elit sit elit anim fugiat sunt amet. Enim consequat ipsum occaecat ipsum tempor deserunt dolor veniam nostrud. Anim cillum ullamco cupidatat aute velit fugiat sit enim in amet anim mollit dolor eiusmod.\r\n",
  83. "registered": "2014-08-02T06:15:44 -08:00",
  84. "latitude": -20.529765,
  85. "longitude": 2.396578,
  86. "tags": [
  87. "consequat",
  88. "enim",
  89. "magna",
  90. "sunt",
  91. "Lorem",
  92. "quis",
  93. "commodo"
  94. ],
  95. "friends": [
  96. {
  97. "id": 0,
  98. "name": "Kenya Rice"
  99. },
  100. {
  101. "id": 1,
  102. "name": "Hale Knowles"
  103. },
  104. {
  105. "id": 2,
  106. "name": "Michael Stephens"
  107. },
  108. {
  109. "id": 3,
  110. "name": "Holder Bailey"
  111. },
  112. {
  113. "id": 4,
  114. "name": "Garner Luna"
  115. },
  116. {
  117. "id": 5,
  118. "name": "Alyce Sawyer"
  119. },
  120. {
  121. "id": 6,
  122. "name": "Rivas Owens"
  123. },
  124. {
  125. "id": 7,
  126. "name": "Jan Petersen"
  127. }
  128. ],
  129. "greeting": "Hello, Butler Best! You have 8 unread messages.",
  130. "favoriteFruit": "banana"
  131. }
  132. ]

 

表单json能达到这么长已经是很极端的情况了。因此这种方法绝对是够用的。

长表单内容加解密方法:


  
  
  1. function encrypt_data(publickey,data)
  2. {
  3. if(data.length> 2691){return;} // length limit
  4. var crypt = new JSEncrypt();
  5. crypt.setPublicKey(publickey);
  6. crypt_res = "";
  7. for(var index=0; index < (data.length - data.length%117)/117+1 ; index++)
  8. {
  9. var subdata = data.substr(index * 117,117);
  10. crypt_res += crypt.encrypt(subdata);
  11. }
  12. return crypt_res;
  13. }
  14. function decrypt_data(privatekey,data)
  15. {
  16. var crypt = new JSEncrypt();
  17. crypt.setPrivateKey(privatekey);
  18. datas=data.split('=');
  19. var decrypt_res="";
  20. datas.forEach(function(item)
  21. {
  22. if(item!=""){de_res += crypt.decrypt(item);}
  23. });
  24. return decrypt_res;
  25. }

 ##########NextPage[title=]##########

PHP的RSA加密

php加密解密类

首先要检查phpinfo里面有没有openssl支持


  
  
  1. class mycrypt {
  2.  
  3. public $pubkey;
  4. public $privkey;
  5.  
  6. function __construct() {
  7. $this->pubkey = file_get_contents('./public.key');
  8. $this->privkey = file_get_contents('./private.key');
  9. }
  10.  
  11. public function encrypt($data) {
  12. if (openssl_public_encrypt($data, $encrypted, $this->pubkey))
  13. $data = base64_encode($encrypted);
  14. else
  15. throw new Exception('Unable to encrypt data. Perhaps it is bigger than the key size?');
  16.  
  17. return $data;
  18. }
  19.  
  20. public function decrypt($data) {
  21. if (openssl_private_decrypt(base64_decode($data), $decrypted, $this->privkey))
  22. $data = $decrypted;
  23. else
  24. $data = '';
  25.  
  26. return $data;
  27. }
  28.  
  29. }

密匙文件位置问题,是放到访问接口的附近就可以了如果是CI的话就放到index.php旁边就行了。 但是要注意一点,一定要做访问设置,不然key会暴出来的,那时候信息一旦截获就惨了。

类的使用


  
  
  1. $rsa = new mycrypt();
  2. echo $rsa -> encrypt('abc');
  3. echo $rsa -> decrypt('W+ducpssNJlyp2XYE08wwokHfT0bm87yBz9vviZbfjAGsy/U9Ns9FIed684lWjYyyofi/1YWrU0Mp8vLOYi8l6CfklBY=');

长数据加密解密


  
  
  1. function encrypt_data($publickey,$data)
  2. {
  3. $rsa = new mycrypt();
  4. if($publickey != ""){
  5. $rsa -> pubkey = $publickey;
  6. }
  7. $crypt_res = "";
  8. for($i=0;$i<((strlen($data) - strlen($data)%117)/117+1); $i++)
  9. {
  10. $crypt_res = $crypt_res.($rsa -> encrypt(mb_strcut($data, $i*117, 117, 'utf-8')));
  11. }
  12. return $crypt_res;
  13. }
  14. function decrypt_data($privatekey,$data)
  15. {
  16. $rsa = new mycrypt();
  17. if($privatekey != ""){ // if null use default
  18. $rsa ->privkey = $privatekey;
  19. }
  20. $decrypt_res = "";
  21. $datas = explode('=',$data);
  22. foreach ($datas as $value)
  23. {
  24. $decrypt_res = $decrypt_res.$rsa -> decrypt($value);
  25. }
  26. return $decrypt_res;
  27. }

JSONP 跨域通讯

我们经过千辛万苦经过加密终于能做到通讯安全了。 当然我们的下一步是通过JSONP 的get通讯来实现跨域通讯啦。 经过测试:我们的JS中最长的Case url长度是3956 在加上跨域url callbac参数,经过测试正好差20到4095 (一般的URI长度限制为4K)


  
  
  1. $.ajax({
  2. type:"get",
  3. async:false, // 设置同步通讯或者异步通讯
  4. url:"http://22500e31b5a12457.sinaapp.com/ubtamat/getPubKey?c=hknHQKIy3dyeeajyAwZ+raUkV1ezFbgU8zk+54cNQtrcEGozUjXpYhbC6fxz2hCOgp9feIsM1xKJFm5pkAGQ2UcUOc5EJNCAz6L0mXkZbTBmh3PufWxOE7TaicqRCRtZGGNB2qpm2WruXjYg1lPcrPz/rhFZx4DSJvEHkCm7ZU0=......(加密后的结果太长,省略)",
  5. dataType:"jsonp",
  6. jsonp: "",
  7. });

 


  
  
  1. header("Content-type: application/javascript; charset=utf-8");
  2. $response = "console.log('test response!')";
  3. $callback = $this->input->GET('callback');
  4. echo $callback.$response;

PHP代码是CI框架controler中的部分代码 并且经过了必要的裁剪。 更加细节的参数都放到GET里面就可以了。 处理之后按照上面的形式处理返回值就ok 如果你配置成功了,你将会在网页的控制台上看到自己动态的, 或者像我一样静态的控制台输出。 如果要是想获取数据到网页的话还是要借助回调函数来实现

JSONP跨域获取通讯结果

请看下面代码:

客户端代码


  
  
  1. var global = null;
  2. function jpc(result)
  3. {
  4. global = result.msg;
  5. }
  6.  
  7. $.ajax({
  8. type:"get",
  9. async:false, // 设置同步通讯或者异步通讯
  10. url:"http://22500e31b5a12457.sinaapp.com/ubtamat/getPubKey",
  11. dataType:"jsonp",
  12. jsonp: "jpc",
  13. });

服务器端代码


  
  
  1. header("Content-type: application/javascript; charset=utf-8");
  2. $response = "jpc({'msg':123456})";
  3. $callback = $this->input->GET('callback');
  4. echo $callback.$response;

此次通讯的结果会在jcp当中调用执行,并且返回的内容会记录到 global 变量当中。

实战

从上文中,我们已经找到了整个加密过程方法了,但是距离实战还是有一定距离的。 首先我们实战的话需要克服接口比较少,功能比较多,单个接口维护用时比较长的问题。

为了解决上面的问题我们做出如下设计。

客户端方面:

设计一个通讯类:只管跟服务器通讯。别的业务什么都不管。


  
  
  1. //create connection object.
  2. var ConnServ = new Object();
  3.  
  4. ConnServ.tmpResponse = "not initial";
  5.  
  6. //call back function register slot.
  7. ConnServ.CallBackFunction=function(){console.log(
  8. "call back function set error ! U must set a business call back function!"
  9. )};
  10.  
  11. //input only encrypted data!!!
  12. //send data to server
  13. ConnServ.send=function(data)
  14. {
  15. data = data.replace(/\+/g,"$"); //replace all + as $
  16. $.ajax({
  17. type:"get",
  18. async:false,
  19. url:"http://22500e317.sinaapp.com/ubtamat?c="+data,
  20. dataType:"jsonp",
  21. jsonp: "jpc"
  22. });
  23. return "Send Finish";
  24. }
  25.  
  26. //default call back funcation
  27. function jpc(res)
  28. {
  29. ConnServ.tmpResponse = res.msg;
  30. ConnServ.CallBackFunction();
  31. }
  32.  
  33.  
  34. //public key store.
  35. ConnServ.getpublickey = function()
  36. {
  37. return "\-" +
  38. "----BEGIN PUBLIC KEY----- " +
  39. ...................................................
  40. "-----END PUBLIC KEY-----";
  41. }

在上面代码中请注意,RSA加密过后的字符串当中有一个非法字符+要转换成其他合法字符发送到服务器才可以。 不然参数会错误。 等传输到服务器中自己转换回来在解密就好了。

服务器端方面:

首先我们接收到消息之后要对消息进行解密,之后根据报文内容选择服务器上的功能。然后把其他参数输入到业务类中执行即可。 因此我们使用了命令模式来实现单一接口的丰富业务功能。 其他的我们需要对CI框架的配置进行调整: 首先global config里面需要调整 $config['global_xss_filtering'] = FALSE; 因为如果传输过来的报文解密不了就直接抛弃不进行处理(防止CC攻击第一层)这样就从url上防止了攻击的可能性。 当然我们还是没有完全避免注入风险这时我们就需要在业务类里面调用安全模块:


  
  
  1. $this->security->xss_clean()

来实现第二层的XSS攻击。这是服务器端设计主要需要说的位置。

服务器获取数据处理全过程

  1. 从get接口获得参数c的加密数据
  2. 对数据进行RSA解密。
  3. 判断数据包时间戳。如果超时直接抛弃(防止从浏览器记录中直接发送request到服务器,下面是安全方面的说明)
    • 首先如果不修改数据只修改时间戳不可能从截获的数据报文中实现,因为需要重新加密,如果想得到内容需要服务器上的privatekey解密保证安全
    • 如果数据包截获直接发送数据包在超时范围内直接获取数据包内容,也不能实现攻击,因为在客户端有临时RSA密匙对生成并且在发送的时候会同时发送publickey 给服务器做session的存储内容并且伪装客户的客户端没有privatekey所以获取任何关于登陆之后的消息根本无法解析。
  4. 对解密后的数据进行xss检查
  5. 解析报文中需要调用什么功能直接调用反射得到业务类的实例
  6. 调度业务类,并且把得到的参数赋值给业务执行函数的参数。

服务器处理数据过程只跟业务有关

服务器返回数据全过程

  1. 业务处理完成之后针对每一个用户的登陆情况对返回值进行加密。
  2. response

以上业务涉及的部分代码(给出的代码未涉及以上说的安全部分。)


  
  
  1. //CI 控制器里面的方法
  2. public function index()
  3. {
  4. header("Content-Type: text/html;charset=UTF-8");
  5. $callback = $this -> input->GET('callback');
  6. $input_data = str_replace("$","+",$this->input->GET('c'));
  7. $input_data =$this -> rsa->decrypt_data($input_data);
  8. if($input_data == ""){return;}//如果数据不对解析就会失败,直接抛弃数据包,避免cracker构造数据包问题
  9. //这里插入时间戳检查代码
  10. //这插入xss检查
  11. $output_data = command($input_data);
  12. $response = "jpc({'msg':".$output_data."})";
  13. $callback = $this->input->GET('callback');
  14. echo $callback.$response;
  15. }

 


  
  
  1. //命令模式中的业务调度方法
  2. function command($input)
  3. {
  4. try
  5. {
  6. $obj_input = json_decode($input);
  7. $action = $obj_input -> {"action"};
  8. $business_action = new ReflectionClass($action);
  9. $instance = $business_action->newInstanceArgs();
  10. $output = $instance->Action($obj_input);
  11. //对output变量进行rsa加密
  12. return "'".$output."'"; // here only accept string
  13. }
  14. catch(Exception $e)
  15. {
  16. return "'".$e->getMessage()."'";
  17. }
  18. }

 

以下是配合业务进行的工具函数:


  
  
  1. //命令接口定义
  2. interface ICommand {
  3. function Action($arg_obj);
  4. }
  5. //把此函数放到system/core/common.php
  6. //实现了输入一个文件夹就自动加载所有文件夹中的所有的类。
  7. if ( ! function_exists('require_once_dir'))
  8. {
  9. function require_once_dir($path)
  10. {
  11. $dir_list = scandir($path);
  12. foreach($dir_list as $file)
  13. {
  14. if ( $file != ".." && $file != "." )
  15. {
  16. require_once($path."/".$file);
  17. }
  18. }
  19. }
  20. }
  21.  
  22. //使用:
  23. //在application/config/autoload.php中添加类似如下代码:
  24. require_once_dir(APPPATH."/controllers/lib");
  25. require_once_dir(APPPATH."/controllers/actions");

以下是实现业务的例子:


  
  
  1. class register implements ICommand{
  2. public function Action($arg_obj)
  3. {
  4. return "we are do nothing: ".json_encode($arg_obj);
  5. }
  6. }

通过以上基本方法,我们可以实现,只要业务继承我们声明的接口就可以开始写业务了。 别的什么都不用管,专注于业务即可,其他的安全、IO等问题都已经一并解决。 并且每一个业务都进行了rsa加密xss攻击过滤伪造数据包攻击。 以及在response加密只能是固定客户端才能看到报文内容的全过程。 但是一定要注意一点,注册这个业务后面要嵌套登陆进行,不然看不到返回值。

数据包必须包含的要素:

  1. acton (业务名)
  2. req_time (请求时间)
  3. public_key (如果是注册跟登陆时候需要提交临时公匙)

总结

因为时间仓促所以只能写到这里了。 如果您发现了我文章中的bug欢迎发email批评指正。非常感谢! 同时本方案也会成为我们开源社区linux52.com后台系统中的接口设计方案。 当然我们社区所有维护的文档都会进行反复验证,如果出问题我们会及时更新。 以维护文档的正确性。 点击=这=里=查看文档最新版本。

原文发布时间:2014-10-11

本文来自云栖合作伙伴“linux中国”

目录
相关文章
|
3月前
|
存储 关系型数据库 MySQL
深入浅出PHP:从基础到实战
【9月更文挑战第20天】本文将带你走进PHP的世界,从最基础的语法开始,逐步深入到实战应用。我们将通过实例讲解PHP的核心概念,如变量、控制结构、函数等,然后介绍如何使用PHP进行数据库操作和文件处理。最后,我们将展示如何用PHP构建一个简单的博客系统,让你在实践中掌握PHP的应用。无论你是编程新手,还是有一定经验的开发者,都能在这篇文章中找到有价值的信息。让我们一起探索PHP的魅力吧!
|
10天前
|
Linux PHP 数据安全/隐私保护
2024授权加密系统PHP网站源码
2024授权加密系统PHP网站源码
89 58
|
22天前
|
存储 安全 网络安全
网络安全的盾与剑:漏洞防御与加密技术的实战应用
在数字化浪潮中,网络安全成为保护信息资产的重中之重。本文将深入探讨网络安全的两个关键领域——安全漏洞的防御策略和加密技术的应用,通过具体案例分析常见的安全威胁,并提供实用的防护措施。同时,我们将展示如何利用Python编程语言实现简单的加密算法,增强读者的安全意识和技术能力。文章旨在为非专业读者提供一扇了解网络安全复杂世界的窗口,以及为专业人士提供可立即投入使用的技术参考。
|
27天前
|
编译器 PHP 开发者
PHP 8新特性解析与实战应用####
随着PHP 8的发布,这一经典编程语言迎来了诸多令人瞩目的新特性和性能优化。本文将深入探讨PHP 8中的几个关键新功能,包括命名参数、JIT编译器、新的字符串处理函数以及错误处理改进等。通过实际代码示例,展示如何在现有项目中有效利用这些新特性来提升代码的可读性、维护性和执行效率。无论你是PHP新手还是经验丰富的开发者,本文都将为你提供实用的技术洞察和最佳实践指导。 ####
29 1
|
1月前
|
自然语言处理 安全 PHP
深入浅出PHP编程:从基础到实战
【10月更文挑战第36天】本文将带你走进PHP的奇妙世界,无论你是初学者还是有一定经验的开发者,都将从中获益。文章首先介绍PHP的基础概念和语法,然后通过实际代码示例,展示如何利用PHP进行高效的Web开发。最后,我们将探讨一些高级主题,如面向对象编程、数据库操作以及安全性问题。让我们一起开启PHP的学习之旅吧!
|
1月前
|
存储 Serverless PHP
PHP编程入门:从基础到实战
【10月更文挑战第35天】本文将带你走进PHP的世界,从最基本的语法开始,逐步深入到实际应用。我们将通过简单易懂的语言和实际代码示例,让你快速掌握PHP编程的基础知识。无论你是初学者还是有一定经验的开发者,都能在这篇文章中找到你需要的内容。让我们一起探索PHP的魅力吧!
|
2月前
|
应用服务中间件 网络安全 Apache
Discuz! X3.5 开启ssl证书加密后微信、公众号无消息、支付宝通讯中断等
Discuz! X3.5 开启ssl证书加密后微信、公众号无消息、支付宝通讯中断等、支付宝支付实际支付成功,显示未支付等,都属于通讯中断,需要联系DZ官方付费修改程序,屏蔽防CC!
65 4
|
1月前
|
前端开发 中间件 PHP
PHP框架深度解析:Laravel的魔力与实战应用####
【10月更文挑战第31天】 本文作为一篇技术深度好文,旨在揭开PHP领域璀璨明星——Laravel框架的神秘面纱。不同于常规摘要的概括性介绍,本文将直接以一段引人入胜的技术剖析开场,随后通过具体代码示例和实战案例,逐步引导读者领略Laravel在简化开发流程、提升代码质量及促进团队协作方面的卓越能力。无论你是PHP初学者渴望深入了解现代开发范式,还是经验丰富的开发者寻求优化项目架构的灵感,本文都将为你提供宝贵的见解与实践指导。 ####
|
1月前
|
存储 Java PHP
PHP编程之旅:从基础到实战
【10月更文挑战第23天】本文将带你走进PHP的世界,探索这个流行的服务器端脚本语言的魅力。无论你是编程新手还是有一定经验的开发者,这篇文章都将为你提供有价值的信息和实用的代码示例。我们将从PHP的基本概念开始,然后深入到实际应用中,最后通过一个实战项目来巩固所学知识。让我们一起开启这段激动人心的PHP编程之旅吧!
26 0
|
3月前
|
JavaScript 前端开发 安全
js逆向实战之烯牛数据请求参数加密和返回数据解密
【9月更文挑战第20天】在JavaScript逆向工程中,处理烯牛数据的请求参数加密和返回数据解密颇具挑战。本文详细分析了这一过程,包括网络请求监测、代码分析、加密算法推测及解密逻辑研究,并提供了实战步骤,如确定加密入口点、逆向分析算法及模拟加密解密过程。此外,还强调了法律合规性和安全性的重要性,帮助读者合法且安全地进行逆向工程。
111 11