0

0

Android开发:通过字符串名称动态加载Drawable资源以避免运行时错误

聖光之護

聖光之護

发布时间:2025-12-05 21:39:25

|

657人浏览过

|

来源于php中文网

原创

Android开发:通过字符串名称动态加载Drawable资源以避免运行时错误

android应用开发中,直接将`r.drawable`生成的整数id存储到外部数据(如json)是不可靠的,因为这些id在不同编译版本或运行时可能发生变化,导致`resources$notfoundexception`。本教程将详细阐述这一问题的原因,并提供一种健壮的解决方案:通过在外部数据中存储drawable资源的字符串名称,然后在运行时使用`context.getresources().getidentifier()`方法动态获取正确的资源id,确保应用稳定加载图像资源。

理解Android资源ID的不稳定性

Android资源系统为应用中的每个资源(如布局、字符串、图片等)分配一个唯一的整数ID,这些ID定义在自动生成的R类中。例如,R.drawable.tomato会对应一个特定的整数值。然而,一个常见的误解是这些整数ID是固定不变的。实际上,这些资源ID是在每次编译时动态生成的,它们并非在不同编译版本之间稳定不变。

当我们将这些整数ID序列化到外部存储(如JSON文件)中时,问题就出现了。如果应用重新编译,或者在不同的设备/Android版本上运行,之前存储的ID可能不再对应正确的资源,甚至可能对应一个不存在的资源。这会导致运行时抛出android.content.res.Resources$NotFoundException异常,提示“Invalid ID”或“Resource ID #0x...”,从而导致应用崩溃。

解决方案:通过字符串名称动态加载Drawable

为了解决资源ID不稳定的问题,最佳实践是在外部数据中存储Drawable资源的字符串名称,而不是其整数ID。例如,对于R.drawable.tomato,我们应该存储字符串"tomato"。然后在应用运行时,利用Android Resources类提供的getIdentifier()方法,根据字符串名称动态查找并获取当前编译环境下的正确资源ID。

1. 更新数据模型

首先,修改你的数据类(例如Ingredient.java),将存储Drawable ID的字段类型从int更改为String。

Ingredient.java 更新示例:

package me.eyrim.foodrecords2;

import com.google.gson.annotations.SerializedName;

public class Ingredient {
    @SerializedName("ingredient_name")
    private String ingredientName;
    // 将 int 更改为 String
    @SerializedName("ingredient_drawable_tag")
    private String ingredientDrawableTag; 

    public String getIngredientName() {
        return this.ingredientName;
    }

    // 返回 String 类型
    public String getIngredientDrawableTag() {
        return this.ingredientDrawableTag;
    }
}

2. 更新JSON数据结构

相应地,修改你的JSON数据,将ingredient_drawable_tag的值从整数改为对应的Drawable文件名(不包含文件扩展名)。

飞书多维表格
飞书多维表格

表格形态的AI工作流搭建工具,支持批量化的AI创作与分析任务,接入DeepSeek R1满血版

下载

JSON数据更新示例:

{
    "recipe_name": "My test recipe 1 updated",
    "recipe_id": "0",
    "recipe_desc": "this is a test desc for my test recipe 1",
    "ingredients": [{
            "ingredient_name": "Tomato",
            // 从整数 ID 更改为字符串名称
            "ingredient_drawable_tag": "tomato" 
        },
        {
            "ingredient_name": "Pepper",
            // 从整数 ID 更改为字符串名称
            "ingredient_drawable_tag": "pepper" 
        }
    ]
}

3. 在RecyclerView Adapter中动态加载Drawable

在你的RecyclerView.Adapter中,当绑定数据到视图时(onBindViewHolder方法),使用Context.getResources().getIdentifier()方法来获取Drawable的实际资源ID。

IngredientRecyclerViewAdapter.java 更新示例:

package me.eyrim.foodrecords2.recipeviewactivity;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;

import androidx.annotation.NonNull;
import androidx.recyclerview.widget.RecyclerView;

import me.eyrim.foodrecords2.Ingredient;
import me.eyrim.foodrecords2.R; // 确保 R 类被导入

public class IngredientRecyclerViewAdapter extends RecyclerView.Adapter<IngredientRecyclerViewAdapter.IngredientViewHolder> {
    private final Ingredient[] ingredients;
    private final Context context; // 存储 Context 实例

