记录一些琐碎的安卓知识

记录一些项目中碰到的小问题,小知识

图片来源于网络

1. jar 包与 aar 包

1.1 Android Studio 导出 jar 包

(1)在一个 Project 中加入 Module,即项目中新建 Module:
  File → new module → Android Library(这里我命名为 mylibrary);

(2)在新建的 library 中新建要用的 java 类;

(3)在 app 中引入此 module,即在 app 的build.gradle 中加入此 module 的 dependency,如下:

dependencies{
     compile 'com.android.support.appcompat-v7:21.0.3'
     compile project (':mylibrary')
}

(4)项目编译之后 jar 包就生成了,可在下面的两个目录中找到:

  • ./mylibrary/build/intermediates/bundles/debug/class.jar
  • ./mylibrary/build/intermediates/bundles/releases/class.jar

1.2 Android Studio 导入 jar 包

复制 jar 包到 app 的 libs 目录下,接着右键 → Add as Library 即可。

1.3 Android Studio 添加 aar 包引用

(1)复制 aar 包到 app 的 libs 目录下;

(2)配置 build.gradle 文件:

a. 添加:

respositories{
     flatDir{
          dirs 'Libs'
     }
}

b. 修改 dependencies,添加一行:

compile(name:'aar文件名称',ext:'aar')

(3)重新编译项目,编译完后在 app/build/intermediates/exploded-aar/ 可看到引用 aar 包的内容。

2. 其他杂项

2.1 在 activity 中获取另一个 xml 布局中的控件:

LayoutInflater layout = this.getLayoutInflater();
View view = layout.inflate(R.layout.布局名, null);
Button b = (Button)view.findViewById(R.id.b);

2.2 尽量减少对变量的重复计算,如:

for(int i = 0; i < list.size(); i++){ ... }

可替换为

for(int i = 0, length = list.size(); i < length; i++){ ... }

2.3 在多个 fragment 中,一些 fragment 特殊需求只支持竖屏,则可以:

在横竖屏都可用的 fragment 中的 onCreateView 中添加:

getActivity().setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_SENSOR);

在只支持竖屏的 fragment 中的 onCreateView 中添加:

getActivity().setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PROTRAIT);

Acivity 的一些屏幕属性:

ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE         // 指定横屏
ActivityInfo.SCREEN_ORIENTATION_PROTRAIT          // 竖屏
ActivityInfo.SCREEN_ORIENTATION_USER              // 用户朝向
ActivityInfo.SCREEN_ORIENTATION_NOSENSOR          // 不受重力影响
ActivityInfo.SCREEN_ORIENTATION_SENSOR_LANDSCAPE  // 横屏动态切换
ActivityInfo.SCREEN_ORIENTATION_SENSOR_PROTRAIT   // 竖屏动态切换
...

2.4 网页界面 WebView 中可返回,则可以:

// 设置返回键监听
 myWebView.setOnKeyListener(backListener);
. . .
private View.OnKeyListener backListener = new View.OnKeyListener() {
        @Override
        public boolean onKey(View view, int i, KeyEvent keyEvent) {
            if (keyEvent.getAction() == KeyEvent.ACTION_DOWN) {
                if (i == KeyEvent.KEYCODE_BACK) {  
                    if(myWebView.canGoBack()) {
                        myWebView.goBack();//返回上一页面
                        return true;
                    } else {
                        . . .
                    }
                }
            }
            return false;
        }
    }; 

2.5 命令行进入线上keystore的文件所在目录,然后输入如下命令,即可查看签名:

keytool -list -v -keystore app-release.jks

其中 app-release.jks 是指你要查看签名的 keystore 文件。

3 项目中碰到的一些bug

3.1 app 安装失败

app 安装失败

报上图中的错误是因为可能有以下几个原因:

  • 如图中所说手机上有其他版本app
     解决方法是直接点击 OK,卸载其他版本的 app

  • 用真机测试时,手机设置了禁止 USB 安装app
     解决方法是手机设置允许 USB 安装app

  • AndroidManifest 编写出错(很常见)
     解决方法是检查AndroidManifest 文件,看看是不是多写或少写一些东西,比如少了诸如 “.”、“:” 等这类标点符号,或者多了其他不必要的东西

  • 其他原因(手动滑稽)

3.2 java.lang.RuntimeException: Parcel: unable to marshal value

使用 Parcel 序列化在两个 Activity 或 Fragment 中进行数据传送,Parcel 需要实现 Parcelable 接口。而如果在 Parcel 类中,若需要读或者写其他的 bean 数据的话,这些 bean 需要实现 Serializable 或 Parcelable 序列化接口。否则会出现如标题所示错误。

3.3 android studio 在电脑蓝屏后或意外重启后 报错的解决方法

问题描述:在用studio开发项目的时候电脑突然蓝屏了,重启电脑之后打开项目各种红线,原本的import全部报错。

