`
喵喵大神
  • 浏览: 37397 次
文章分类
社区版块
存档分类
最新评论

通过聚合数据API实现快递数据查询-短信验证码

阅读更多

有位朋友让我给他新开的网站帮忙做几个小功能,如下:

  1. 输入快递公司、快递单号,查询出这个快件的所有动态(从哪里出发,到了哪里)
  2. 在注册、登录等场景下的手机验证码(要求有一定的防刷策略)
  3. 通过输入公司名的关键词,查看这个公司是否已经注册、法人信息、有类似名称的公司等等

并且可以用的接口、文档都提供给我了。
其中需求 1、2,都通过 聚合数据 这家网站提供的接口实现;需求 3 通过 云聚数据 来实现。

本项目的文件

因为朋友的网站是用 ThinkPHP 写的,为了保持将来代码的兼容,这三个功能也用 ThinkPHP 写成。

项目的所有文件都放在了 GitHub 上,部分敏感数据已经隐藏,你需要自行替换,地址如下:

GitHub 地址:使用聚合数据API查询快递数据-短信验证码-企业核名

因为这三个功能并不是正式产品,将来会需要嵌入到网站的各个功能模块中去,所以为了查看起来方便,三个功能的代码都写在一个文件里,你只要重点关注以下几个文件就好:

  • /Home/Conf/config.php 参数配置文件
  • /Home/Controller/IndexController.class.php 后端代码、API的请求与处理
  • /Home/View/Index_index.html 前端 html

申请 KEY 和阅读开发文档

分别到上面两家网站上找到“快递”、“短信”、“核名”的文档地址,根据里面的说明,把 KEY、URL 等信息放入配置文件Home/Conf/config.php,方便后面重复使用。

常用快递API文档

短信API文档

核名-文档

注意 短信的 API 服务,要先到网站的后台添加“短信模板”,让管理员审核后才可以正常调用

最后,Home/Conf/config.php 配置文件里的内容如下

 
return array(
    // 快递查询
    'EXPRESS_APP_KEY' => '你的快递 APPKEY',
    'EXPRESS_QUERY_URL' => 'http://v.juhe.cn/exp/index', //快递单号查询
    'EXPRESS_COM_URL' => 'http://v.juhe.cn/exp/com', //快递公司查询
    // 发短信
    'SEND_SMS_KEY' => '你的短信接口 APPKEY',
    'SEND_SMS_URL' => 'http://v.juhe.cn/sms/send',
    // 核名
    'COMPANY_KEY' => '核名 APPKEY',
    'COMPANY_URL' => 'http://eci.yjapi.com/ECIFast/Search',
    //数据库配置信息
    'DB_TYPE'   => 'mysql', // 数据库类型
    'DB_HOST'   => 'localhost', // 服务器地址
    'DB_NAME'   => '你的数据库名', // 数据库名
    'DB_USER'   => '你的数据库用户名', // 用户名
    'DB_PWD'    => '密码', // 密码
    'DB_PORT'   => 3306, // 端口
    'DB_PREFIX' => 'pre_', // 数据库表前缀 
    'DB_CHARSET'=> 'utf8', // 字符集
);
 

设置数据库

为了防止恶意用户利用暴露在外的短信接口捣乱,需要对每个手机号码的行为进行记录。例如:

  • 向某个手机号码发送短信验证码后, 60 秒内不能再次发送
  • 必须在 1 小时内填写短信验证码,否则会过期
  • 避免 ajax 请求地址被机器人程序利用,在表单里要使用图片验证码才能提交数据

关于“表单验证码”我们后面代码会说明,这里先创建表结构如下,用于记录短信发送记录:

 
CREATE TABLE `pre_smsrecord` (
  `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
  `mobile` varchar(11) NOT NULL DEFAULT '',
  `tpl_id` int(11) NOT NULL,
  `code` int(6) NOT NULL,
  `time` int(11) NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=M
 
  • mobile 是手机号码
  • tpl_id是在网站后台添加并通过审核的短信模板
  • code是发送的验证码(一般是4位或6位)
  • time是发送时间戳

直接下载sql进行还原:在本项目的 GitHub 地址上也可以直接从 /Pubic 目录找到 sql 文件,你可以直接把它还原你的 MySQL 上。

