2024年安卓最新Android AIDL 传递对象(Parceable),2024年最新android界面开发实验报告

最后

感谢您的阅读,在文末给大家准备一个福利。本人从事Android开发已经有十余年,算是一名资深的移动开发架构师了吧。根据我的观察发现,对于很多初中级Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。

所以在此将我十年载,从萌新小白一步步成长为Android移动开发架构师的学习笔记,从Android四大组件到手写实现一个架构设计,我都有一一的对应笔记为你讲解。

当然我也为你们整理好了百度、阿里、腾讯、字节跳动等等互联网超级大厂的历年面试真题集锦。这也是我这些年来养成的习惯,一定要学会把好的东西,归纳整理,然后系统的消化吸收,这样才能极大的提高学习效率和成长进阶。碎片、零散化的东西,我觉得最没有价值的。就好比你给我一张扑克牌,我只会觉得它是一张废纸,但如果你给我一副扑克牌,它便有了它的价值。这和我们收集资料就要收集那些系统化的,是一个道理。

最后,赠与大家一句诗,共勉!

不驰于空想,不骛于虚声。不忘初心,方得始终。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

private long id;

private String title;

private String album;

private int duration;

private long size;

private String artist;

private String url;

private String displayName;

public MusicInfo(long id, String title, String album, int duration, long size, String artist,

String url, String displayName) {

this.id = id;

this.title = title;

this.album = album;

this.duration = duration;

this.size = size;

this.artist = artist;

this.url = url;

this.displayName = displayName;

}

public MusicInfo(){

}

protected MusicInfo(Parcel in) {

id = in.readLong();

title = in.readString();

album = in.readString();

duration = in.readInt();

size = in.readLong();

artist = in.readString();

url = in.readString();

displayName = in.readString();

}

//必须提供一个名为CREATOR的static final属性 该属性需要实现android.os.Parcelable.Creator接口

public static final Creator CREATOR = new Creator() {

@Override

public MusicInfo createFromParcel(Parcel in) {

return new MusicInfo(in);

}

@Override

public MusicInfo[] newArray(int size) {

return new MusicInfo[size];

}

};

public MusicInfo(long id, String title) {

this.id=id;

this.title=title;

}

@Override

public int describeContents() {

return 0;

}

@Override

public void writeToParcel(Parcel dest, int flags) {

dest.writeLong(id);

dest.writeString(title);

dest.writeString(album);

dest.writeInt(duration);

dest.writeLong(size);

dest.writeString(artist);

dest.writeString(url);

dest.writeString(displayName);

}

public void readFromParcel(Parcel reply) {

id=reply.readLong();

title=reply.readString();

album=reply.readString();

duration=reply.readInt();

size=reply.readLong();

artist=reply.readString();

url=reply.readString();

displayName=reply.readString();

}

}

接下来看 writeToParcel 和 readFromParcel 方法,需要注意的是 writeToParcel 和 readFromParcel 方法读写的顺序是一一对应的。

这里有一点要提醒大家的是 AndroidStudio 中,我们通过插件会自动帮我们生成 writeToParcel 方法及 CREATOR,通常 readFromParcel 方法是不会自动生成的,需要我们自己手动编写,不然会编译不过。

@Override

public void writeToParcel(Parcel dest, int flags) {

dest.writeLong(id);

dest.writeString(title);

dest.writeString(album);

dest.writeInt(duration);

dest.writeLong(size);

dest.writeString(artist);

dest.writeString(url);

dest.writeString(displayName);

}

public void readFromParcel(Parcel reply) {

id=reply.readLong();

title=reply.readString();

album=reply.readString();

duration=reply.readInt();

size=reply.readLong();

artist=reply.readString();

url=reply.readString();

displayName=reply.readString();

}

注意了,接下来我们需要写一个 MusicInfo.aidl 文件

package xj.musicserver;

// Declare any non-default types here with import statements

parcelable MusicInfo;

指定包名,并声明 MusicInfo 是 parcelable,注意 parcelable 是小写的 p,不是大写的 P。这是一个规范,google 官方指定需要的。同时 MusicInfo.aidl 和 MusicInfo.java 需要放置在同个包中。

关于怎样在 AndroidStudio 中配置 aidl 的可以参考我的这一篇博客。AndroidStudio 引用 aidl 文件的两种方法

第二步编写一个 Service,实现接口,处理客户端的请求,并将 binder 返回回去;

IPlayService.Stub mIPlayService=new IPlayService.Stub() {

@Override

public void play(String name, final IPlayListener iPlayListener) throws RemoteException {

MusicTask musicTask = new MusicTask(getApplicationContext(), name, “”);

musicTask.setIResultListener(new MusicTask.IResultListener() {

@Override

public void onSuccess(MusicInfo musicInfo) {

try {

iPlayListener.onSuccess(0,musicInfo);

} catch (RemoteException e) {

e.printStackTrace();

}

}

@Override

public void onFail(int code, MusicInfo musicInfo) {

try {

iPlayListener.onError(0);

} catch (RemoteException e) {

e.printStackTrace();

}

}

});

musicTask.execute();

}

};