解决方法:File -> Invalidate Cashes / Restart

参考:
  在电脑蓝屏后 报错的解决方法
  android studio意外重启后 原本的import全部报错

3.4 Glide 加载图片底色变绿

Glide 使用 bitmap 的编码为 RGB565,所以有时由于过度压缩导致图片变绿,因此要改变一下 Glide 的 bitmap 编码:

public class GlideConfiguration implements GlideModule {
    @Override
    public void applyOptions(Context context, GlideBuilder builder) {
        builder.setDecodeFormat(DecodeFormat.PREFER_ARGB_8888);
    }

    @Override
    public void registerComponents(Context context, Glide glide) {

    }
}

接着在 AndroidManifest 中添加:

  <!-- 防止 Glide 加载图片底色变绿 -->
 <meta-data
     android:name="com.xxx.xxx.GlideConfiguration"
     android:value="GlideModule"/>

参考: Android Glide 图片加载框架图片变色变绿解决方法

3.5 ImageView 设置了 scaleType="centerCrop" Glide 加载圆角图片无效

public class GlideRoundedCornersTransform extends CenterCrop {
    private float mRadius;
    private CornerType mCornerType;
    private static final int VERSION = 1;
    private static final String ID = BuildConfig.APPLICATION_ID + "GlideRoundedCornersTransform." + VERSION;
    private static final byte[] ID_BYTES = ID.getBytes(CHARSET);

    /**
     * 待处理的圆角枚举
     */
    public enum CornerType {
        ALL,
        TOP_LEFT, TOP_RIGHT, BOTTOM_LEFT, BOTTOM_RIGHT,
        TOP, BOTTOM, LEFT, RIGHT,
        TOP_LEFT_BOTTOM_RIGHT,
        TOP_RIGHT_BOTTOM_LEFT,
        TOP_LEFT_TOP_RIGHT_BOTTOM_RIGHT,
        TOP_RIGHT_BOTTOM_RIGHT_BOTTOM_LEFT,
        TOP_LEFT_TOP_RIGHT
    }

    public GlideRoundedCornersTransform(float radius, CornerType cornerType) {
        super();
        mRadius = radius;
        mCornerType = cornerType;
    }

    @Override
    protected Bitmap transform(BitmapPool pool, Bitmap toTransform, int outWidth, int outHeight) {
        Bitmap transform = super.transform(pool, toTransform, outWidth, outHeight);
        return roundCrop(pool, transform);
    }

    private Bitmap roundCrop(BitmapPool pool, Bitmap source) {
        if (source == null) {
            return null;
        }
        int width = source.getWidth();
        int height = source.getHeight();
        Bitmap result = pool.get(source.getWidth(), source.getHeight(), Bitmap.Config.ARGB_8888);

        if (result == null) {
            result = Bitmap.createBitmap(source.getWidth(), source.getHeight(), Bitmap.Config
                    .ARGB_8888);
        }
        Canvas canvas = new Canvas(result);
        Paint paint = new Paint();
        paint.setShader(new BitmapShader(source, BitmapShader.TileMode.CLAMP, BitmapShader
                .TileMode.CLAMP));
        paint.setAntiAlias(true);

        Path path = new Path();
        drawRoundRect(canvas, paint, path, width, height);

        return result;
    }

