Geofire - 未找到位置

Posted

技术标签:

【中文标题】Geofire - 未找到位置【英文标题】:Geofire - not found locations 【发布时间】:2019-02-14 09:27:55 【问题描述】:

如何使用Geo Queries 获取靠近通过的GeoLocation(double lat, double lng) 的所有位置。 我有以下代码(它什么也没发生):

public void setCurrentLatLng(double lat, double lng)
    this.lat = lat;
    this.lng = lng;
    GeoLocation geoLocation = new GeoLocation(lat, lng);
    updateCurrenLocation(geoLocation);
    GeoQuery geoQuery = geoFire.queryAtLocation(geoLocation, 8f);
    geoQuery.addGeoQueryDataEventListener(new GeoQueryDataEventListener() 

        @Override
        public void onDataEntered(DataSnapshot dataSnapshot, GeoLocation location) 
            Log.d("geoQuery","onDataEntered "+dataSnapshot.toString());
            // ...
        

        @Override
        public void onDataExited(DataSnapshot dataSnapshot) 
            Log.d("geoQuery","onDataExited "+dataSnapshot.toString());
            // ...
        

        @Override
        public void onDataMoved(DataSnapshot dataSnapshot, GeoLocation location) 
            Log.d("geoQuery","onDataMoved "+dataSnapshot.toString());
            // ...
        

        @Override
        public void onDataChanged(DataSnapshot dataSnapshot, GeoLocation location) 
            Log.d("geoQuery","onDataChanged "+dataSnapshot.toString());
            // ...
        

        @Override
        public void onGeoQueryReady() 
            // ...
            Log.d("geoQuery","onGeoQueryReady");
        

        @Override
        public void onGeoQueryError(DatabaseError error) 
            Log.d("geoQuery","onGeoQueryError");
            // ...
        

    );
    this.setChanged();
    notifyObservers();
    this.clearChanged();
    Log.d("update","clearChanged");

这是我的 Firebase 数据:

如果需要,我想我可以修改数据结构。

日志

09-12 08:55:33.818 17710-17710/es.rchampa.weirdo D/geoQuery: lat=40.4430883 lng=-3.721805
09-12 08:55:33.982 17710-17710/es.rchampa.weirdo D/geoQuery: lat=40.4430883 lng=-3.721805
09-12 08:55:33.986 17710-17710/es.rchampa.weirdo D/geoQuery: onGeoQueryReady
09-12 08:55:34.025 17710-17710/es.rchampa.weirdo D/geoQuery: onGeoQueryReady

Gradle 文件

....
// Firebase
implementation 'com.google.firebase:firebase-database:16.0.1'
implementation 'com.google.firebase:firebase-storage:16.0.1'
implementation 'com.google.firebase:firebase-auth:16.0.3'
implementation 'com.google.firebase:firebase-crash:16.2.0'
implementation 'com.google.firebase:firebase-core:16.0.3'

// Firebase UI
implementation 'com.firebaseui:firebase-ui-database:1.2.0'

//Firebase GeoFire
implementation 'com.firebase:geofire-android:2.3.1'

// Google Play Services
implementation 'com.google.android.gms:play-services-auth:16.0.0'
implementation 'com.google.android.gms:play-services-maps:15.0.1'
implementation 'com.google.android.gms:play-services-location:15.0.1'
....

更新

如果您愿意,我可以授予对我私人仓库的访问权限。

【问题讨论】:

您是否尝试从onGeoQueryError() 方法记录error?它打印什么? @AlexMamo 我已更新代码以添加日志。仅执行 onGeoQueryReady。有什么帮助吗? 我认为您设计数据库的方式正在产生问题。 ***.com/a/42632772/6082859 你能添加一些关于 lat、lng、geolocation 以及 geoquery 的日志并在此处发布日志吗? @Nishita 设计没问题medium.com/google-cloud/… 【参考方案1】:

您将值8f(浮点数)作为radius 传递,而radius 应该是8.0dDouble.valueOf(8.0),其中MAX_SUPPORTED_RADIUS 等于8587 公里。

虽然实际的问题是,GeoFire 已经需要知道.child("location"),但不可能用Reference 来表示;只有DataSnapshotgetChildren()

底线是:

您必须创建一个单独的locations Reference,以避免嵌套。尽管如此,您仍然可以为这些节点使用相关的 uid 键(或至少将其添加为子节点),以便能够在 users Reference 中查找。这是一个1:1 关系,在两个References 之间。

所以这是一个有效的Java 示例,只是因为...

我们假设以下结构(如上所述):


  "locations" : 
    "CR88aa9gnDfJYYGq5ZTMwwC38C12" : 
      ".priority" : "9q8yywdgue",
      "g" : "9q8yywdgue",
      "l" : [ 37.7853889, -122.4056973 ]
    
  ,
  "users" : 
    "CR88aa9gnDfJYYGq5ZTMwwC38C12" : 
      "displayName" : "user 01",
      ...
    
  

数据库规则应该为locations 字段g 设置.indexOn


  "rules": 
    ...
    "locations": 
      ".indexOn": "g"
    
  

模块build.gradle中的依赖:

dependencies 
    ...
    implementation "com.firebase:geofire-android:2.3.1"

这里演示了如何通过GeoQuery结果获取用户的快照;

注意GeoQueryEventListener 而不是GeoQueryDataEventListener