@Nullable

@Override

public IBinder onBind(Intent intent) {

LogUtil.i(TAG, "onBind: intent = " +intent.toString());

return mIPlayService;

}

这里我们所做的工作就是到数据库里面查询看是否有相应的歌曲,如果有,通过 aidl 回调,告诉客户端我们查找成功,调用 onSuccess 方法,没有找到,调用客户端的 onError 方法。

package xj.musicserver;

import android.content.ContentResolver;

import android.content.Context;

import android.database.Cursor;

import android.net.Uri;

import android.os.AsyncTask;

import android.provider.MediaStore;

import android.support.annotation.NonNull;

import android.text.TextUtils;

import android.util.Log;

import java.util.ArrayList;

/**

  • @author meitu.xujun on 2017/10/17

  • @version 0.1

*/

public class MusicTask extends AsyncTask<Void,Void,Integer> {

// 这里只贴出主要代码,详细代码可到文章的末尾下载。

public MusicTask(Context context, String name, String artist){

mContext = context.getApplicationContext();

mName = name;

mArtist = artist;

}

@Override

protected Integer doInBackground(Void… params) {

LogUtil.i(TAG,“doInBackground: mName=”+mName +" mArtist"+mArtist);

mResult = “”;

ContentResolver contentResolver = mContext.getContentResolver();

Cursor cursor;

if (TextUtils.isEmpty(mArtist)) {

cursor = contentResolver.query(contentUri, projection, where_title, new String[]{getFixName(mName)},null);

}else{

cursor=contentResolver.query(contentUri, projection,

where_title_and_artist, new String[]{getFixName(mName),getFixName(mArtist)},null);

if(cursor==null || cursor.getCount()<=0){

cursor = contentResolver.query(contentUri, projection,

where_title, new String[]{getFixName(mName)},null);

}

}

if(cursor==null || cursor.getCount()<=0){

return RESULT_FAIL_MUSIC_NULL;

}

int displayNameCol = cursor.getColumnIndex(MediaStore.Audio.Media.DISPLAY_NAME);

int albumCol = cursor.getColumnIndex(MediaStore.Audio.Media.ALBUM);

int idCol = cursor.getColumnIndex(MediaStore.Audio.Media._ID);

int durationCol = cursor.getColumnIndex(MediaStore.Audio.Media.DURATION);

int sizeCol = cursor.getColumnIndex(MediaStore.Audio.Media.SIZE);

int artistCol = cursor.getColumnIndex(MediaStore.Audio.Media.ARTIST);

int urlCol = cursor.getColumnIndex(MediaStore.Audio.Media.DATA);

int titleCol = cursor.getColumnIndex(MediaStore.Audio.Media.TITLE);

mMusicInfos = new ArrayList<>();

String songName=“”;

while (cursor.moveToNext()){

songName = cursor.getString(titleCol);

MusicInfo musicInfo = getMusicInfo(cursor, displayNameCol, albumCol, idCol,

durationCol, sizeCol, artistCol, urlCol,titleCol);

mMusicInfos.add(musicInfo);

if(songName.equals(mName)){

mResult =mName;

mMusicInfo=musicInfo;

break;

}

}

if(mMusicInfo==null){

mMusicInfo =mMusicInfos.get(0);

}

return RESULT_SUCUESS;

}

@Override

protected void onPostExecute(Integer result) {

super.onPostExecute(result);

Log.i(TAG, “onPostExecute: result =” +result);

if(mIResultListener==null){

return;

}

if(result==RESULT_SUCUESS){

mIResultListener.onSuccess(mMusicInfo);

}else{

mIResultListener.onFail(result,mMusicInfo);

}

}

Android进阶资料

以下的资料是近年来,我和一些朋友面试收集整理了很多大厂的面试真题和资料,还有来自如阿里、小米、爱奇艺等一线大厂的大牛整理的架构进阶资料。希望可以帮助到大家。

Android进阶核心笔记

百万年薪必刷面试题

最全Android进阶学习视频

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

.onSuccess(mMusicInfo);

}else{

mIResultListener.onFail(result,mMusicInfo);

}

}

Android进阶资料

以下的资料是近年来,我和一些朋友面试收集整理了很多大厂的面试真题和资料,还有来自如阿里、小米、爱奇艺等一线大厂的大牛整理的架构进阶资料。希望可以帮助到大家。

Android进阶核心笔记

[外链图片转存中…(img-8tekeOvV-1715008188397)]

百万年薪必刷面试题

[外链图片转存中…(img-3bnTcqmq-1715008188398)]

最全Android进阶学习视频

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 17
    点赞
  • 30
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值