表单验证码的实现

最终效果如下:

验证码

因为三个功能都需要表单验证码,所以首先实现它。

打开 /Home/View/Index_index.html,注意里面图片验证码 img 标签,以及对应的 javascript

 
<p>
    (通用)输入验证码 <input type="text" name="verify" id="verify">
    <img id="verify-img" src="/?m=Home&c=Index&a=verify" alt="">
    <a id="btn-refresh-verify" href="javascript:;" title="">刷新</a>
</p>
<script type="text/javascript" charset="utf-8">
jQuery(document).ready(function($) {
    // 刷新验证码按钮
    $("#btn-refresh-verify").click(function(event) {
    
        $("#verify-img").attr('src', '/?m=Home&c=Index&a=verify' + "&time=" + new Date().getTime());
    });
});
</script>
 

对应的后端代码在 /Home/Controller/IndexController.class.php 中

 
/**
* +--------------------------------------------------------------------------
* 生成验证码
*
* +--------------------------------------------------------------------------
*/
public function verify(){
    $Verify = new \Think\Verify();
    $Verify->entry();
}
/**
* +--------------------------------------------------------------------------
* 检查验证码
*
* @param string $code 输入的验证码
* @return boolean
* +--------------------------------------------------------------------------
*/
protected function check_verify($code){
    $verify = new \Think\Verify();
    return $verify->check($code);
}
 

通用的 Curl 方法,用来向第三方网站的 API 发起请求并获取返回值

因为 3 个功能实际上都是通过网络来请求一个第三方网站的 API 接口地址,因此可以统一成一个通用的方法。代码如下,可以传入三个变量,分别为 :url、参数数组、请求方式(是否是post,默认为0),返回一个 json 格式的数据。

复制代码
/**
* +--------------------------------------------------------------------------
* 通用的“聚合数据”请求接口,返回JSON数据
*
* @param string $url 接口地址
* @param array $params 传递的参数
* @param int $ispost 是否以POST提交,默认GET
* @return json
* +--------------------------------------------------------------------------
*/
public function juhecurl($url,$params=false,$ispost=0){
    $httpInfo = array();
    $ch = curl_init();
    curl_setopt( $ch, CURLOPT_HTTP_VERSION , CURL_HTTP_VERSION_1_1 );
    curl_setopt( $ch, CURLOPT_USERAGENT , 'Mozilla/5.0 (Windows NT 5.1) AppleWebKit/537.22 (KHTML, like Gecko) Chrome/25.0.1364.172 Safari/537.22' );
    curl_setopt( $ch, CURLOPT_CONNECTTIMEOUT , 30 );
    curl_setopt( $ch, CURLOPT_TIMEOUT , 30);
    curl_setopt( $ch, CURLOPT_RETURNTRANSFER , true );
    if( $ispost )
    {
        curl_setopt( $ch , CURLOPT_POST , true );
        curl_setopt( $ch , CURLOPT_POSTFIELDS , $params );
        curl_setopt( $ch , CURLOPT_URL , $url );
    }
    else
    {
        if($params){
            curl_setopt( $ch , CURLOPT_URL , $url.'?'.$params );
        }else{
            curl_setopt( $ch , CURLOPT_URL , $url);
        }
    }
    $response = curl_exec( $ch );
    if ($response === FALSE) {
          //echo "cURL Error: " . curl_error($ch);
        return false;
    }
    $httpCode = curl_getinfo( $ch , CURLINFO_HTTP_CODE );
    $httpInfo = array_merge( $httpInfo , curl_getinfo( $ch ) );
    curl_close( $ch );
    return $response;
}
 

后面我们获取快递数据、发送短信、查询企业名称,都可以调用这个通用的方法。

获取快递数据列表的实现

最终效果如下:

快递查询

打开 /Home/View/Index_index.html

 
<h1>获取快递数据</h1>
<div id="express-module">
    <p>
        选择公司
        <select name="express-company" id="express-company">
            <option value="sf">顺丰</option>
            <option value="sto">申通</option>
            <option value="yt">圆通</option>
            <option value="yd">韵达</option>
            <option value="tt">天天</option>
            <option value="ems">EMS</option>
            <option value="zto">中通</option>
            <option value="ht">汇通</option>
        </select>
    </p>
    <p> 
        输入单号 <input type="text" name="express-number" id="express-number"> 
    </p>
    <p>
        <button id="btn-query-express" type="button" class="btn btn-default">查询快递</button> 
    </p>
    
    <p>返回结果:</p>
    <p id="reason" style="color:#FF0000"></p>
    
    <p>快件动态:</p>
    <ul id="express-result">
        
    </ul>