    private void drawRoundRect(Canvas canvas, Paint paint, Path path, int width, int height) {
        float[] rids;
        switch (mCornerType) {
            case ALL:
                rids = new float[]{mRadius, mRadius, mRadius, mRadius, mRadius, mRadius, mRadius, mRadius};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case TOP_LEFT:
                rids = new float[]{mRadius, mRadius, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case TOP_RIGHT:
                rids = new float[]{0.0f, 0.0f, mRadius, mRadius, 0.0f, 0.0f, 0.0f, 0.0f};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case BOTTOM_RIGHT:
                rids = new float[]{0.0f, 0.0f, 0.0f, 0.0f, mRadius, mRadius, 0.0f, 0.0f};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case BOTTOM_LEFT:
                rids = new float[]{0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, mRadius, mRadius};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case TOP:
                rids = new float[]{mRadius, mRadius, mRadius, mRadius, 0.0f, 0.0f, 0.0f, 0.0f};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case BOTTOM:
                rids = new float[]{0.0f, 0.0f, 0.0f, 0.0f, mRadius, mRadius, mRadius, mRadius};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case LEFT:
                rids = new float[]{mRadius, mRadius, 0.0f, 0.0f, 0.0f, 0.0f, mRadius, mRadius};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case RIGHT:
                rids = new float[]{0.0f, 0.0f, mRadius, mRadius, mRadius, mRadius, 0.0f, 0.0f};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case TOP_LEFT_BOTTOM_RIGHT:
                rids = new float[]{mRadius, mRadius, 0.0f, 0.0f, mRadius, mRadius, 0.0f, 0.0f};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case TOP_RIGHT_BOTTOM_LEFT:
                rids = new float[]{0.0f, 0.0f, mRadius, mRadius, 0.0f, 0.0f, mRadius, mRadius};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case TOP_LEFT_TOP_RIGHT_BOTTOM_RIGHT:
                rids = new float[]{mRadius, mRadius, mRadius, mRadius, mRadius, mRadius, 0.0f, 0.0f};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case TOP_RIGHT_BOTTOM_RIGHT_BOTTOM_LEFT:
                rids = new float[]{0.0f, 0.0f, mRadius, mRadius, mRadius, mRadius, mRadius, mRadius};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            case TOP_LEFT_TOP_RIGHT:
                rids = new float[]{mRadius, mRadius, mRadius, mRadius, 0.0f, 0.0f, 0.0f, 0.0f};
                drawPath(rids, canvas, paint, path, width, height);
                break;
            default:
                throw new RuntimeException("RoundedCorners type not belong to CornerType");
        }
    }

    /**
     * @param rids 圆角的半径,依次为左上角xy半径,右上角,右下角,左下角
     */
    private void drawPath(float[] rids, Canvas canvas, Paint paint, Path path, int width, int height) {
        path.addRoundRect(new RectF(0, 0, width, height), rids, Path.Direction.CW);
        canvas.drawPath(path, paint);
    }

    @Override
    public boolean equals(Object o) {
        return o instanceof GlideRoundedCornersTransform;
    }

    @Override
    public int hashCode() {
        return ID.hashCode();
    }

    @Override
    public void updateDiskCacheKey(MessageDigest messageDigest) {
        messageDigest.update(ID_BYTES);
    }
}

使用:

RequestOptions mOptions = new RequestOptions().optionalTransform(new 
        GlideRoundedCornersTransform(10, GlideRoundedCornersTransform.CornerType.ALL));
Glide.with(this).load(imgUrl).apply(mOptions).into(imageView);

参考: Android Glide4.7.1下的圆角处理与高斯模糊

3.6 报错:BroadcastReceiver components are not allowed to register to receive intents

原因:context.registerReceiver() 不能在 BroadcastReceiver 的 onReceive()里面调用。

解决:提高context的级别,使用context.getApplicationContext().registerReceiver()即可。

3.7 volatile 关键字的作用(针对多线程情况下)

  • CPU高速缓存和可见性问题
  • 禁止指令重排

使用方法:
Java:变量的前面加上 volatile 关键字即可
Kotlin:变量的前面加上 @Volatile 注解即可

3.8 AndroidStudio更新4.1无法启动问题解决方法

问题:最近AndroidStudio4.1正式版本发布,点了更新提示先更新插件,下载完后重启AndroidStudio,结果无法启动了,报如下错误:

java.lang.NoSuchMethodError: com.intellij.ide.plugins.PluginManagerCore.getEnabledPluginRawList()Ljava/util/List;

解决:
windows:删除 ...\.AndroidStudio4.0\config\plugins 下面的 marketplace 文件夹,可以启动但没有更新到4.1成功。
mac电脑可参考:https://blog.csdn.net/bingjianIT/article/details/109142316

3.9 AndroidStudio 启动模拟器问题

3.9.1 Emulator: glTexImage2D: got err 类错误的解决方法

打开模拟器 Settings ,选择 Advanced:
OpenGL ES Renderer 改为 SwiftShader
OpenGL ES API level 改为 Renderer maximum

打开模拟器设置
修改 Settings--Advanced

参考:android - glTexImage2D: got err pre :( 0x506 internal 0x1908 format 0x1908 type 0x1401 - Stack Overflow


持续更新中...

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 157,298评论 4 360
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,701评论 1 290
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 107,078评论 0 237
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,687评论 0 202
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,018评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,410评论 1 211
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,729评论 2 310
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,412评论 0 194
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,124评论 1 239
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,379评论 2 242
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,903评论 1 257
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,268评论 2 251
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,894评论 3 233
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,014评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,770评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,435评论 2 269
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,312评论 2 260

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 170,544评论 25 707
  • afinalAfinal是一个android的ioc,orm框架 https://github.com/yangf...
    passiontim阅读 15,072评论 2 44
  • 今天我像所有电视剧里面的职场小白一样,因为工作的原因哭的稀里哗啦。我一直坚信要与人为善,当我被别人恶意相待时我就觉...
    我是你的大大大阅读 183评论 0 0
  • 很多時候 任性自我 很多時候 在執拗中撕扯 有人說 偏執是一種錯 會讓人想逃離想躲 有人說 去我執 才會...
    紫色茜茜阅读 159评论 0 0
  • 急了 心口堵住了一般,好难受。 今天老邻居来我家找我妈聊天。 说着我的青梅,现在生活的极其美好。有一双大小儿子,漂...
    JK_Joe阅读 178评论 0 0