public class GeofireActivity extends AppCompatActivity 

    private static final String LOG_TAG = GeofireActivity.class.getSimpleName();

    private DatabaseReference refBase     = null;
    private DatabaseReference refLocation = null;
    private DatabaseReference refUser     = null;

    private GeoFire geoFire = null;

    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        this.setContentView(R.layout.fragment_geofire);
        this.setReferences();
    

    private void setReferences() 
        this.refBase = FirebaseDatabase.getInstance().getReference();
        this.refUser = refBase.child("users");
        this.refLocation = refBase.child("locations");
        this.geoFire = new GeoFire(this.refLocation);
    

    private void searchNearby(double latitude, double longitude, double radius) 
        this.searchNearby(new GeoLocation(latitude, longitude), radius);
    

    private void searchNearby(GeoLocation location, double radius) 

        GeoQuery geoQuery = this.geoFire.queryAtLocation(location, radius);
        geoQuery.addGeoQueryEventListener(new GeoQueryEventListener() 

            @Override
            public void onKeyEntered(String key, GeoLocation location) 

                String loc = String.valueOf(location.latitude) + ", " + String.valueOf(location.longitude);
                Log.d(LOG_TAG, "onKeyEntered: " + key + " @ " + loc);

                /* once the key is known, one can lookup the associated record */
                refUser.child(key).addListenerForSingleValueEvent(new ValueEventListener() 

                    @Override
                    public void onDataChange(@NonNull DataSnapshot dataSnapshot) 
                        Log.d(LOG_TAG, "onDataChange: " + dataSnapshot.toString());
                    

                    @Override
                    public void onCancelled(@NonNull DatabaseError firebaseError) 
                        Log.e(LOG_TAG, "onCancelled: " + firebaseError.getMessage());
                    
                );
            

            @Override
            public void onKeyExited(String key) 
                Log.d(LOG_TAG, "onKeyExited: " + key);
            

            @Override
            public void onKeyMoved(String key, GeoLocation location) 
                Log.d(LOG_TAG, "onKeyMoved: " + key);
            

            @Override
            public void onGeoQueryReady() 
                Log.d(LOG_TAG, "onGeoQueryReady");
            

            @Override
            public void onGeoQueryError(DatabaseError error) 
                Log.e(LOG_TAG, "onGeoQueryError" + error.getMessage());
            
        );
    

为了保持完整性,当删除用户记录时,需要删除关联的位置记录 - 否则会导致无法再查找密钥。

【讨论】:

我已经更新了帖子,因为仍然找不到位置。 @Ricardo 您使用哪个版本的库以及哪个侦听器?您的代码仍然使用GeoQueryDataEventListener,我认为 GeoFire v1 和 v2 库之间存在差异(这就是我发布示例代码的原因,这对我有用)。 我正在使用:com.firebase:geofire-android:2.3.1 @Ricardo 尝试实现 GeoQueryEventListener,它具有其他所有事件 - 以及其他含义;另一个GeoQueryDataEventListener 非常适合通知服务器端更改,例如。当另一个客户端更改数据时。 我的错。我使用了错误的路径参考。现在正在工作!谢谢。【参考方案2】:

问题是当你通过半径时 根据问题https://github.com/firebase/geofire-java/issues/72

    double radius = 8589; // Fails
//  double radius = 8587.8; //Passes

尝试像这样传递值可能会有所帮助

//GeoQuery geoQuery = geoFire.queryAtLocation(geoLocation, 8f);
GeoQuery geoQuery = geoFire.queryAtLocation(geoLocation, radius);

传递值 8f (float) 作为半径,而半径 应该是 8.0d 或 Double.valueOf(8.0),其中 MAX_SUPPORTED_RADIUS 等于 8587 公里。

【讨论】:

我已经更新了帖子,因为仍然找不到位置。【参考方案3】:

就目前而言,geofire 可用作进行地理查询的索引,并提供所需文档的键(将存储在单独的“集合”中)。

您应该使用geofire 和一个单独的“集合”(称之为usersLocations

DatabaseReference ref = FirebaseDatabase.getInstance().getReference("usersLocations");
GeoFire geoFire = new GeoFire(ref);

现在您可以将其用作用户的索引,并可以像这样向其中添加项目。

geoFire.setLocation('QymlMpC0Zc', new GeoLocation(40.2334983, -3.7185183));

您的 Firebase RTDB 现在将如下所示:


   'users': 
        'QymlMpC0Zc': 
            // All your data here
        
    ,
   'usersLocations': 
        'QymlMpC0Zc': 
            'g': 'ezjkgkk305',
            'l': 
                '0': 40.2334983,
                '1': -3.7185183
            
        
    

所以最后,当您对 geoFire 进行查询时,您最终会触发您拥有的任何侦听器。

作为一个小提示...我不是 Java 开发人员,但我确实使用/知道 geofire 一般。希望我的建议/想法会有所帮助。

【讨论】:

我已经更新了帖子,因为仍然找不到位置。

以上是关于Geofire - 未找到位置的主要内容,如果未能解决你的问题,请参考以下文章

用户位置的 GeoFire 查询

如何使用geofire从firebase获取位置

如何让用户在我在 geofire、Firebase 的位置附近

如何从 GeoFire 中检索持续更新的位置数据并将其放在 Google 地图上?

如何在 Swift 中使用 Geofire 从一个模拟器存储位置,然后在另一个模拟器中显示该位置。

如何在没有 Geofire 的情况下将位置对象保存在 Firebase 中?