提问者:小点点

如何在节点火库中获得唯一的随机产品?


数据:

- products
   - -L74Pc7oVY22UsCETFBv
       - name: "gjwj"
       - category: "hreggrrg"
       - location: "vjhiwehifwe"
       - price: 44
       - color: fassaf
   - -L74uJ7oVYcVNyCteFBz
       - name: "uygfwh"
       - category: "hhhjwwwom"
       - location: "pervrr"
       - price: 33
       - color: yrtrr
   ......................

我有很多产品在节点产品,超过1000个产品。我可以在ListView中显示所有内容。我想只显示一个独特的随机给用户,像亮点,但不是下载所有,只有一个。

代码:

ValueEventListener v = new ValueEventListener() {
    @Override
    public void onDataChange(DataSnapshot dataSnapshot) {
        for(DataSnapshot child : dataSnapshot.getChildren()) {
            String name = (String) child.child("name").getValue().toString();
            Toast.makeText(MainActivity.this, name, Toast.LENGTH_SHORT).show();
            //How to get?????
        }
    }

    @Override
    public void onCancelled(DatabaseError databaseError) {}
};
FirebaseDatabase.getInstance().getReference().addListenerForSingleValueEvent(v);

如何在节点火库中获得唯一的随机产品?


共1个答案

匿名用户

在SOF上有一些很好的答案,但是它们是分开的,所以我将尝试用两种方法来回答你的问题。

但首先,在编写一些代码之前,我可以告诉您,这段代码不起作用,因为您在引用中传递了一个子代码,它是products,这显然是假设products节点是Firebase数据库根的直接子代码。

实际答案是:

假设您的数据库结构如下所示(其中products节点是Firebase数据库的直接子级):

Firebase-root
   |
   --- products
         |
         --- productIdOne
         |      |
         |      --- name: "gjwj"
         |      |
         |      --- category: "hreggrrg"
         |      |
         |      --- location: "vjhiwehifwe"
         |      |
         |      --- price: 44
         |      |
         |      --- color: "fassaf"
         |
         --- productIdTwo
         |      |
         |      --- name: "uygfwh"
         |      |
         |      --- category: "hhhjwwwom"
         |      |
         |      --- location: "pervrr"
         |      |
         |      --- price: 33
         |      |
         |      --- color: "yrtrr"
         |
         --- //And so on

要获得随机产品,请使用以下代码:

ListView listView = (ListView) findViewById(R.id.list_view);
ArrayAdapter arrayAdapter = new ArrayAdapter<>(context, android.R.layout.simple_list_item_1, randomProductList);
listView.setAdapter(arrayAdapter);
DatabaseReference rootRef = FirebaseDatabase.getInstance().getReference();
DatabaseReference productsRef = rootRef.child("products"); //Added call to .child("products")
ValueEventListener valueEventListener = new ValueEventListener() {
    @Override
    public void onDataChange(DataSnapshot dataSnapshot) {
        List<String> productList = new ArrayList<>();
        for(DataSnapshot ds : dataSnapshot.getChildren()) {
            String name = ds.child("name").getValue(String.class);
            productList.add(name);
        }

        int productListSize = productList.size();
        List<String> randomProductList = new ArrayList<>();

        randomProductList.add(new Random().nextInt(productListSize)); //Add the random product to list
        arrayAdapter.notifyDatasetChanged();
    }

    @Override
    public void onCancelled(DatabaseError databaseError) {
        Log.d(TAG, "Error: ", task.getException()); //Don't ignore errors!
    }
};
productsRef.addListenerForSingleValueEvent(valueEventListener);

为了获得所有产品,您需要循环遍历products节点的所有子节点。因此必须调用子(“products”)

如果您想要一个以上的随机产品,那么您可以创建一个循环,并在RandomProductList中添加任意数量的随机产品。

这被称为经典解决方案,您可以将它用于只包含几条记录的节点,但是如果您害怕获得大量数据,那么我将推荐您使用第二种方法。这还涉及到通过添加名为productids的新节点来对数据库进行一点更改。您的数据库结构应该如下所示:

Firebase-root
   |
   --- products
   |     |
   |     --- productIdOne
   |     |      |
   |     |      --- //details
   |     |
   |     --- productIdTwo
   |            |
   |            --- //details
   |      
   --- productIds
          |
          --- productIdOne: true
          |
          --- productIdTwo: true
          |
          --- //And so on

因此,正如您在问题中提到的,如果希望避免下载包含所有产品和所有属性的整个products节点,则必须创建一个名为productids的单独节点。因此,要获得单个产品,只需下载一个仅包含产品ID的简单节点。

这种做法被称为反规范化(复制数据),在涉及到FireBase时是一种常见的做法。为了更好的理解,我建议你看这段视频,反规范化在Firebase数据库中是正常的。

但是请记住,在这个新创建的节点中添加随机产品的方式,与在不再需要它们时删除它们的方式相同。

因此,要获得随机产品,您需要查询数据库两次。请看下面的代码:

DatabaseReference rootRef = FirebaseDatabase.getInstance().getReference();
DatabaseReference productIdsRef = rootRef.child("productIds");
ValueEventListener valueEventListener = new ValueEventListener() {
    @Override
    public void onDataChange(DataSnapshot dataSnapshot) {
        List<String> productIdsList = new ArrayList<>();
        for(DataSnapshot ds : dataSnapshot.getChildren()) {
            String productId = ds.getKey();
            productIdsList.add(productId);
        }

        int productListSize = productList.size();
        List<String> randomProductList = new ArrayList<>(););

        DatabaseReference productIdRef = rootRef.child("products").child(productIdsList.get(new Random().nextInt(int productListSize));
        ValueEventListener eventListener = new ValueEventListener() {
            @Override
            public void onDataChange(DataSnapshot dataSnapshot) {
                String name = dataSnapshot.child("name").getValue(String.class);
                Log.d("TAG", name);
            }

            @Override
            public void onCancelled(DatabaseError databaseError) {
                Log.d(TAG, "Error: ", task.getException()); //Don't ignore errors!
            }
        };
        productIdRef.addListenerForSingleValueEvent(eventListener);
    }

    @Override
    public void onCancelled(DatabaseError databaseError) {
        Log.d(TAG, "Error: ", task.getException()); //Don't ignore errors!
    }
};
productIdsRef.addListenerForSingleValueEvent(valueEventListener);

当您对Firebase数据库执行查询时,可能会有多个结果。因此DataSnapShot包含这些结果的列表。即使只有单个结果,DataSnapShot对象也会包含一个结果列表。