2014年1月19日 17:30:27
我这篇文章就说了一句话:用cookie保存页面间跳转时的参数
情景: 客服在后台操作的时候,经常从列表页进入到编辑页,编辑完信息后,还要自动返回之前的列表页
问题:
1.如果客服是从列表第二页跳转到编辑页面,那么编辑完成后应该会自动跳转到列表页的第二页,这样就牵扯到这个"第二页"的参数的保存问题
2.如果是多个页面一层层深入,那么中间参数保存是很浪费脑细胞的
方案:
之前想过两种方法:一是用smarty模版插件保存,二是:用程序保存.但这都会使程序中徒添很多与逻辑无关的代码,跳转赋值保存非常麻烦
现在又想起了两种方法:
前提:用PHP数组模拟"栈",将客服操作的URL用栈保存下来
第一种方案:用cookie/session保存这个"栈",达到跳转间保存URL的目的(建议使用)
第二种方案:用memcache保存这个栈(或者用其他nosql直接使用其栈的功能)
我刚刚实现了用memcache保存栈的目的,用cookie和session也一样
优缺点:
cookie/session可以设置expire=0(有效期)来保证管理员关闭浏览器后清除所有操作历史(cookie栈)
也可以设置expire=n,来指定保存栈的时间
选用memcache没什么优点,只是我在一开始想到了用memcache来实现这个功能,后来代码写完了,才想起来cookie/session方案,不过原理都一样
用服务器内存存储时要注意,栈名称要根据管理员来区分,用cookie的话则不用担心这个
缺点:
栈内只保存了URL,也就是说只能保存get中间参数,对于中间参数有post的就不能记录了
cookie的数量大小各个浏览器有限制:http://blog.sina.com.cn/s/blog_59b6f23e0100a703.html
注意:如果程序执行时,连续设置同名cookie,而不刷新页面,只有最后一步的cookie设置会生效
例如:
你的程序有两段代码:
第一段先执行清空cookie的代码
第二段再重新给cookie赋值,
这两步之间不把cookie输出到客户端浏览器,则清空cookie那个步骤是不起作用的,不管是用setcookie()还是header('Set-Cookie: ....'),都不行
下边贴出memcache测试代码:
1 <form method="post"> 2 <input type="submit" name="submit" value="ok !"> 3 </form> 4 <?php 5 $m = new Memcache(); 6 $m->connect('localhost', '11211'); 7 8 //栈的有效期为从现在开始今天的剩余时间 9 $tomorrow = strtotime('tomorrow'); 10 $now = time(); 11 $expire = $time_remaind = $tomorrow-$now; 12 13 if (!empty($_POST['submit'])) { 14 $url_refer = out($m, $expire); 15 if ($url_refer) { 16 header("Location: http://$url_refer"); 17 exit; 18 } 19 } else { 20 in($m, $expire); 21 } 22 23 //入栈 24 //判断是刷新的话不再将当前URL记入栈 25 function in($m, $expire) 26 { 27 $url_now = $_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI']; 28 29 $jsonStack = $m->get('zzb'); 30 if (empty($jsonStack)) { 31 $arrStack = array(); 32 $arrStack[] = $url_now; 33 } else { 34 $arrStack = json_decode($jsonStack, true); 35 $url_refer = array_pop($arrStack); 36 if ($url_refer == $url_now) { 37 $arrStack[] = $url_refer; 38 } else { 39 $arrStack[] = $url_refer; 40 $arrStack[] = $url_now; 41 } 42 } 43 $jsonStack = json_encode($arrStack); 44 $m->set('zzb', $jsonStack, MEMCACHE_COMPRESSED, $expire); 45 } 46 47 //出栈 48 //如果没有获得来源URL,则还是跳转到本页面 49 function out($m, $expire) 50 { 51 $jsonStack = $m->get('zzb'); 52 if (empty($jsonStack)) { 53 echo 'empty stack<br>'; 54 } 55 $arrStack = json_decode($jsonStack, true); 56 57 $url_now = array_pop($arrStack); 58 $url_refer = array_pop($arrStack); 59 60 $jsonStack = json_encode($arrStack); 61 $m->set('zzb', $jsonStack, MEMCACHE_COMPRESSED, $expire); 62 63 return $url_refer; 64 }
html 按钮表示进入此页操作完成后,提交返回
下边是cookie的测试代码(判断是否是刷新行为时,可以用PHP的end()函数代替array_pop()):
1 <?php 2 ob_start(); 3 if (!empty($_POST['submit'])) { 4 $url_refer = out(); 5 if ($url_refer) { 6 header("Location: http://$url_refer"); 7 exit; 8 } 9 } else { 10 in(); 11 } 12 13 //入栈 14 //判断是刷新的话不再将当前URL记入栈 15 function in() 16 { 17 $url_now = $_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI']; 18 19 $jsonStack = empty($_COOKIE['zzb']) ? '' : $_COOKIE['zzb']; 20 if (empty($jsonStack)) { 21 $arrStack = array(); 22 $arrStack[] = $url_now; 23 } else { 24 $arrStack = json_decode($jsonStack, true); 25 $url_refer = array_pop($arrStack); 26 if ($url_refer == $url_now) { 27 $arrStack[] = $url_refer; 28 } else { 29 $arrStack[] = $url_refer; 30 $arrStack[] = $url_now; 31 } 32 } 33 echo '<pre>'; 34 print_r($arrStack); 35 $jsonStack = json_encode($arrStack); 36 setcookie('zzb', $jsonStack); 37 } 38 39 //出栈 40 //如果没有获得来源URL,则还是跳转到本页面 41 function out() 42 { 43 $jsonStack = empty($_COOKIE['zzb']) ? '' : $_COOKIE['zzb']; 44 if (empty($jsonStack)) { 45 echo 'empty stack<br>'; 46 } 47 $arrStack = json_decode($jsonStack, true); 48 49 $url_now = array_pop($arrStack); 50 $url_refer = array_pop($arrStack); 51 echo '<pre>'; 52 var_dump($url_refer, $url_now); 53 print_r($arrStack); 54 55 $jsonStack = json_encode($arrStack); 56 setcookie('zzb', $jsonStack); 57 58 return $url_refer; 59 } 60 ?> 61 <form method="post"> 62 <input type="submit" name="submit" value="ok !"> 63 </form>
-------------分割线---------
刚才准备把这个功能应用到项目中去,但是发现,当时写代码的时候只图方便省事,把所有的功能都写到了一个controller里了,导致该功能入口需要很大的改变
如果把不同的功能写到不同的控制器中,这样可以为每个控制器开一个栈,逻辑实现会简单些,but ...... 错呀
--------------再分------------
上边说的和解决的都是理论情景:列表页->子列表页->子列表页->...->进入某列表项->一层层返回
但实际经常遇到情况是:列表页->进入某列表项->返回列表
如果功能逻辑被设计成理论情况,即多层进入,多层返回,那就是设计有问题,需要重新设计
下边是用cookie写的单层进入返回(列表页->进入某列表项->返回列表):
public function in($FunctionModule) { ob_start(); $url_now = $_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI']; //每次都重写栈,保证栈内只有一个元素,适合单层跳转 $arrStack = array($url_now); $jsonStack = json_encode($arrStack); setcookie($FunctionModule, $jsonStack); return; } public function out($FunctionModule) { ob_start(); $jsonStack = empty($_COOKIE[$FunctionModule]) ? '' : $_COOKIE[$FunctionModule]; if (empty($jsonStack)) { return ''; } $arrStack = json_decode($jsonStack, true); $url_refer = array_pop($arrStack); $jsonStack = json_encode($arrStack); setcookie($FunctionModule, $jsonStack); return $url_refer; }
没有显示行号,方便有用到的同学复制(*^__^*)