jjzjj

android - 具有分页响应的 SyncAdapter 和 Rest Server

coder 2023-12-06 原文

我需要知道如何处理来自 Rest Server 的同步适配器和分页响应。我正在制作一个检索数据集合的 Android 应用程序。每页有 20 个项目,我现在在一个请求中检索我的所有项目。我认为我能做到的最好方法是检索一个页面,例如,滚动到 ListView 的末尾时使用 syncAdapter 发出另一个请求,但我不确定。

我正在搜索如何在 Android 中处理 REST 中的分页,但我没有找到任何有用的东西。我想知道是否有人可以帮助我。

谢谢。

这是我现在如何检索项目的示例。

public ArrayList<ContentProviderOperation> parse(String json) throws IOException, NullPointerException {

    final ArrayList<ContentProviderOperation> batch = new ArrayList<ContentProviderOperation>();

    AccountManager manager = AccountManager.get(mContext);
    Account account = ((KipptApplication)mContext.getApplicationContext()).getCurrentAccount();
    String authToken = manager.peekAuthToken(account, AuthenticatorActivity.PARAM_AUTHTOKEN_TYPE);

    Header[] headers = new Header[]{
            new BasicHeader(KipptConstants.API_USERNAME_KEY,account.name),
            new BasicHeader(KipptConstants.API_TOKEN_KEY,authToken)
    };


    Gson gson = new Gson();
    Type responseType = new TypeToken<Response<ClipObject>>() {}.getType();
    Response<ClipObject> inbox = gson.fromJson(json,responseType);

    List<ClipObject> clips = inbox.getObjects();

    String response = null;
    String next = inbox.getMeta().getNext();

    while(next !=null){

        try {
            Log.d(TAG,"Fetching more clips from: " + next);
            response = HttpHelper.getHttpResponseAsString(KipptConstants.DOMAIN_URL +
                    next, null, headers);
            inbox = gson.fromJson(response,responseType);
            /*Updating next uri*/
            next = inbox.getMeta().getNext();
            if(!inbox.getObjects().isEmpty()){
                clips.addAll(inbox.getObjects());
            }else{
                Log.d(TAG,"No more clips");
                break;
            }
        } catch (PersonalizedException e) {
            Log.e(TAG,e.toString());
            e.printStackTrace();
        }
    }

    for(ClipObject clip : clips){
        if(mKipptDAO.isClipInDb(mContext.getContentResolver(),clip.getId(),true)== null){
            Log.i(TAG,"Adding new clip");
            /*Parsing clip*/
            parseClip(clip,batch,false /*Clip isn't in database so update=false*/);
            /*Parsing media*/
            parseMedia(clip.getMedia(),clip.getId(),batch,false);
            /*Parsing comments if clip contains it*/
            if(clip.getCommentObjects().getCount()>0) {
                List<CommentObject> comments = clip.getCommentObjects().getListElements();
                for(CommentObject comment: comments){
                    parseComments(comment,clip.getId(),batch,false);
                }
            }
            /*TODO Parse Likes*/

            /*Parsing user creator*/
            parseCreator(clip.getUserCreator(),batch,false);

        }else{
            Log.i(TAG,"Modifying clip");
            /*Clip is in database*/
            if(!(clip.getUpdated()<= timestamp)){
                /*Parsing clip and update it in database*/
                parseClip(clip,batch,true);
                /*Parsing media and update it in database*/
                parseMedia(clip.getMedia(),clip.getId(),batch,true);
                /*Parsing comments and update it in database*/
                if(clip.getCommentObjects().getCount()>0) {
                    List<CommentObject> comments = clip.getCommentObjects().getListElements();
                    for(CommentObject comment: comments){
                        parseComments(comment,clip.getId(),batch,true);
                    }
                }
                /*TODO parse likes*/

                /*Parse Creator*/
                parseCreator(clip.getUserCreator(),batch,true);
            }
        }
        /*Updating timestamp*/
        if(timestamp<=clip.getUpdated())timestamp = clip.getUpdated();

    }

    /*Saving timestamp modified value in preferences file*/
    this.sharedPreferences.edit().putLong(KipptConstants.loadTimeStamp(currentFragmentIndex), timestamp).commit();

    return batch;
}

最佳答案

使用 SyncAdapter 建议您尝试 this .

也可以使用这个来检测 RecyclerView 中列表的结尾:

@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
                    super.onScrollStateChanged(recyclerView, newState);

                // When went to the end of the list, load more posts
                if (newState == RecyclerView.SCROLL_STATE_IDLE) {

                    if (linearLayoutManager.findLastVisibleItemPosition() >= linearLayoutManager.getItemCount() - 1) {

                      // Grow List
                    }
                }
}

关于android - 具有分页响应的 SyncAdapter 和 Rest Server,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22743052/