</div>    
<!-- 引入jquery库 -->
<script src="__PUBLIC__/jquery.min.js" type="text/javascript"></script>
<script type="text/javascript" charset="utf-8">
jQuery(document).ready(function($) {
    //点击快递查询按钮
    $("#btn-query-express").click(function(event) {
        $("#reason").html("");
        // 更新验证码
        $("#verify-img").attr('src', '/?m=Home&c=Index&a=verify' + "&time=" + new Date().getTime());
        $.getJSON(
            '/?m=Home&c=Index&a=getExpressData', 
            {
                company: $("#express-company").val(),
                number: $("#express-number").val(),
                verify: $("#verify").val(),
            }, 
            function(json, textStatus) {
                if(json['resultcode'] == 200){
                    var result_list = json['result']['list'];
                    $("#express-result").html("");
                    for(var i = 0, l = result_list.length; i < l; i++) {
                        $("#express-result").append("<li>" + result_list[i]['datetime'] + "," + result_list[i]['remark'] + "," +result_list[i]['zone'] + "</li>");
                    }
                }
                $("#reason").html(json['reason']);
        });
    });
});
</script>
 

对应的后端代码为如下,特别注意,你要把 $content = $this->juhecurl(C("EXPRESS_QUERY_URL"), $params, 1); 这段的注释去掉(因为我开发的时候查询余额不足,所以使用了一个写死的数组来让程序能正常运行)

 
/** 
* +--------------------------------------------------------------------------
* 获取快递数据
*
* @param string $get.company 快递公司代码
* @param string $get.number 快递单号
* @return json
* +--------------------------------------------------------------------------
*/
public function getExpressData(){
    // 传入 get 参数,包括公司代号、快递单号、验证码
    $com    = I("get.company");
    $no     = I("get.number");
    $verify = I("get.verify");
    // 处理验证码
    if ( !$this->check_verify($verify) ) {
        $content = array('resultcode'=>1000, 'reason'=>'验证码填写错误');
        echo json_encode($content);
        exit();
    }
    // 处理机器人程序刷接口(目前通过IP判断)
    $ip = get_client_ip(0, true);
    $Record = M("expressrecord");
    $express_record = $Record->where("ip='" . $ip . "'")->find();
    if( $express_record && ( (time() - $express_record['time']) < 60 ) ){
        echo json_encode(array('reason'=>'60秒内只能查询一次'));
        exit();
    }
    if ( $com && $no ) {
        $params = array(
            'key' => C("EXPRESS_APP_KEY"),
            'com'  => $com,
            'no' => $no
        );
        // 开发测试阶段直接返回值,不请求 API
        // $content = $this->juhecurl(C("EXPRESS_QUERY_URL"), $params, 1);
        $content = array('resultcode'=>200, 'reason'=>'查询成功', 'result'=>array('list'=>array()));
        // 删除旧记录(如果有),然后添加新的记录
        $Record->where("ip='" . $ip . "'")->delete();
        $data = array(
            'ip' => $ip,
            'time'=>time()
        );
        $Record->add($data);
        //$returnArray = json_decode($content,true);
        echo json_encode($content, true);
    }
}
 

短信验证码的发送和检验

短信验证码

废话不多说,前端html直接上代码。这里实际上有两个动作,一个是“给我手机号码 xxxxx 发个验证码”,另一个是“我已经收到了,并填写了,请看我填写的验证码对不对”。

 
<h1>发送短信验证码与检查</h1>
<div id="sms-module">
    <p>
        短信模板:
        <select name="send-sms-tplid" id="send-sms-tplid">
            <option value="5596">申请注册</option>
            <option value="5602">申请找回密码</option>
            <option value="5603">在线核名</option>
        </select>
    </p>
    
    <p>
        手机号码:<input type="text" name="send-sms-mobile" id="send-sms-mobile">
    </p>
    
    <p>
        输入手机验证码:<input type="text" name="send-sms-code" id="send-sms-code">
        <span id="sms-count-down"></span>
        <button id="btn-send-sms" type="button" class="btn btn-default">发送验证码</button>
    </p>
    
    <p>
        <button id="btn-check-sms" type="button" class="btn btn-default">提交手机验证码</button>
    </p>
