分类目录归档:图片/文字

《蓝树叶》 联想到现在的网络时代

蓝树叶

     美术课上,老师教同学们画风景,要画上树、房子和小山。李丽画好了近处的房子、远处的小山。她正要画树,可是绿色铅笔找不到了。
     李丽看看旁边的林园园已经把树画好了,树叶那么绿,真惹人爱。李丽小声对林园园说:“把绿铅笔借给我用一用行吗?”林园园吞吞吐吐地说:“我还没有画完 呢。”李丽只好坐在那里看林园园画。等她画完了,李丽说:“现在可以把绿铅笔借给我了吧?”林园园说:“我怕你把笔尖弄断了。”李丽说:“我小心一点儿”
林园园从盒子里拿出绿铅笔,说:“你要注意,不要削,画的时候不要用力,不要画得太多!”李丽连忙说:“我只画树叶和小草。”林园园皱着眉头,说:“还要 画小草吗?”李丽看了看林园园,没有接她的铅笔。李丽拿起自己的蓝铅笔,用心地画着一片片树叶。林园园看见这些蓝树叶,不由得脸红了。

篇小学时的课文相信大家都很熟悉,反正我一直以来,只要一想到借东西和自私这两样生活中最常见的事物,就想到蓝树叶课文中的借绿色铅笔画树叶,从’借’又想到现在的网络时代,网络时代讲究的是共享,在计算机网络中,大量的各种各样的信息都是无私的人们奉献出来的,总之不管是‘借’还是‘共享’其实都是给予别人方便。
最近一直想用自己的所学做一个网站,现在想好了,网站名称就叫 《蓝树叶》 其主题也是共享信息,给予别人方便的同时,也分享到别人给予的方便!

快乐分享 分享快乐》

今天开始写一个通用的企业网站程序

最近要帮我哥的公司做一个企业网站,再加以我叔公司的网站到现在还是半成品就挂到网上了,心中老是过意不去,但是自己都去好好解决一下,最近也有空了,准备花上一周左右的时间,把这两个企业网站做掉。

做完这二个网站后,自己也准备做一个网站自己用了,时间过得真快,还有二个月就快过年了。。。感慨一下

下面是今天用了一下午修改了下自己以前的网站后台ui,后台界面就不重新做了凑合着用了。。

主是把数据库的视图查询部分做了下,然后完善了下框架的路由!

 