    // 构造函数需要接收 Context
    public IngredientRecyclerViewAdapter(Ingredient[] ingredients, Context context) {
        this.ingredients = ingredients;
        this.context = context; // 初始化 Context
    }

    @NonNull
    @Override
    public IngredientViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
        View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.ingredient_row, parent, false);
        return new IngredientViewHolder(view);
    }

    @Override
    public void onBindViewHolder(@NonNull IngredientViewHolder holder, int position) {
        String drawableName = this.ingredients[position].getIngredientDrawableTag();

        // 使用 getIdentifier() 动态获取资源 ID
        int drawableId = context.getResources().getIdentifier(
            drawableName,      // 资源名称 (字符串)
            "drawable",        // 资源类型 (例如 "drawable", "string", "layout")
            context.getPackageName() // 应用包名
        );

        // 检查是否成功找到资源ID,避免找不到资源时崩溃
        if (drawableId != 0) {
            holder.imageView.setImageResource(drawableId);
        } else {
            // 如果找不到对应的 drawable,可以设置一个默认的占位符图片
            holder.imageView.setImageResource(R.drawable.default_ingredient_placeholder); // 假设你有一个默认图片
            // 或者记录日志,以便调试
            // Log.w("IngredientAdapter", "Drawable not found for name: " + drawableName);
        }
    }

    @Override
    public int getItemCount() {
        return this.ingredients.length;
    }

    public static class IngredientViewHolder extends RecyclerView.ViewHolder { // 静态内部类
        private final ImageView imageView;
        // 如果有 TextView,也在这里初始化
        // private final TextView textView;

        public IngredientViewHolder(@NonNull View itemView) {
            super(itemView);
            imageView = itemView.findViewById(R.id.ingredient_image);
            // textView = itemView.findViewById(R.id.ingredient_name_text); // 如果有的话
        }
    }
}

关键点说明:

  • Context 实例: getIdentifier() 方法需要一个Context实例来访问应用的资源。在RecyclerView.Adapter中,通常通过构造函数传入Context并作为成员变量保存。
  • getIdentifier(name, defType, defPackage) 参数:
    • name: 要查找的资源名称(例如,"tomato")。
    • defType: 资源的类型,对于图片通常是"drawable"。其他常见类型包括"string"、"layout"、"id"等。
    • defPackage: 资源的包名,通常是context.getPackageName(),即当前应用的包名。
  • 错误处理: 如果getIdentifier()找不到对应的资源,它会返回0。务必在设置图片前检查这个返回值,以避免Resources$NotFoundException,并可以设置一个默认的占位符图片来提升用户体验。

注意事项与最佳实践

  1. 命名规范: 确保JSON中的字符串名称与res/drawable文件夹下的实际Drawable文件名(不含扩展名)完全匹配,包括大小写。
  2. 性能考虑: getIdentifier()方法在运行时通过字符串查找资源,相比直接使用整数ID会有轻微的性能开销。但在大多数RecyclerView场景下,这种开销通常可以忽略不计。如果需要极致性能且资源数量非常庞大,可以考虑在应用启动时将所有字符串名称映射到整数ID的HashMap中,以空间换时间。
  3. 默认图片: 强烈建议在getIdentifier()返回0时,设置一个默认的占位符图片。这可以防止在资源名称拼写错误或资源缺失时,应用界面出现空白或崩溃。
  4. 资源管理: 保持Drawable资源文件的命名清晰和一致性,这有助于代码的可读性和维护性。

总结

通过将Drawable资源的整数ID替换为其字符串名称,并在运行时使用Context.getResources().getIdentifier()动态获取ID,我们可以有效地解决Android资源ID不稳定导致的问题。这种方法使得应用能够更健壮地处理外部配置的资源,提高了应用的可维护性和用户体验,是Android开发中处理动态资源加载的推荐实践。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

455

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

546

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

335

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

82

2025.09.10

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1010

2023.08.02

resource是什么文件
resource是什么文件

Resource文件是一种特殊类型的文件,它通常用于存储应用程序或操作系统中的各种资源信息。它们在应用程序开发中起着关键作用,并在跨平台开发和国际化方面提供支持。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

181

2023.12.20

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

760

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

76

2026.03.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.2万人学习

Java 教程
Java 教程

共578课时 | 81万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号