使用Maps API计算两个地址的距离,然后将每个距离保存到数据库

时间:2018-02-22 16:03:45

标签: ruby-on-rails google-maps-api-3 open-uri

我正在根据距离将儿童与幼儿园相匹配的应用程序。

我正在尝试使用Maps API来计算多个地址之间的距离。 我想使用open-uri

https://maps.googleapis.com/maps/api/distancematrix/json?origins=[address of child]&destinations=[adress of kindergarden]&key=XYZ

这将是对api的请求。

我的想法是在孩子的创建过程之后发送请求。 我会从数据库中获取子地址和幼儿园地址并构建请求 响应的距离将保存在每个孩子和幼儿园的关系表中 如何访问孩子和幼儿园的数据库条目? 然后将其保存到关系表中?

我目前的代码如下:

    def create
@child = Child.new child_params
@child.user = current_user
@child.save
origin = @child.adress
destination = @kiga.adress
response = open('https://maps.googleapis.com/maps/api/distancematrix/json?origins=' + [address of child] + '&destinations=' + [adress of kindergarden] + '&key=XYZ').read
relations.distance = response.rows.elements[0]

respond_to do |format|
  if @child.save
    format.html { redirect_to @child, notice: 'Child was successfully created.' }
    format.json { render :show, status: :created, location: @child }
  else
    format.html { render :new }
    format.json { render json: @child.errors, status: :unprocessable_entity }
  end
end

帮助将不胜感激?

2 个答案:

答案 0 :(得分:2)

这里的问题是您尝试将URL作为字符串打开,因此它正在寻找以这种方式命名的文件并且它不存在。

这里有两个选项:

1。使用HTTParty gem发出HTTP请求

这是我最喜欢的方法。

在您的Gemfile中加入宝石:

gem 'HTTParty'

在您的控制器中发出请求:

url = "https://maps.googleapis.com/maps/api/distancematrix/json?origins=" \
      "#{@child.adress}&destinations=#{@kiga.adress}&key=XYZ"
response = HTTParty.get(url)

现在,如果您打印出response["rows"].first["elements"].first的结果,您会看到如下输出:

{"distance"=>{"text"=>"15.4 km", "value"=>15405}, "duration"=>{"text"=>"17 mins", "value"=>1001}, "status"=>"OK"}

正如您所看到的,您有距离和持续时间信息。我猜你是在以米为单位的距离值之后:

relations.distance = response["rows"].first["elements"].first["distance"]["value"]

注意1 :我故意在字地址(@ child.adress,@ kiga.adress)中复制拼写错误,以便代码适用于您的设置,但请考虑将其修复为@ child.address和@ kiga.address。

注意2 :为简单起见,我没有对响应进行任何错误检查,这是您应该注意的事项。

注意3 :请记住将api密钥更改为有效密钥,我将其简化为XYZ,就像您在问题中所做的那样。

2。使用open-uri将您的url解析为有效的URI

您必须在控制器的开头要求库'open-uri''JSON'

require 'open-uri'
require 'JSON'

    def create
    @child = Child.new child_params
    @child.user = current_user
    @child.save
    origin = @child.adress
    destination = @kiga.adress

    # Parse the string url into a valid URI
    url = URI.parse(
        "https://maps.googleapis.com/maps/api/distancematrix/json?origins=" \
        "#{@child.adress}&destinations=#{@kiga.adress}&key=XYZ"
    )
    # Open the URI just like you were doing
    response = open(url).read
    # Parse the string response in JSON format
    result = JSON.parse(response)

    # Extract the distance value in meters
    relations.distance = result["rows"].first["elements"].first["distance"]["value"]

    respond_to do |format|
        if @child.save
            format.html { redirect_to @child, notice: 'Child was successfully created.' }
            format.json { render :show, status: :created, location: @child }
        else
            format.html { render :new }
            format.json { render json: @child.errors, status: :unprocessable_entity }
    end
end

变量result的内容(解析后)如下所示:

{"destination_addresses"=>["Destination address"], "origin_addresses"=>["Origin address"], "rows"=>[{"elements"=>[{"distance"=>{"text"=>"15.4 km", "value"=>15405}, "duration"=>{"text"=>"17 mins", "value"=>1001}, "status"=>"OK"}]}], "status"=>"OK"}

如果您使用HTTParty方式或open-uri + JSON解析方式,请确保检查响应状态代码。这两种方法都在我的计算机上进行了本地测试并取得了成功

我希望它有所帮助,欢呼!

答案 1 :(得分:0)

另一种解决方案是看看这个宝石 https://github.com/alexreisner/geocoder

如果您对您的儿童模型和您的Kiga模型进行地理编码,那么您可以轻松地从一个模型到另一个模型。

enter code here

这是在没有进行任何API调用的情况下,GEM会为您执行此操作。

它还有一个非常方便的方法来返回位置附近的记录

@kiga.near(@child.coordinates, 10)

将以10km的@child坐标返回所有Kiga