草庐IT

java - 如何访问 RecyclerView 适配器的 ViewHolder 的数据源?

coder 2023-12-10 原文

我的 RecyclerView 适配器的构造函数如下所示:

Context context;
List<ConnectionItem> connections;

public ConnectionsListAdapter(Context context, List connections) {
    this.context = context;
    this.connections = connections;
}

在适配器声明之后,我为 RecyclerView 声明了一个静态 ViewHolder 类,它还处理任何按钮的 onClicks:

public static class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {

    public Context context;

    public ImageView connectionImage;
    public TextView connectionName;
    public ImageButton startMsg;

    public ViewHolder(View itemView, List<ConnectionItem> connections) {
        super(itemView);
        ...
        startMsg.setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        Intent intent = new Intent(v.getContext(), ChatActivity.class);
        intent.putExtra("name", connections.get(getAdapterPosition()).getName()); // Error because accessing from static context
        intent.putExtra("id", connections.get(getAdapterPosition()).getUid()); // Error because accessing from static context
        context.startActivity(intent);
    }
}

问题是 connections 无法从 ViewHolder 静态类中的静态上下文访问。我的 ViewHolder 从 RecyclerView 适配器获取信息的最佳方式是什么?作为一种解决方法,我将数据源传递给 ViewHolder 的构造函数,并在 ViewHolder 中为数据源创建一个新的实例变量:

public static class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
    public Context context;
    public List<ConnectionItem> connections;

    public ImageView connectionImage;
    public TextView connectionName;
    public ImageButton startMsg;

public ViewHolder(View itemView, List<ConnectionItem> connections) {
        super(itemView);
        this.connections = connections;
        ...
        startMsg.setOnClickListener(this);
}

    @Override
    public void onClick(View v) {
        Intent intent = new Intent(v.getContext(), ChatActivity.class);
        intent.putExtra("name", connections.get(getAdapterPosition()).getName()); // Okay now
        intent.putExtra("id", connections.get(getAdapterPosition()).getUid()); // Okay now
        context.startActivity(intent);
    }

我的方法是否会破坏 ViewHolder 模式或在未来产生任何其他问题?

最佳答案

您的责任/顾虑有点困惑。

在 MVC 术语中,ViewHolder 实际上是一个 View 。它只是一个引用 subview 的对象,因此不必一遍又一遍地调用 findViewById()。但它仍然是一个真正的观点。

但是,您有一个以模型数据作为参数的构造函数,这有一股难闻的气味。

  • 您的 ViewHolder 应具有的唯一属性/变量是 View

  • ViewHolder 构造函数唯一应该调用的是 findViewById()

您并没有真正谈论过您的适配器,但您会注意到您覆盖了两个方法:onCreateViewHolderonBindViewHolder

onCreateViewHolder 中,您构建适用于您的 View 类型的 ViewHolder。而已。您可能需要根据 View 类型膨胀不同的布局,但您只是在这里处理 View 。不要将任何模型数据传递给构造函数中的 ViewHolder

onBindViewHolder 中,这里 是您将 View 连接到模型数据的地方。

我经常做的事情是为我的 ViewHolder 定义一个 bind() 方法,这样就可以非常清楚地传递哪些模型数据。 bind() 方法获取数据并调用 setText 和类似方法以使 View 反射(reflect)该适配器位置的模型数据。

但现在我们来到了 RecyclerViewAdapter 设计中丑陋的部分。适配器没有 OnItemClickListener

谷歌认为这是一个更好的设计;无论如何,事件应该由列表项 View 处理。我明白了。但问题是事件必须满足它的模型数据,并且它是具有模型数据的适配器,而不是列表项 View 。

并且 Google 强调您不能对位置等值使用 final;您需要调用 getAdapterPosition() 来索引模型数据。

所以现在我找到了一个可以适应所有这些约束的模式。

  • 我使用采用位置参数的方法为事件监听器定义了一个接口(interface)

  • 我在适配器中创建一个监听器实例

  • 每次绑定(bind)到 ViewHolder 时,我都会传递此监听器实例(因此 ViewHolder 确实有对监听器的引用,但没有 适配器)

  • ViewHolder 的事件处理程序上,我调用 getAdapterPosition() 来获取列表项的位置,然后我用它调用监听器方法位置

  • 适配器获取监听器回调,访问正确的模型数据并执行所需的操作

下面是一个示例:我有一个已选择的项目列表,即在线购物车中的产品有一个带有大 X 的删除按钮。现在我必须处理该删除按钮。

定义接口(interface):

    interface OnItemRemovedListener {

        void itemRemoved(int position);
    }

在适配器中创建监听器实例

    private OnItemRemovedListener mCallback;

在适配器构造函数中设置它:

    mCallback = new OnItemRemovedListener() {
        @Override
        public void itemRemoved(int position) {
            mItemList.remove(position);
            notifyDataSetChanged();
        }
    };

ViewHolder 子类:

public static class ProductItemViewHolder extends RecyclerView.ViewHolder {

    private TextView mProductName;

    private Button mRemoveButton;

    private OnItemRemovedListener mListener;

    public ProductItemViewHolder(View itemView) {
        super(itemView);
        mProductName = (TextView) itemView.findViewById(R.id.product_name);
        mRemoveButton = (Button) itemView.findViewById(R.id.remove_button);
    }

    public void bind(Model data, OnItemRemovedListener listener) {

        mProductName.setText(data.getProductName());
        mListener = listener;
        mRemoveButton.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View view) {
                int position = getAdapterPosition();
                if (position >= 0) {
                    mListener.itemRemoved(position);
                }
            }
        });
    }
}

然后 onBindViewHolder 覆盖看起来像这样:

    @Override
    public void onBindViewHolder(ProductItemViewHolder holder, int position) {

        holder.bind(mItemList.get(position), mCallback);
    }

关于java - 如何访问 RecyclerView 适配器的 ViewHolder 的数据源?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38574912/

有关java - 如何访问 RecyclerView 适配器的 ViewHolder 的数据源?的更多相关文章

  1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

    我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

  2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  3. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  4. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  5. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

    给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

  6. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

    我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

  7. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  8. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  9. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  10. ruby - 如何每月在 Heroku 运行一次 Scheduler 插件? - 2

    在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/

随机推荐