</div>
<script type="text/javascript" charset="utf-8">
/**
* 发送短信验证码后,60秒倒计时
*/
var seconds_left = 60;
function sms_count_down(){
    if(seconds_left > 0){
        seconds_left = seconds_left-1;
        //var b=new Date();
        
        document.getElementById("sms-count-down").innerHTML = seconds_left + "";
        setTimeout("sms_count_down()",1000);
    } else {
        //根据 http://stackoverflow.com/questions/7526601/setattributedisabled-false-changes-editable-attribute-to-false
        // 不能为 setAttribute 设置任何值,都会变成 ”disabled“,要使用 removeAttribute
        document.getElementById("btn-send-sms").removeAttribute("disabled");
        document.getElementById("btn-send-sms").innerHTML = "重新发送";
        document.getElementById("sms-count-down").innerHTML = "";
    }
}
jQuery(document).ready(function($) {
//发送短信
    $("#btn-send-sms").click(function(event) {
        $("#reason").html("");
        $.getJSON(
            '/?m=Home&c=Index&a=sendSMS', 
            {
                tplid: $("#send-sms-tplid").val(),
                mobile: $("#send-sms-mobile").val(),
            }, 
            function(json, textStatus) {
                $("#reason").html(json['reason']);
                var error_code = json['error_code'];
                if(error_code == 0){
                    // 测试阶段,不禁用,可检查 60 秒重复发送
                    //$("#btn-send-sms").attr("disabled", true);
                    //开始倒计时
                    sms_count_down();
                }else{
                    $("#btn-send-sms").html("重新发送");
                }
    
        });
    });
    //检查短信验证码
    $("#btn-check-sms").click(function(event) {
        $("#reason").html("");
        // 更新验证码
        $("#verify-img").attr('src', '/?m=Home&c=Index&a=verify' + "&time=" + new Date().getTime());
        $.getJSON(
            '/?m=Home&c=Index&a=checkSmsCode', 
            {
                tplid: $("#send-sms-tplid").val(),
                mobile: $("#send-sms-mobile").val(),
                code: $("#send-sms-code").val(),
                verify: $("#verify").val(),
            }, 
            function(json, textStatus) {
                $("#reason").html(json['reason']);
    
        });
    });
    
});
 

