新闻中心
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.j*a),将存储Drawable ID的字段类型从int更改为String。
Ingredient.j*a 更新示例:
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文件名(不包含文件扩展名)。
Moshi Chat
法国AI实验室Kyutai推出的端到端实时多模态AI语音模型,具备听、说、看的能力,不仅可以实时收听,还能进行自然对话。
160
查看详情
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.j*a 更新示例:
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,并可以设置一个默认的占位符图片来提升用户体验。
注意事项与最佳实践
- 命名规范: 确保JSON中的字符串名称与res/drawable文件夹下的实际Drawable文件名(不含扩展名)完全匹配,包括大小写。
- 性能考虑: getIdentifier()方法在运行时通过字符串查找资源,相比直接使用整数ID会有轻微的性能开销。但在大多数RecyclerView场景下,这种开销通常可以忽略不计。如果需要极致性能且资源数量非常庞大,可以考虑在应用启动时将所有字符串名称映射到整数ID的HashMap中,以空间换时间。
- 默认图片: 强烈建议在getIdentifier()返回0时,设置一个默认的占位符图片。这可以防止在资源名称拼写错误或资源缺失时,应用界面出现空白或崩溃。
- 资源管理: 保持Drawable资源文件的命名清晰和一致性,这有助于代码的可读性和维护性。
总结
通过将Drawable资源的整数ID替换为其字符串名称,并在运行时使用Context.getResources().getIdentifier()动态获取ID,我们可以有效地解决Android资源ID不稳定导致的问题。这种方法使得应用能够更健壮地处理外部配置的资源,提高了应用的可维护性和用户体验,是Android开发中处理动态资源加载的推荐实践。
以上就是Android开发:通过字符串名称动态加载Drawable资源以避免运行时错误的详细内容,更多请关注其它相关文章!
# 不稳定
# 黔江企业网站建设
# 济南网站推广模板
# 昌平建设官方网站
# 中国联通营销推广定位
# 租车网站建设公司待遇
# 吉林品牌网站建设定制
# 一流的网站建设推广
# 南通抖音seo报名
# 洗鞋营销推广文案范文
# lg的营销推广活动
# 这一
# 时长
# 转换为
# java
# 以避免
# 好了
# 是在
# 数据结构
# 找不到
# 加载
# red
# 应用开发
# google
# go
# json
# js
# android
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
vivo云服务网页版登录 怎么登录vivo云服务网页版
京东单号查询入口_京东快递订单追踪入口
Lar*el 递归关系中排除指定分支的教程
美团外卖商家服务中心入口 美团商家版官网入口
微信网页版官方快速登录入口 微信网页版网页版账号直达
Win11怎么设置鼠标指针速度_Win11提高鼠标指针精确度选项
C++如何生成随机数_C++ random库使用方法与范围设置
小红书怎么解除第三方平台绑定_小红书多平台登录解绑方法介绍
KFC早餐时段怎么领特惠代码_KFC早餐订餐优惠代码获取与使用说明
Pyrogram与g4f集成:异步编程实践与常见错误解决
网易大神账号申诉需要多久_网易大神账号申诉流程说明
QQ邮箱在线使用入口 QQ邮箱个人账号网页版登录
Win10系统服务哪些可以禁用 Win10安全优化服务列表【干货】
优化 Python 函数中的条件逻辑:解决 if-else 嵌套与参数选择问题
“音游” × “怪文书” 题材的节奏冒险游戏 《晕晕电波症候群》确定于2026年4月发售!
谷歌浏览器无痕模式怎么开 Chrome开启无痕浏览设置方法【教程】
CKEditor 5 自定义构建在React应用中渲染失败的调试与解决
React Router v6 教程:构建认证保护的私有路由与重定向策略
内存疯狂猛猛涨价:主板销量直接腰斩!
俄罗斯方块最新版入口 俄罗斯方块在线玩官网入口
Shopware订单对象中获取产品自定义字段的正确方法
外媒分析《GTA6》定价:卖100美元可以但真没必要!
Lar*el用户头像管理:实现图片缩放、存储与旧文件安全删除的最佳实践
Python字典中优雅地迭代剩余元素的方法
抖音网页版企业服务中心登录入口_抖音网页版企业登录平台
4399免费游戏网址入口 4399小游戏免费入口点开即玩
Python中高效访问嵌套字典与列表中的键值对
jQuery Mask 插件中实现电话号码固定前导零的教程
Safari浏览器输入栏卡顿如何解决 Safari搜索建议与缓存清理
mc.js免安装版 mc.js一键畅玩入口
深入理解J*a合成构造器:何时以及为何阻止其生成
Eclipse怎么运行工程_Eclipse工程运行配置说明
手机屏幕碎了但能正常使用怎么办 手机外屏碎裂的修复建议
Promise错误处理:在catch后终止链式then执行的策略
Golang如何使用net/url解析URL_Golang URL解析与处理方法
PHP URL参数传递与500错误调试指南
蛙漫2台版漫画地址 Manwa2正版网页版链接
抖音极速版最新版本 抖音极速版官方下载地址
c++ dfs和bfs代码 c++深度广度优先搜索算法
必由学在线入口 必由学网页版快速登录入口
神庙逃亡小游戏在线玩 神庙逃亡小游戏入口
J*aScript生成器_j*ascript异步迭代
HTML5原生日期选择器与jQuery UI:实现日期选择器的联动与程序化控制
新手怎么开始学化妆 零基础化妆入门教程
黑鲨3Pro怎样在相册开漫画风滤镜_iPhone黑鲨3Pro相册开漫画风滤镜【趣味滤镜】
处理Kafka消费者会话超时:深入理解消息处理语义与幂等性
漫蛙2正版漫画站 漫蛙2网页版快速访问入口
解决深度学习模型训练初期异常高损失与完美验证准确率问题
AO3网页版最新入口合集 Archive of Our Own在线访问指南
Golang如何实现容器化日志收集与分析_Golang容器日志收集分析方法


2025-12-05
浏览次数:次
返回列表