目前已知有两个方法能实现,其一是利用微博官方的开放平台的发微博 API 实现。这种方法也是我最先考虑的方法,因为毕竟是官方渠道,应该靠谱些。
然而经过一番折腾后,我发现这一方法根本不可行。因为微博限制了这一 API 的调用频次,在连续上传时立即就会被封堵。
方法二是利用微博的 MINI 发布框(http://weibo.com/minipublish)的图片上传功能。
正是这个发布框暴露了微博的一个图片上传接口
http://picupload.service.weibo.com/interface/pic_upload.php
网上的这些微博图床大都也是利用这个接口实现的。
利用这一接口的优点是可以实现快速连续上传图片而不被封。缺点是必须要带着微博登录的 Cookie 才能访问。
说了这么多,现在贴一下相关php代码:
/** * 上传图片到微博图床 * @author mengkun http://mkblog.cn * @param $file 图片文件/图片url * @param $multipart 是否采用multipart方式上传 * @return 返回的json数据 */ function upload($file, $multipart = true) { $cookie = ''; // 微博cookie $url = 'http://picupload.service.weibo.com/interface/pic_upload.php' .'?mime=image%2Fjpeg&data=base64&url=0&markpos=1&logo=&nick=0&marks=1&app=miniblog'; if($multipart) { $url .= '&cb=http://weibo.com/aj/static/upimgback.html?_wv=5&callback=STK_ijax_'.time(); if (class_exists('CURLFile')) { // php 5.5 $post['pic1'] = new CURLFile(realpath($file)); } else { $post['pic1'] = '@'.realpath($file); } } else { $post['b64_data'] = base64_encode(file_get_contents($file)); } // Curl提交 $ch = curl_init($url); curl_setopt_array($ch, array( CURLOPT_POST => true, CURLOPT_VERBOSE => true, CURLOPT_RETURNTRANSFER => true, CURLOPT_HTTPHEADER => array("Cookie: $cookie"), CURLOPT_POSTFIELDS => $post, )); $output = curl_exec($ch); curl_close($ch); // 正则表达式提取返回结果中的json数据 preg_match('/({.*)/i', $output, $match); if(!isset($match[1])) return ''; return $match[1]; }
如你所见,代码中包含两种上传方式,一种是 multipart 方式上传,通过这种方式上传效率较高,局限性是只能上传服务器的本地图片。
另一种方式是base64方式,可以上传远程图片。(文字的表述可能不清楚,用代码就明白了 )
upload('mypic.jpg', true); // multipart方式上传
upload('https://www.ynceyi.com/d/file/img/mypic.jpg', false); // 非multipart方式(base64)上传
无论通过以上哪种方式,返回的都会是一段如下所示的json数据。
{"code":"A20001","data":{"count":2,"data":"eyJ1aWQiOjIyODczOTc1NzUsImFwcCI6Im1pbmlibG9nIiwiY291bnQiOjIsInRpbWUiOjE0OTM3ODE0NzQuMzQ5LCJwaWNzIjp7InBpY18yIjp7InJldCI6LTIsIm5hbWUiOiJwaWNfMiJ9LCJwaWNfMSI6eyJ3aWR0aCI6MzAwLCJzaXplIjo4Njk5LCJyZXQiOjEsImhlaWdodCI6MzAwLCJuYW1lIjoicGljXzEiLCJwaWQiOiI4ODU2ZWFjN2d5MWZmODA4NGhtanpqMjA4YzA4YzN5ZSJ9fX0=","pics":{"pic_2":{"ret":-2,"name":"pic_2"},"pic_1":{"width":300,"size":8699,"ret":1,"height":300,"name":"pic_1","pid":"8856eac7gy1ff8084hmjzj208c08c3ye"}}}}
其它的不用管,我们只需要 pic_1 这个子节点中的数据
如图所示,pid码就是忙活了这么久之后最终要用到的。
http://ww2.sinaimg.cn/large/ + 获取到的pid 就能组合成图片的外链。
比如上面示范的那段数据中包含的图片外链地址就是
http://ww2.sinaimg.cn/large/8856eac7gy1ff8084hmjzj208c08c3ye
其实 pid 中还包含了更多“秘密”,这里提供一下 pid 的终极解码算法 :
/** * 获取图片链接(本函数修改自 https://github.com/consatan/weibo_image_uploader) * * @param string $pid 微博图床pid,或者微博图床链接。传递的是链接的话, * 仅是将链接的尺寸更改为目标尺寸而已。 * @param string $size 图片尺寸 * @param bool $https (true) 是否使用 https 协议 * @return string 图片链接 * 当 $pid 既不是 pid 也不是合法的微博图床链接时返回空值 */ function getImageUrl($pid, $size = 0, $https = true) { $sizeArr = array('large', 'mw1024', 'mw690', 'bmiddle', 'small', 'thumb180', 'thumbnail', 'square'); $pid = trim($pid); $size = $sizeArr[$size]; // 传递 pid if (preg_match('/^[a-zA-Z0-9]{32}$/', $pid) === 1) { return ($https ? 'https' : 'http') . '://' . ($https ? 'ws' : 'ww') . ((crc32($pid) & 3) + 1) . ".sinaimg.cn/" . $size . "/$pid." . ($pid[21] === 'g' ? 'gif' : 'jpg'); } // 传递 url $url = $pid; $imgUrl = preg_replace_callback('/^(https?://[a-z]{2}d.sinaimg.cn/)' . '(large|bmiddle|mw1024|mw690|small|square|thumb180|thumbnail)' . '(/[a-z0-9]{32}.(jpg|gif))$/i', function ($match) use ($size) { return $match[1] . $size . $match[3]; }, $url, -1, $count); if ($count === 0) { return ''; } return $imgUrl; }
本文转载自:精准像素:11px.cn ,感谢作者分享实用知识