同样因为有两个动作,后端会稍微复杂一点点

 
/**
* +--------------------------------------------------------------------------
* 请求发送短信接口,60秒后才能重新发送
*
* @param int $get.tplid 短信模板id
* @param string $get.mobile 手机号码
* @return json
* +--------------------------------------------------------------------------
*/
public function sendSMS(){
    $tpl_id = I("get.tplid"); // 短信模板id:注册 5596 找回密码 5602 在线核名 5603
    $mobile = I("get.mobile"); // 手机号码
    // 检查数据库记录 ,是否在 60 秒内已经发送过一次
    $Record = M("smsrecord");
    $where = array(
        'mobile' => $mobile,
        'tpl_id' => $tpl_id,
    );
    $sms_record = $Record->where($where)->find();
    if( $sms_record && ( (time() - $sms_record['time']) < 60 ) ){
        echo json_encode(array('reason'=>'60秒内不能多次发送'));
        exit();
    }
    // 如果60秒内没有发过,则发送验证码短信(6位随机数字)
    $code = mt_rand(100000, 999999);
    $smsConf = array(
        'key'   => C("SEND_SMS_KEY"), //您申请的APPKEY
        'mobile'    => $mobile, //接受短信的用户手机号码
        'tpl_id'    => $tpl_id, //您申请的短信模板ID,根据实际情况修改
        'tpl_value' =>'#code#=' . $code //您设置的模板变量,根据实际情况修改 '#code#=1234&#company#=聚合数据'
    );
     
    //测试阶段,不发短信,直接设置一个“发送成功” json 字符串
    $content = $this->juhecurl(C("SEND_SMS_URL") ,$smsConf, 1); //请求发送短信
    //$content = json_encode(array('error_code'=>0, 'reason'=>'发送成功'));
    if($content){
        $result = json_decode($content,true);
        $error_code = $result['error_code'];
        if($error_code == 0){
            // 状态为0,说明短信发送成功
            // 数据库存储发送记录,用于处理倒计时和输入验证,首先要删除旧记录
            $Record->where("mobile=" . $mobile)->delete();
                $data = array(
                    'mobile' => $mobile,
                    'tpl_id'=> $tpl_id,
                    'code'=>$code,
                    'time'=>time()
                );
                $Record->data($data)->add();
            //echo "短信发送成功,短信ID:".$result['result']['sid'];
        }else{
            //状态非0,说明失败
            //echo "短信发送失败(".$error_code."):".$msg;
        }
    }else{
        //返回内容异常,以下可根据业务逻辑自行修改
        //$result['reason'] = '短信发送失败';
    }
    echo $content;
}
/**
* +--------------------------------------------------------------------------
* 检查填写的手机验证码是否填写正确
* 可以添加更多字段改造成注册、登录等表单
*
* @param string $get.verify 验证码
* @param string $get.mobile 手机号码
* @param int $get.tplid 短信模板ID
* @param int $get.code 手机接收到的验证码
* +--------------------------------------------------------------------------
*/
public function checkSmsCode(){
    $verify = I("get.verify");
    $tpl_id = I("get.tplid"); // 短信模板id:注册 5596 找回密码 5602 在线核名 5603
    $mobile = I("get.mobile"); // 手机号码
    $code = I("get.code"); // 手机收到的验证码
    if(!$this->check_verify($verify)){
        $content = array('resultcode'=>1000, 'reason'=>'验证码填写错误');
        echo json_encode($content);
        exit();
    }
    // 检查数据库记录,输入的手机验证码是否和之前通过短信 API 发送到手机的一致
    $Record = M("smsrecord");
    $where = array(
        'mobile' => $mobile,
        'tpl_id' => $tpl_id,
        'code' => $code,
    );
    $sms_record = $Record->where($where)->find();
    if($sms_record){
        echo json_encode(array('reason'=>'短信验证码核对成功'));
        // 处理后面的程序(如继续登录、注册等)
    }else{
        echo json_encode(array('reason'=>'短信验证码错误'));
    }
}
 

最后是企业核名

因为这个接口本来只提供了每次查询一个关键词,而朋友的网站要求能每次查询用“,”分隔的多个关键词,因此需要对提交的数据和返回到前端的进行一番处理(循环处理多个列表)。

企业核名

前端代码比较简单

复制代码
<h1>企业核名</h1>
<div id="company-module">
    <p>
        省份:上海
        <input type="hidden" name="company-province" id="company-province" value="SH">
    </p>
    <p>
        <input type="text" name="company-name" id="company-name">
    </p>
</div>
<button id="btn-company" type="button" class="btn btn-default">企业核名</button>
<p>公司查询列表:</p>
<ul id="company-result">
    
</ul>
<javascript 略……
//企业核名(可输入多个以逗号分隔的名称来多次查询)
$("#btn-company").click(function(event) {
    $("#reason").html("");
    $("#company-result").html("");
    // 更新验证码
    $("#verify-img").attr('src', '/?m=Home&c=Index&a=verify' + "&time=" + new Date().getTime());
    $.getJSON(
        '/?m=Home&c=Index&a=getCompanyName', 
        {
            province: $("#company-province").val(),
            companyName: $("#company-name").val(),
            verify: $("#verify").val(),
        }, 
        function(json, textStatus) {
            if(json['reason']){
                $("#reason").html(json['reason']);
            } else {
                //console.log(json);
                var json_list = JSON.parse(json);
                if (json_list.length > 0){
                    for(i in json_list){
                        var result_list = json_list[i].Result;
                        if (result_list.length > 0){
                            for(j in result_list){
                                //console.log(result_list[j]);
                                $("#company-result").append("<li> 名称:" + result_list[j].Name + ", 法人:" + result_list[j].OperName + ", 状态:" + result_list[j].Status + "</li>");
                            }
                        }
                    }
                }
            }
    });
});
</javascript>
 