PHP代码
  1. <?php  
  2. class Model{  
  3.   
  4.     public $db;  
  5.   
  6.     public function __construct()  
  7.     {  
  8.         $this->db= new PDO(conf(‘dbType’).‘:host=’.conf(‘dbHost’).‘;dbname=’.conf(‘dbName’), conf(‘dbUser’), conf(‘dbPass’));  
  9.         $this->db->exec(‘SET NAMES ‘.conf(‘charset’));   
  10.     }  
  11.   
  12.     function query($sql)  
  13.     {  
  14.         $rs = $this->db->query($sql);  
  15.         return $rs;  
  16.     }  
  17.   
  18.   
  19.     function beginTransaction()  
  20.     {  
  21.         return $this->db->beginTransaction();  
  22.     }  
  23.   
  24.     function commit()  
  25.     {  
  26.         return $this->db->commit();  
  27.     }  
  28.   
  29.     function rollBack()  
  30.     {  
  31.         return $this->db->rollBack();  
  32.     }  
  33.   
  34.     function exec($sql)  
  35.     {  
  36.         return $this->db->exec($sql);  
  37.     }  
  38.   
  39.     function lastInsertId()  
  40.     {  
  41.         return $this->db->lastInsertId();  
  42.     }  
  43.   
  44.   
  45.     function setAttribute($attr$value)  
  46.     {  
  47.         return $this->db->setAttribute($attr$value);  
  48.     }  
  49.   
  50.   
  51.     function insert($table,$data)  
  52.     {  
  53.         $table=conf(‘dbprefix’).$table;  
  54.         $fields = "";  
  55.         $values = "";  
  56.         foreach($data as $field=>$value)  
  57.         {  
  58.             $fields .= "`$field`, ";  
  59.             $values .= "’$value’, ";  
  60.         }  
  61.           
  62.         $fields = substr_replace($fields"", -2, 1);  
  63.         $values = substr_replace($values"", -2, 1);  
  64.   
  65.         $sql = "insert into `$table` ($fields) values ($values)";  
  66.         //echo $sql;  
  67.         $rs = $this->db->query($sql) ;  
  68.         return $this->db->lastInsertId();  
  69.     }  
  70.   
  71.     function del($table,$keyarr){  
  72.         $table=conf(‘dbprefix’).$table;  
  73.         $where = "";  
  74.         foreach($keyarr as $key=>$key_value)  
  75.         {  
  76.             $where .= "`$key`=’$key_value’";  
  77.         }  
  78.         $sql = "delete from `$table` where $where";  
  79.         $rs = $this->db->query($sql);  
  80.         return $rs;  
  81.     }  
  82.   
  83.     function update($table,$data$keyarr)  
  84.     {  
  85.         $table=conf(‘dbprefix’).$table;  
  86.         $set = "";  
  87.         foreach($data as $field=>$value)  
  88.         {  
  89.             $set .= "`$field`=’$value’, ";  
  90.         }  
  91.         $set = substr_replace($set"", -2, 1);  
  92.   
  93.         $where = "";  
  94.         foreach($keyarr as $key=>$key_value)  
  95.         {  
  96.             $where .= "`$key`=’$key_value’";  
  97.         }  
  98.           
  99.   
  100.         $sql = "update `$table` set $set where $where";  
  101.         //echo $sql;  
  102.         $rs = $this->db->query($sql);  
  103.           
  104.         return $rs;  
  105.     }  
  106.   
  107.     function find($table,$where=,$field=‘*’,$order=,$group=)  
  108.     {  
  109.         $sql=$this->_map($table,$where,$field,$order,,$group);  
  110.         //echo $sql;  
  111.         return $this->_find($sql);  
  112.     }  
  113.       
  114.     function findAll($table,$where=,$field=‘*’,$order=,$limit=,$group=)  
  115.     {  
  116.         $sql=$this->_map($table,$where,$field,$order,$limit,$group);  
  117.         return $this->_findAll($sql);  
  118.     }  
  119.   
  120.     function count($table,$where=,$field=‘*’,$order=,$group=)  
  121.     {  
  122.         $sql=$this->_map($table,$where,$field,$order,,$group);  
  123.         return $this->_count($sql);  
  124.     }  
  125.       
  126.     function _count($sql)  
  127.     {  
  128.         $rs=$this->query($sql);  
  129.         $count=$rs->fetchColumn();  
  130.         $count=emptyempty($count)?0:$count;  
  131.         return $count;  
  132.     }  
  133.       
  134.     function _find($sql)  
  135.     {  
  136.         $rs=$this->query($sql);  
  137.         $row=$rs->fetch(PDO::FETCH_ASSOC);  
  138.         return $row;  
  139.     }  
  140.       
  141.     function _findAll($sql)  
  142.     {  
  143.         $rs=$this->query($sql);  
  144.         $row=$rs->fetchall(PDO::FETCH_ASSOC);  
  145.         return $row;  
  146.     }  
  147.      
  148.     function _map($table,$where=,$field=‘*’,$order=,$limit=,$group=)  
  149.     {  
  150.         $order=emptyempty($order)?:‘order by ‘.$order;  
  151.         $limit=emptyempty($group)?:‘limit ‘.$limit;  
  152.         $group=emptyempty($group)?:‘group by ‘.$group;  
  153.         $w=;  
  154.           
  155.         if(!emptyempty($where)){  
  156.             if(is_array($where)){  
  157.                 $w.=‘where’;  
  158.                 foreach($where as $k=>$v){  
  159.                     $w.=" $k=’$v’ and";  
  160.                 }  
  161.                 $w=substr($w,0,strlen($w)-3);  
  162.             }else{  
  163.                 $w=$where;  
  164.             }  
  165.         }  
  166.           
  167.         if(is_array($table)){  
  168.             $table1=conf(‘dbprefix’).$table[0];  
  169.             //echo $table1;  
  170.             $count=count($table);  
  171.             $joinleft=;  
  172.             array_shift($table);  
  173.             foreach($table as $v){  
  174.                 foreach($v as $k2=>$v2){  
  175.                     $k2=conf(‘dbprefix’).$k2;  
  176.                     foreach($v2 as $k3=>$v3){  
  177.                         $l="$table1.$k3=$k2.$v3";  
  178.                     }  
  179.                     $joinleft.="left join $k2 on $l ";  
  180.                 }  
  181.                   
  182.             }  
  183.             $sql="select $field from $table1 $joinleft $w $order $group $limit";  
  184.         }else{  
  185.             $sql="select $field from $table $w  $order $group $limit";  
  186.         }  
  187.         return $sql;  
  188.     }  
  189.   
  190. }  
  191. //fetch(‘PDO_FETCH_ASSOC’)  
  192. ?>  

 

