实现思路:
将传入的长链接存入内存数据库中,并记录创建时间,返回自增ID。
将返回的id进行10进制转64进制编码:如自增id为10时,64进制转换为A;即使当我们记录到1亿条时,1亿的64进制为:F9eEA,才5个字符长度,非常适合作为短链接的参数。然后组合成 http://域名/编码参数/。我们既得到了短链接。当我们访问短链接时,解析参数转为10进制。到数据库中查找获得相应的链接,进行301跳转。此功能既为完成。
var_dump(2147483647); // int
var_dump( 2147483648); // float
PHP int型的最大值就是 2147483647,即2的31次方 -1,因为32位的最高位要用来表示正负。
当数值超过一定大小会解析失败。
数据表字段设计简单如下:
id int,//主键自增ID
url varchar ,//要转换短链接的url
time int,//创建时间
function b64dec($b64) { //64进制转换成10进制
$map = array(
'0'=>0,'1'=>1,'2'=>2,'3'=>3,'4'=>4,'5'=>5,'6'=>6,'7'=>7,'8'=>8,'9'=>9,
'A'=>10,'B'=>11,'C'=>12,'D'=>13,'E'=>14,'F'=>15,'G'=>16,'H'=>17,'I'=>18,'J'=>19,
'K'=>20,'L'=>21,'M'=>22,'N'=>23,'O'=>24,'P'=>25,'Q'=>26,'R'=>27,'S'=>28,'T'=>29,
'U'=>30,'V'=>31,'W'=>32,'X'=>33,'Y'=>34,'Z'=>35,'a'=>36,'b'=>37,'c'=>38,'d'=>39,
'e'=>40,'f'=>41,'g'=>42,'h'=>43,'i'=>44,'j'=>45,'k'=>46,'l'=>47,'m'=>48,'n'=>49,
'o'=>50,'p'=>51,'q'=>52,'r'=>53,'s'=>54,'t'=>55,'u'=>56,'v'=>57,'w'=>58,'x'=>59,
'y'=>60,'z'=>61,'_'=>62,'='=>63
);
$dec = 0;
$len = strlen($b64);
for ($i = 0; $i < $len; $i++) {
$b = $map[$b64{$i}];
if ($b === NULL) {
return FALSE;
}
$j = $len - $i - 1;
$dec += ($j == 0 ? $b : (2 << (6 * $j - 1)) * $b);
}
return $dec;
}
function decb64($dec) { //10进制转换成64进制
if ($dec < 0) {
return FALSE;
}
$map = array(
0=>'0',1=>'1',2=>'2',3=>'3',4=>'4',5=>'5',6=>'6',7=>'7',8=>'8',9=>'9',
10=>'A',11=>'B',12=>'C',13=>'D',14=>'E',15=>'F',16=>'G',17=>'H',18=>'I',19=>'J',
20=>'K',21=>'L',22=>'M',23=>'N',24=>'O',25=>'P',26=>'Q',27=>'R',28=>'S',29=>'T',
30=>'U',31=>'V',32=>'W',33=>'X',34=>'Y',35=>'Z',36=>'a',37=>'b',38=>'c',39=>'d',
40=>'e',41=>'f',42=>'g',43=>'h',44=>'i',45=>'j',46=>'k',47=>'l',48=>'m',49=>'n',
50=>'o',51=>'p',52=>'q',53=>'r',54=>'s',55=>'t',56=>'u',57=>'v',58=>'w',59=>'x',
60=>'y',61=>'z',62=>'_',63=>'=',
);
$b64 = '';
do {
$b64 = $map[($dec % 64)] . $b64;
$dec /= 64;
} while ($dec >= 1);
return $b64;
}
短链接实现方式一调用示例:
echo decb64('要加密的字符串');
//结果:加密后的字符串
echo b64dec('加密后的字符串');
//结果:要加密的字符串
短链接实现方式二:
class short_url{
private static $charset = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz_=";
public static function short($url,$host='http://www.t.cn/'){
$key = "DiorsCoder";
$url_hash = md5($key.$url);
$len = strlen($url_hash);//值为32
#将加密后的串分成4段,每段4字节,对每段进行计算,一共可以生成四组短连接
for($i=0;$i<4;$i++){
$url_hash_piece = substr($url_hash,$i*$len/4,$len/4);
#将分段的位与0x3fffffff做位与,0x3fffffff表示二进制数的30个1,即30位以后的加密串都归零
$hex = hexdec($url_hash_piece) & 0x3fffffff;#此处需要用到hexdec()将16进制字符串转为10进制数值型,否则运算会不正常
$short_url = "";#生成6位短连接
for($j=0;$j<6;$j++){
#将得到的值与0x0000003d,3d为61,即charset的坐标最大值
$short_url .= @self::$charset[$hex & 0x0000003d];//charset 坐标最大值 0x0000003d
$hex = $hex >> 5; #循环完以后将hex右移5位
}
$short_url_arr[] = $host.$short_url;
}
return $short_url_arr;
}
}
短链接实现方式二调用示例:
$url = new short_url();
$short_url = $url->short("http://www.baidu.com");
print_r($short_url);
打印结果:
/** 生成短网址
* @paramString $url 原网址
* @return String
*/
function dwz($url){
$code=floatval(sprintf('%u', crc32($url)));
$surl='';
while($code){
$mod=fmod($code, 62);
if($mod>9 && $mod<35){
$mod=chr($mod + 61);
}
$surl .= $mod;
$code = floor($code/62);
}
return $surl;
}
echo dwz('http://www.baidu.com');