后端代码

 
/** 
* +--------------------------------------------------------------------------
* 企业核名
* 该接口 http://eci.yjapi.com/ECIFast/Search 仅支持 GET 方式,因此 $param 为字符串形式
*
* @param string $get.province 省份
* @param string $get.companyName 公司名
* @return json
* +--------------------------------------------------------------------------
*/
public function getCompanyName(){
    // 传入 get 参数
    $province                = I("get.province");
    $companyName    = I("get.companyName");
    $verify         = I("get.verify");
    // 处理验证码
    if ( !$this->check_verify($verify) ) {
        $content = array('resultcode'=>1000, 'reason'=>'验证码填写错误');
        echo json_encode($content);
        exit();
    }
    // 处理机器人程序刷接口(目前通过IP判断)
    $ip = get_client_ip(0, true);
    $Record = M("companyrecord");
    $company_record = $Record->where("ip='" . $ip . "'")->find();
    // if( $company_record && ( (time() - $company_record['time']) < 60 ) ){
    //     echo json_encode(array('reason'=>'60秒内只能查询一次企业名称'));
    //     exit();
    // }
    $resultArray = array();
    if ( $province && $companyName ) {
        // 删除旧记录(如果有),然后添加新的记录
        $Record->where("ip='" . $ip . "'")->delete();
        // 循环处理多个公司名
        $companies = explode(",", $companyName);
        foreach ($companies as $key => $value) {
            $params = "key=" . C("COMPANY_KEY") . "&province={$province}&companyName={$value}";
            // 开发测试阶段直接返回值,不请求 API
            $content = $this->juhecurl(C("COMPANY_URL"), $params, 0);
            $returnArray = json_decode($content,true);
            // 循环插入新的结果
            $resultArray[] = $returnArray;
            //$content = array('resultcode'=>200, 'reason'=>'查询成功', 'result'=>array('list'=>array()));
        }
        $data = array(
            'ip' => $ip,
            'time'=>time()
        );
        $Record->add($data);
        $content = json_encode($resultArray, true);
        echo json_encode($content, true);
    }
}
 

结束

至此三个功能都已经完成了,但是在防止恶意用户利用接口干坏事的策略上,还可以更进一步。例如

  • 必须要注册用户才能发送请求
  • 手机号码每天只能限制发送有限的验证码(因为目前即使每分钟发送一次,一天下来也比较可观了)
  • 手机号码注册后,再次请求发送验证码时,只能发给“已有”的手机号码数据表里的手机号码,而不能把“找回密码”发给一个还没注册的人。
分享到:
评论