有关android - 具有分页响应的 SyncAdapter 和 Rest Server的更多相关文章

  1. ruby - 具有身份验证的私有(private) Ruby Gem 服务器 - 2

    我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..

  2. ruby-on-rails - 每次我尝试部署时,我都会得到 - (gcloud.preview.app.deploy) 错误响应 : [4] DEADLINE_EXCEEDED - 2

    我是Google云的新手,我正在尝试对其进行首次部署。我的第一个部署是RubyonRails项目。我基本上是在关注thisguideinthegoogleclouddocumentation.唯一的区别是我使用的是我自己的项目,而不是他们提供的“helloworld”项目。这是我的app.yaml文件runtime:customvm:trueentrypoint:bundleexecrackup-p8080-Eproductionconfig.ruresources:cpu:0.5memory_gb:1.3disk_size_gb:10当我转到我的项目目录并运行gcloudprevie

  3. ruby-on-rails - Rails 3.1 中具有相同形式的多个模型? - 2

    我正在使用Rails3.1并在一个论坛上工作。我有一个名为Topic的模型,每个模型都有许多Post。当用户创建新主题时,他们也应该创建第一个Post。但是,我不确定如何以相同的形式执行此操作。这是我的代码:classTopic:destroyaccepts_nested_attributes_for:postsvalidates_presence_of:titleendclassPost...但这似乎不起作用。有什么想法吗?谢谢! 最佳答案 @Pablo的回答似乎有你需要的一切。但更具体地说...首先改变你View中的这一行对此#

  4. 安卓apk修改(Android反编译apk) - 2

    最近因为项目需要,需要将Android手机系统自带的某个系统软件反编译并更改里面某个资源,并重新打包,签名生成新的自定义的apk,下面我来介绍一下我的实现过程。APK修改,分为以下几步:反编译解包,修改,重打包,修改签名等步骤。安卓apk修改准备工作1.系统配置好JavaJDK环境变量2.需要root权限的手机(针对系统自带apk,其他软件免root)3.Auto-Sign签名工具4.apktool工具安卓apk修改开始反编译本文拿Android系统里面的Settings.apk做demo,具体如何将apk获取出来在此就不过多介绍了,直接进入主题:按键win+R输入cmd,打开命令窗口,并将路

  5. ruby - 具有两个参数的 block - 2

    我从用户Hirolau那里找到了这段代码:defsum_to_n?(a,n)a.combination(2).find{|x,y|x+y==n}enda=[1,2,3,4,5]sum_to_n?(a,9)#=>[4,5]sum_to_n?(a,11)#=>nil我如何知道何时可以将两个参数发送到预定义方法(如find)?我不清楚,因为有时它不起作用。这是重新定义的东西吗? 最佳答案 如果您查看Enumerable#find的文档,您会发现它只接受一个block参数。您可以将它发送两次的原因是因为Ruby可以方便地让您根据它的“并行赋

  6. ruby-on-rails - 在 RSpec 中,如何以任意顺序期望具有不同参数的多条消息? - 2

    RSpec似乎按顺序匹配方法接收的消息。我不确定如何使以下代码工作:allow(a).toreceive(:f)expect(a).toreceive(:f).with(2)a.f(1)a.f(2)a.f(3)我问的原因是a.f的一些调用是由我的代码的上层控制的,所以我不能对这些方法调用添加期望。 最佳答案 RSpecspy是测试这种情况的一种方式。要监视一个方法,用allowstub,除了方法名称之外没有任何约束,调用该方法,然后expect确切的方法调用。例如:allow(a).toreceive(:f)a.f(2)a.f(1)

  7. ruby-on-rails - 在 Ruby on Rails 中发送响应之前如何等待多个异步操作完成? - 2

    在我做的一些网络开发中,我有多个操作开始,比如对外部API的GET请求,我希望它们同时开始,因为一个不依赖另一个的结果。我希望事情能够在后台运行。我找到了concurrent-rubylibrary这似乎运作良好。通过将其混合到您创建的类中,该类的方法具有在后台线程上运行的异步版本。这导致我编写如下代码,其中FirstAsyncWorker和SecondAsyncWorker是我编写的类,我在其中混合了Concurrent::Async模块,并编写了一个名为“work”的方法来发送HTTP请求:defindexop1_result=FirstAsyncWorker.new.async.

  8. ruby-on-rails - 具有同名的模块和类 - 2

    我有一个模块stat存在于目录结构中:lib/stat_creator/stat/在lib/stat_creator/stat.rb中,我在lib/stat_creator/stat/目录中有我需要的文件,以及:moduleStatCreatormoduleStatendend当我使用该模块时,我将这些类称为StatCreator::Stat::Foo.new现在我想要一个存在于应用程序中的根Stat类。我在app/models中制作了我的Stat类,并在routes.rb中进行了设置。但是,如果我转到Rails控制台并尝试在应用程序/模型中使用Stat类,例如:Stat.by_use

  9. ruby-on-rails - 在具有 ActiveRecord 条件的相关模型中按字段排序 - 2

    我正在尝试按Rails相关模型中的字段进行排序。我研究的所有解决方案都没有解决如果相关模型被另一个参数过滤?元素模型classItem相关模型:classPriority我正在使用where子句检索项目:@items=Item.where('company_id=?andapproved=?',@company.id,true).all我需要按相关表格中的“位置”列进行排序。问题在于,在优先级模型中,一个项目可能会被多家公司列出。因此,这些职位取决于他们拥有的company_id。当我显示项目时,它是针对一个公司的,按公司内的职位排序。完成此任务的正确方法是什么?感谢您的帮助。PS-我

  10. ruby-on-rails - Sunspot:如何对具有不同值的多个字段进行全文查询? - 2

    我想用sunspot重现以下原始solr查询q=exact_term_text:fooORterm_textv:foo*ORalternate_text:bar*但我无法通过标准的太阳黑子界面理解这是否可能以及如何实现,因为看起来:fulltext方法似乎不接受多个文本/搜索字段参数我不知道将什么参数作为第一个参数传递给fulltext,就好像我通过了"foo"或"bar"结果不匹配如果我传递一个空参数,我得到一个q=*:*范围过滤器(例如with(:term).starting_with('foo*')(顾名思义)作为过滤器查询应用,因此不参与评分。似乎可以手动编写字符串(或者可能使

随机推荐