网站后台界面 调整了部分css,对ff的兼容性更好些

11.jpg

网站的url形式

2222.jpg

简单网站统计功能的实现(PV IP 真实访客数)

主要统计三个数值,网站的pv 和ip 以及真实访客数 因为有详细的统计记录,所以可以实现的功能还有很大的扩展!把tallydate使用内存表的话,速度就会更快了

ss.jpg

需要用到二张表

 

SQL代码
  1. CREATE TABLE `tally` (   
  2.   `datedate NOT NULL,   
  3.   `pvtotal` int(10) NOT NULL,   
  4.   `iptotal` int(10) NOT NULL,   
  5.   `dltotal` int(10) NOT NULL,   
  6.   PRIMARY KEY  (`date`)   
  7. ) ENGINE=InnoDB DEFAULT CHARSET=utf8;   
  8.   
  9.   
  10.   
  11. CREATE TABLE `tallydata` (   
  12.   `tdid` int(10) NOT NULL auto_increment,   
  13.   `ip` int(10) NOT NULL,   
  14.   `cookie` varchar(32) NOT NULL,   
  15.   `datedate NOT NULL,   
  16.   `timeint(10) NOT NULL,   
  17.   `uri` varchar(255) default NULL,   
  18.   `referer` varchar(255) default NULL,   
  19.   PRIMARY KEY  (`tdid`)   
  20. ) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=5 ;   

 

统计的调用文件(用于所在统计的页面进行js调入):

(注:因项目使用的是ThinkPHP框架,所以贴出的只是代码片段!因为代码刚完成,所以没有做优化以及注释)

PHP代码
  1. <?php    
  2. class TallyAction extends Action{   
  3.     public function index(){   
  4.         if($_SERVER[‘SERVER_NAME’]!=‘fullyee.com’ && $_SERVER[‘SERVER_NAME’]!=‘www.fullyee.com’ && $_SERVER[‘SERVER_NAME’]!=‘company.fullyee.com’exit;   
  5.         $add[‘ip’]=ip2long(get_client_ip());   
  6.         if(!Cookie::is_set(‘fullyeetally’)){   
  7.             $value=md5(microtime().$add[‘ip’].rand());   
  8.                
  9.             $overTime=mktime(0,0,0,date(‘m’),date(‘d’)+1,date(‘Y’))-time();   
  10.             Cookie::set("fullyeetally",$value,time()+$overTime);   
  11.         }   
  12.         $fullyeetally=Cookie::get(‘fullyeetally’);   
  13.         $add[‘cookie’]=$fullyeetally;   
  14.         $add[‘date’]=date(‘Y-m-d’);   
  15.         $add[‘time’]=time();   
  16.         $add[‘uri’]=$_SERVER[‘SERVER_NAME’].$_SERVER[‘REQUEST_URI’];   
  17.         $add[‘referer’]=$_SERVER[‘HTTP_REFERER’];   
  18.         $tallydata_=D(‘Tallydata’);   
  19.         $tallydata_->create($add);   
  20.         $tallydata_->add();   
  21.            
  22.     }   
  23.   
  24.     public function saveData()   
  25.     {   
  26.         $tally_=D(‘Tally’);   
  27.         $tallydata_=D(‘Tallydata’);   
  28.         $nowDate=date(‘Y-m-d’,time()-3600*24);   
  29.   
  30.         $now[‘date’]=$nowDate;   
  31.         $now[‘iptotal’]=$this->gototal($nowDate,‘ip’);   
  32.         $now[‘pvtotal’]=$tallydata_->count(array(‘date’=>$nowDate),‘tdid’);   
  33.         $now[‘dltotal’]=$this->gototal($nowDate,‘cookie’);   
  34.   
  35.         if($tally=$tally_->find(array(‘date’=>$nowDate))){   
  36.             $tally_->save(array(‘iptotal’=>$now[‘iptotal’],‘pvtotal’=>$now[‘pvtotal’],‘dltotal’=>$now[‘dltotal’]),array(‘date’=>$nowDate));   
  37.         }else{   
  38.             $tally_->create($now);   
  39.             $tally_->add();   
  40.         }   
  41.         $timeDel=time()-3600*24*50;   
  42.         $tallydata_->query("delete from `tallydate` where `time`<$timeDel");   
  43.         echo ‘Success ‘+date(‘Y-m-d H:i:s’);   
  44.     }   
  45.   
  46.     function gototal($nowDate,$a)   
  47.     {   
  48.         $tallydata_=D(‘Tallydata’);   
  49.         $now[‘iptotal’]=$tallydata_->query("select count(distinct $a) from `tallydata` where `date`=’$nowDate’ ");   
  50.         return $now[‘iptotal’][0]["count(distinct $a)"];   
  51.     }   
  52. }    
  53. ?>  

程序中的 saveDate 部分,用于服务器的定时运行,最简单的方法就是用linux的crontab在一个访问人数比较少的时间wget一下saveDate代码就行了。。这段代码的作用是,把前一天的访问数据全部统计,以天为单位写入tally表!并且删除一定时间外的统计记录。默认是删除50天以前的全部统计记录

显示部分:

 

PHP代码
  1. <?php   
  2. class configAction extends Action{   
  3.     function _initialize(){   
  4.         header("Content-Type:text/html; charset=utf-8");   
  5.     }   
  6.        
  7.     public function tally()   
  8.     {   
  9.         $tally_=D(‘Tally’);   
  10.         $tallydata_=D(‘Tallydata’);   
  11.         $nowDate=date(‘Y-m-d’);   
  12.         $now[‘date’]=$nowDate;   
  13.         $now[‘iptotal’]=$this->gototal($nowDate,‘ip’);   
  14.         $now[‘pvtotal’]=$tallydata_->count(array(‘date’=>$nowDate),‘tdid’);   
  15.         $now[‘dltotal’]=$this->gototal($nowDate,‘cookie’);   
  16.            
  17.         if($tally=$tally_->find(array(‘date’=>$nowDate))){   
  18.             $tally_->save(array(‘iptotal’=>$now[‘iptotal’],‘pvtotal’=>$now[‘pvtotal’],‘dltotal’=>$now[‘dltotal’]),array(‘date’=>$nowDate));   
  19.         }else{   
  20.             $tally_->create($now);   
  21.             $tally_->add();   
  22.         }   
  23.         $today[‘pv’]=$now[‘pvtotal’];   
  24.         $today[‘ip’]=$now[‘iptotal’];   
  25.         $today[‘dl’]=$now[‘dltotal’];   
  26.         $yesterdayDate=date(‘Y-m-d’,time()-3600*24);   
  27.         //echo $yesterdayDate;   
  28.         $yesterday=$tally_->find(array(‘date’=>$yesterdayDate));   
  29.         //dump($yesterday);   
  30.         $yesterday[‘pv’]=isset($yesterday[‘pvtotal’])?$yesterday[‘pvtotal’]:‘0’;   
  31.         $yesterday[‘ip’]=isset($yesterday[‘iptotal’])?$yesterday[‘iptotal’]:‘0’;   
  32.         $yesterday[‘dl’]=isset($yesterday[‘dltotal’])?$yesterday[‘dltotal’]:‘0’;   
  33.   
  34.         $maxpv=$this->gomax(‘pvtotal’);   
  35.         $maxip=$this->gomax(‘iptotal’);   
  36.         $maxdl=$this->gomax(‘dltotal’);   
  37.         $max[‘pv’]=$maxpv[‘pvtotal’];   
  38.         $max[‘pvdate’]=$maxpv[‘date’];   
  39.         $max[‘ip’]=$maxip[‘iptotal’];   
  40.         $max[‘ipdate’]=$maxip[‘date’];   
  41.         $max[‘dl’]=$maxdl[‘dltotal’];   
  42.         $max[‘dldate’]=$maxdl[‘date’];     
  43.         $this->assign(‘today’,$today);   
  44.         $this->assign(‘yesterday’,$yesterday);   
  45.         $this->assign(‘max’,$max);   
  46.         $this->assign(‘nowtime’,date(‘Y年m月d日 H:i:s’));   
  47.         $this->display();   
  48.     }   
  49.     function gomax($a)   
  50.     {   
  51.         $tally_=D(‘Tally’);   
  52.         $max=$tally_->query("select * from `tally` order by `$a` desc limit 1");   
  53.         return $max[0];   
  54.     }   
  55.        
  56.     function gototal($nowDate,$a)   
  57.     {   
  58.         $tallydata_=D(‘Tallydata’);   
  59.         $now[‘iptotal’]=$tallydata_->query("select count(distinct $a) from `tallydata` where `date`=’$nowDate’ ");   
  60.         return $now[‘iptotal’][0]["count(distinct $a)"];   
  61.     }   
  62. }   
  63. ?>  

 

jquery使用Jcrop插件轻松实现上传图片后选取区域做头像

jquery使用Jcrop插件轻松实现上传图片后选取区域做头像

一般网站上传头像部分会比较麻烦,如果完全程序控制的话,程序把上传的图片自动裁切成指定大小的头像的话,很有可能会破坏头像的整体美观。所以现在大多数web2.0网站都有用js或flash来实现头像的上传与选取!

现在用jquery加上jcrop插件的话,就可以非常简单的实现这些功能!如果再加上编写的一些ajax处理部分的话就基本上可以满足一般web2.0的使用要求了

 aaaa1.jpg

aaaa2.jpg

PHP代码
  1. <?php  
  2. if($_GET[‘act’]==‘saveThumb’){  
  3.     $targ_w = $targ_h = 150;  
  4.     $jpeg_quality = 100;  
  5.   
  6.     $src = $_POST[‘bigImage’];  
  7.     $img_r = imagecreatefromjpeg($src);  
  8.     $dst_r = ImageCreateTrueColor( $targ_w$targ_h );  
  9.   
  10.     imagecopyresampled($dst_r,$img_r,0,0,$_POST[‘x’],$_POST[‘y’],$targ_w,$targ_h,$_POST[‘w’],$_POST[‘h’]);  
  11.   
  12.     header(‘Content-type: image/jpeg’);  
  13.     imagejpeg($dst_r,null,$jpeg_quality);  
  14.       
  15.     exit;  
  16. }  
  17. ?>  
  18. <html>  
  19.     <head>  
  20.   
  21.         <script src="jquery.pack.js"></script>  
  22.         <script src="jquery.Jcrop.pack.js"></script>  
  23.           
  24.         <link rel="stylesheet" href="jquery.Jcrop.css" type="text/css" />  
  25.           
  26.         <script language="Javascript">  
  27.   
  28.             $(function(){  
  29.   
  30.                   
  31.             });  
  32.               
  33.             function goss(){  
  34.   
  35.                 jQuery(‘#cropbox’).Jcrop({  
  36.                     onChange: showPreview,  
  37.                     onSelect: showPreview,  
  38.                     onSelect: updateCoords,  
  39.                     aspectRatio: 1  
  40.                 });  
  41.   
  42.             }  
  43.   
  44.             function updateCoords(c)  
  45.             {  
  46.                 $(‘#x’).val(c.x);  
  47.                 $(‘#y’).val(c.y);  
  48.                 $(‘#w’).val(c.w);  
  49.                 $(‘#h’).val(c.h);  
  50.             };  
  51.   
  52.             function checkCoords()  
  53.             {  
  54.                 if ($(‘#x’).val()==){  
  55.                     alert(‘请先上传头像然后选择裁切头像最后进行保存!’);  
  56.                     return false;  
  57.                 }  
  58.             };  
  59.               
  60.             function showPreview(coords)  
  61.             {  
  62.                 var rx = 150 / coords.w;  
  63.                 var ry = 150 / coords.h;  
  64.                 var w2=$("#bigwidth").val();  
  65.                 var h2=$("#bigheight").val();  
  66.                 jQuery(‘#preview’).css({  
  67.                     width: Math.round(rx * w2) + ‘px’,  
  68.                     height: Math.round(ry * h2) + ‘px’,  
  69.                     marginLeft: ‘-‘ + Math.round(rx * coords.x) + ‘px’,  
  70.                     marginTop: ‘-‘ + Math.round(ry * coords.y) + ‘px’  
  71.                 });  
  72.             }  
  73.   
  74.         </script>  
  75.   
  76.     </head>  
  77.   
  78.     <body>  
  79. <?php  
  80. if($_GET[‘act’]==‘upload’){  
  81.     if($_POST[‘upload’]==‘upload’){  
  82.           
  83.         $uploaddir =‘upload/’;  
  84.         $uploadfile = $uploaddir . basename($_FILES[‘file’][‘name’]);  
  85.         //print_r($_FILES[‘file’]);  
  86.         //echo $uploadfile;  
  87.       
  88.         if (move_uploaded_file($_FILES[‘file’][‘tmp_name’], $uploadfile)) {  
  89.             list($w$h$type$attr)=getimagesize($uploadfile);  
  90.             $str=;  
  91.             if($w>550){  
  92.                 $str="width:550px;";  
  93.             }  
  94.             if($h>550){  
  95.                 $str.="  height:550px;";  
  96.             }  
  97.             $str=emptyempty($str)?:"style=’ ".$str." ‘";  
  98.             $f1="<img src=’$uploadfile’ border=0 $str id=’cropbox’ >";  
  99.             $f2="<img src=’$uploadfile’ border=0  $str id=’preview’ >";  
  100.   
  101.               
  102.               
  103.             echo ‘<script language="javascript">parent.$("#showBig").html("’.$f1.‘");parent.$("#showThumb").html("’.$f2.‘");parent.goss();parent.$("#bigwidth").val("’.$w.‘");parent.$("#bigheight").val("’.$h.‘");parent.$("#bigImage").val("’.$uploadfile.‘");</script>’;  
  104.                   
  105.         }else {  
  106.             echo "<script>alert(‘文件上传失败!’);</script>";  
  107.         }  
  108.           
  109.     }  
  110. ?>  
  111.   
  112. <div style="margin:0px;font-size:12px;">  
  113. <FORM ACTION="?act=upload" METHOD=POST enctype="multipart/form-data">  
  114.   
  115.     <input type="file" name="file" id="file" />  
  116.   
  117.   <input type="submit" name="button" id="button" value="提交" />  
  118.   
  119.   <input name="upload" type="hidden" id="upload" value="upload" /><input type="hidden" name="MAX_FILE_SIZE" value="3000000" />  
  120. </FORM>  
  121. </div>  
  122.   
  123.   <?php  
  124.   exit;  
  125. }  
  126. ?>  
  127.   
  128.   
  129. <div id="showBig" style="width:500px;height:500px;border:2px solid #E6E0CE;padding:3px;"></div>  
  130. <iframe style="width:500px;height:60px;padding:0px;" src="?act=upload"></iframe>  
  131.   
  132.   
  133. <div id="showThumb" style="width:152px;height:152px;border:1px solid #cccccc;padding:1px; overflow: hidden;"></div>  
  134. <div style="margin-top:20px;">  
  135.     <form action="?act=saveThumb" method="post" onsubmit="return checkCoords();">  
  136.         <input type="hidden" id="bigImage" name="bigImage" />  
  137.         <input type="hidden" id="bigwidth" name="bigwidth" />  
  138.         <input type="hidden" id="bigheight" name="bigheight" />  
  139.         <input type="hidden" id="x" name="x" />  
  140.         <input type="hidden" id="y" name="y" />  
  141.         <input type="hidden" id="w" name="w" />  
  142.         <input type="hidden" id="h" name="h" />  
  143.         <input type="submit" value="保存用户头像" />  
  144.     </form>  
  145. </div>  
  146.     </body>  
  147.   
  148. </html>  

 

 

jcrop.zip

在windows下启用php的mail()函数进行发信

其实要想在windows下使用php的mail()函数进行发信的话,只要机器里安装了smtp就可以了

当然iis有内置的smtp,可是如果web服务器安装的是apache的话总不可能为了一个smtp而再去安装一个iis吧

所以找了个简单的smtp服务器软件(1st SMTP Server)找的是一个老版本的,才700多k,没有其它的无用功能!

先下载好1st SMTP Server后,运行里面的注册机,再运行主程序,进行注册。注册完了后就可以关闭主程序窗口了,在任务栏里双击1st SMTP Server图标,可以看到它的主界面!这时,应该smtp服务就正常运行了,像dns和smtp端口之类的默认就好了。

未命名.jpg

第二步。设置php.ini

找到

[mail function]
SMTP = localhost
smtp_port = 25
sendmail_from = web@phpd.cn

 

如果加了;的话,去掉就可以了,apache重启一下,这样php的mail()就可以正常工作了

 

1st SMTP Server 下载

fstsmtp.rar

 

今天花了一天时间整了个简单的框架

写时好像老是受thinkphp及fleaphp的影响,所以写了个不伦不类

而且好像效率上也不太行,光框架就占内存221k,执行速度也在0.02秒左右 如果加上smarty的话,内存立即增到500多k执行时间0.03秒左右!

以后把它再优化一下,写这个框架的初衷就是为了快速,低消耗!写下来后才发现要改正的地方还不少!

目前框架还是个半成品,只是实现了控制器部分,视图用的是smarty 然后数据模型部分现在还没有完善,缓存处理方面还没写!

未命名.jpg

解读 QQWry.Data ——IP地址库文件数据 [转自PHPCHINA 作者:terry39]

解读 QQWry.Data ——IP地址库文件数据

按项目计划,最近要实现用户IP地址到所在地址区域的转换,这里不仅只是解读出地区文字,还需要对应原有的地区数据;所以要把地区代码(机构组织码)与IP地址建立对应关系。我选择纯真版IP地址库来建立对应关系表。

从网上下载了最新的 QQWry.Data 库文件,首先碰到的一个问题是解析这个文件。根据网上的资料(LuamaQQ作者写的日志),根据自己的摸索,总结出了此文件的内容结构,以及解读方式。

一、文件结构
文件主要分三个结构
1、文件头,8个字节;
2、数据记录区,不定长度;
3、索引区,长度为 7 的整数倍;

二、文件头
文件头的8个字节分两部分,每个部分4个字节,分别指定了索引区的开始地址和结束地址。所以可以通过两个地址的差值 除 7 后 加 1 可以计算出总的记录数。

二、记录区
记录区的数据需要通过索引区的数据来获得各个数据的起始位置;本区数据记录了IP地址的结束地址和地区字符串;所有地区字符串都以 0x00 为结束。

三、索引区
检索IP对应的地区,关键就是找到IP起始地址对应的索引内容。一个IP索引数据包含7个字节,前4个字节是IP地址起始值,后3个字节是对应的IP数据记录在文件内的偏移地址;IP数据记录中,前 4 个字节是IP结束地址;紧跟的数据有两种模式: 0x01 模式 和 0x02 模式。

0x01模式,即在IP数据的第5个字节是 0x01,则在后面的 3 个字节是国家地区数据的偏移地址;国家地区数据包括国家和地区这两个字符串。即
—————————————————————
4字节 |  3字节 重定向 0x NN NN NN -> 国家地区数据的文件偏移地址
—————————————————————

0x02模式,即在IP数据的第5个字节是 0x02,则在后面的 3 个字节是国家数据的偏移地址,地区数据是再往后的字符串,以 0x00 截至。即
—————————————————————————–
4字节 | 3字节 重定向 0x NN NN NN -> 国家数据的文件偏移地址 | 地区字符串 | 0x00
—————————————————————————–

对于 0x01 模式所得到的 国家地区数据中,它可能又带有一个重定向结构,即
————————————–
国家字符串 | 0x00 | 地区字符串 | 0x00
————————————–

————————————————————————-
国家字符串 | 0x00 | 0x02 | 3字节 0x NN NN NN -> 地区字符串的文件偏移地址
————————————————————————-

对于前一种情况,比较简单,直接读出两个字符串数据就可以了;对于后一种情况,需要再次重定向到地区字符串的偏移地址,然后读取到 0x00 为字符串结尾。

对于这种采取地址映射实际字符串值的方式,主要作用是避免重复记录字符串值。在整个IP地址库文件中,有太多相同字符串记录了,采用 3 字节的映射地址要比重复记录字符串值节省太多空间了。

PHP代码
  1. <?php   
  2. function bin2ip($bin)   
  3. {   
  4.     $ip = ;   
  5.     $bd = str_split($bin, 1);   
  6.     for($i = 4; $i > 0; $i–){   
  7.         $ip .= "." . sprintf("%03d", implode(, unpack(‘s’$bd[$i-1] . chr(0))));   
  8.     }   
  9.     return substr($ip, 1);   
  10. }   
  11.   
  12. //————————————————–   
  13.   
  14. $f = fopen(‘QQWry.Dat’‘r’);   
  15.   
  16. $c = fread($f, 4);   
  17. $d = fread($f, 4);   
  18.   
  19. $index_begin = implode(, unpack(‘L’$c));   
  20. $index_end   = implode(, unpack(‘L’$d));   
  21. if($index_begin < 0) $index_begin += pow(2, 32);   
  22. if($index_end   < 0) $index_end   += pow(2, 32);   
  23.   
  24. $ip_num = ($index_end – $index_begin) / 7 + 1;   
  25.   
  26. echo "index begin at: $index_begin\n";   
  27. echo "index end   at: $index_end\n";   
  28. echo "ip data count : $ip_num\n";   
  29.   
  30. $output = ;   
  31.   
  32. for($i = 0; $i < $ip_num$i++){   
  33.   
  34.     //文件指针指到每个IP数据文件的索引取得索引数据(7字节)上   
  35.     fseek($f$i * 7 + $index_begin);   
  36.     $ip4 = fread($f, 4);                //IP起始地址   
  37.     if(strlen($ip4) < 4) exit(‘data file error’);   
  38.   
  39.     $ip3 = fread($f, 3);                //IP记录偏移地址   
  40.     if(strlen($ip3) < 3) exit(‘data file error’);   
  41.   
  42.     $dataseek = implode(, unpack(‘L’$ip3 . chr(0)));   
  43.     if($dataseek < 0) $index_ip_record += pow(2, 32);   
  44.   
  45.     //指向记录区 $dataseek 位置查找记录   
  46.     fseek($f$dataseek);   
  47.     $ipdata = fread($f, 4);                //IP结束地址   
  48.     if(strlen($ipdata) < 4) exit(‘data file error’);   
  49.        
  50.     $area = ;   
  51.     $country = ;   
  52.   
  53.     //读一个标记位   
  54.     $flag = fread($f, 1);   
  55.     if($flag == chr(1)){                //国家名偏移标记位    模式一 0x01   
  56.         $area1seek = fread($f, 3);   
  57.         if(strlen($area1seek) < 3) exit(‘data file error’);   
  58.         $area1seek = implode(, unpack(‘L’$area1seek . chr(0)));   
  59.         fseek($f$area1seek);   
  60.   
  61.         $flag = fread($f, 1);            //可能又是标记位   
  62.            
  63.            
  64.     }   
  65.     if($flag == chr(2)){                //国家地区 重定向   
  66.         $area1seek = fread($f, 3);   
  67.         if(strlen($area1seek) < 3) exit(‘data file error’);   
  68.         $area1seek = implode(, unpack(‘L’$area1seek . chr(0)));   
  69.   
  70.         $flag = fread($f, 1);   
  71.         if($flag == chr(2)){   
  72.             $area2seek = fread($f, 3);   
  73.             $area2seek = implode(, unpack(‘L’$area2seek . chr(0)));   
  74.             fseek($f$area2seek);   
  75.         }else{   
  76.             fseek($f, -1, SEEK_CUR);   
  77.         }   
  78.         while(($c = fread($f, 1)) != chr(0)) $area .= $c;   
  79.            
  80.         fseek($f$area1seek);   
  81.         while(($c = fread($f, 1)) != chr(0)) $country .= $c;    
  82.     }else{   
  83.         fseek($f, -1, SEEK_CUR);   
  84.         while(($c = fread($f, 1)) != chr(0)) $country .= $c;    
  85.   
  86.         $flag = fread($f, 1);        //如果地区是重定向的   
  87.         if($flag == chr(2)){   
  88.             $area2seek = fread($f, 3);   
  89.             $area2seek = implode(, unpack(‘L’$area2seek . chr(0)));   
  90.             fseek($f$area2seek);   
  91.         }else{   
  92.             fseek($f, -1, SEEK_CUR);   
  93.         }   
  94.   
  95.         while(($c = fread($f, 1)) != chr(0)) $area .= $c;   
  96.     }   
  97.     $adata = trim($country) . trim($area); //$country是国家字符串 , $area 是地区字符串   
  98.   
  99. }   
  100.   
  101.   
  102. fclose($f);   
  103. ?>