相关推荐

    基于聚合数据API的新闻app

    【基于聚合数据API的新闻app】是一个利用API接口获取新闻数据并进行展示的移动应用程序,主要针对Android平台设计。在这款应用中,开发者利用了聚合数据API提供的服务,该服务通常会提供各种实时更新的新闻资讯,...

    微信小程序学习demo:笑话大全;使用聚合数据API

    这个小程序通过调用聚合数据API来获取笑话资源。聚合数据是一个开放的数据服务平台,提供各种API接口,包括新闻、天气、笑话等各类实用信息。在小程序中,开发者通常会使用网络请求模块(wx.request)来与服务器进行...

    聚合数据 Android 项目开发实战:

    这个实战案例将聚焦于如何利用聚合数据的API来实现短信验证码的发送与验证。 首先,我们需要了解聚合数据提供的短信验证码服务。聚合数据是一家提供多种API服务的公司,包括地理位置、天气、短信验证等。在短信...

    违章查询完成工程---聚合数据

    5. 违章查询:这个应用的核心功能是查询用户的车辆违章记录,通过调用聚合数据的违章查询API,获取并显示违章的详细信息,如违章时间、地点、罚款金额等。 6. Android Studio:作为Google官方推荐的Android开发环境...

    一个使用vue20开发聚合api数据的简易demo

    这个简易DEMO可以帮助初学者理解如何使用Vue 2.0与API进行交互,实现数据的动态加载和显示。通过学习和分析这个项目,你可以深入理解Vue.js的核心概念,为进一步学习和开发Vue应用打下坚实的基础。

    spring Boot + mybatis +maven,spring boot 公众号开发,有接入聚合数据API

    spring boot 微信公众号开发 demo ,maven 引入jar spring Boot + mybatis 聚合数据API接入 搞笑段子 新闻头条

    微服务API聚合网关 An Aggregation API Gateway-fizz-gateway-node.zip

    微服务API聚合网关,通常被称为API Gateway,是微服务架构中的一个重要组件,它作为客户端与后端微服务之间的统一入口。在Fizz-Gateway-Node项目中,我们看到的是一个基于Node.js实现的API Gateway。这个网关的主要...

    短信验证码

    聚合数据提供了API接口,开发者可以通过调用这些接口来发送和验证短信验证码。首先,你需要在聚合数据官网注册账号,并创建应用获取API密钥,这是后续调用接口的基础。 在Android项目中,我们需要创建一个能够发送...

    ArcGIS APi结合天地图聚合效果的数据

    在本文中,我们将深入探讨如何使用ArcGIS API for JavaScript(简称ArcGIS API)与天地图进行集成,以实现散点聚合效果。ArcGIS API是一款强大的Web GIS开发工具,它允许开发者在网页上创建交互式的地图应用。而天...

    股票查询---OKhttp StockSearch.zip

    "利用聚合数据 实时在线股票查询"表明,项目的数据来源是聚合数据,这是一家提供各种API服务的数据供应商,其中包括股票市场数据。通过聚合数据的API,开发者可以获取到实时的股票报价、交易信息等。 【标签】...

    arcgis api for javascript 4.x 聚合

    本文将深入探讨4.x版本的ArcGIS API 在三维场景中的点聚合功能,以及如何在SceneView和MapView中实现这一功能。 首先,我们要理解“聚合”在GIS中的含义。在地图上,当我们有大量分散的点数据时,为了提高视觉效果...

    Java基于聚合网API做的很基础的手机号归属地查询

    在本项目中,我们主要探讨的是如何利用Java编程语言与聚合网API进行集成,实现一个基础的手机号码归属地查询功能。聚合网提供了一系列的开放接口,开发者可以通过这些接口获取到丰富的数据信息,包括但不限于手机...

    02.hive查询语法--分组聚合--groupby查询--where过滤和having过滤的区别.mp4

    02.hive查询语法--分组聚合--groupby查询--where过滤和having过滤的区别.mp4

    基于PHP的聚合数据天气预报api调用示例.pdf

    聚合数据(Juhe Data)提供了一个天气预报API服务。通过该API,开发者可以获取到包括但不限于城市天气、IP定位天气等信息。调用API需要一个有效的appkey来进行认证。 #### 3. API调用示例 文档提供了一个PHP示例...

    arcgis api for javascript 聚合(3.20)直接运行

    在ArcGIS API中,这一功能主要通过`esri/geometry/GeometryEngine`模块的`cluster`方法实现,它可以对几何对象数组进行聚类,生成新的聚合几何对象。 二、聚合函数的使用 在3.20版本中,`cluster`方法接收两个参数...

    百度地图点聚合相关代码

    通过分析和实践“BaiduMapsApiASDemo”中的代码,开发者可以掌握百度地图API的点聚合功能,并将其应用于自己的项目中,提升地图应用的数据展示能力。同时,社区交流也是学习过程中不可或缺的一部分,有问题可以去...

    20180913-聚合支付API在线文档1

    超赢聚合支付API是用于集成多种支付渠道的接口,通过这个接口,开发者可以实现与多个支付平台的无缝对接,为用户提供便捷的支付体验。本文档主要涵盖以下几个核心知识点: 1. **协议规则**: - 为了保证交易的安全...

    Prometheus数据API导出Python脚本(export Prometheus metric data by http api)

    对于更复杂的查询,比如聚合操作(如平均值、最大值、最小值等)或时间序列的组合,可以在`query`参数中使用Prometheus的查询语言(PromQL)。例如,计算过去1小时CPU使用率的平均值: ```python query = 'avg_over...

    jdo2-api-2.3-ec hive hdfs 所需jar

    Hive通过JDO2-API-2.3-EC与Metastore服务进行通信,将SQL语句转换为Hadoop MapReduce任务,从而实现了对大数据集的高效查询。JDO的使用简化了Hive的开发和维护,使得数据存储和查询的逻辑更加清晰。 三、HDFS与JDO...

Global site tag (gtag.js) - Google Analytics