摘要:本人面試現(xiàn)在的公司就是來做安卓的,但是人手不夠讓我做做前端正文安卓現(xiàn)在比較熱火的四大框架應(yīng)該是,,,。
前言這段時(shí)間我已經(jīng)在一個(gè)公司實(shí)習(xí)了,雖然工資不高,我并沒有太多的介意。但是有一點(diǎn)是我不舒服的,負(fù)責(zé)我們部門的經(jīng)理助理似乎看不起實(shí)習(xí)生,對(duì)我有些輕蔑。主要是他技術(shù)也不是很牛逼那種。整個(gè)公司用的技術(shù)還是比較落后那種。當(dāng)我推薦Vue時(shí)竟然說這種別人封裝好的js不太好。當(dāng)時(shí)內(nèi)心有一千條草泥馬奔跑,那你為什么還要用jQuery?so,我打算跳槽,跳去更有發(fā)展的公司,畢竟我現(xiàn)在是實(shí)習(xí),主要還是希望得到成長(zhǎng)~于是接到了一個(gè)安卓面試,所以特地前來寫下這篇文章進(jìn)行復(fù)習(xí)。(本人面試現(xiàn)在的公司就是來做安卓的,但是人手不夠讓我做做前端)
正文安卓現(xiàn)在比較熱火的四大框架應(yīng)該是Rxjava,retrofit,Okhttp,Dagger。于是我去翻各種博文自己嘗試著去寫一個(gè)小demo。首先來看看我們這個(gè)小demo的一個(gè)效果圖:
可以看到我們這個(gè)demo就是調(diào)用了一個(gè)接口,有一個(gè)滑動(dòng)列表,有圖片,有的是視頻,點(diǎn)進(jìn)去可以觀看視頻。一個(gè)很簡(jiǎn)單的App DEMO。好了廢話不多說,讓我們開始吧! 第一,先來看看我們整個(gè)項(xiàng)目引入的依賴:
implementation fileTree(dir: libs, include: [*.jar])
implementation com.android.support:appcompat-v7:27.1.0
implementation com.android.support:recyclerview-v7:27.1.0
implementation com.android.support:cardview-v7:27.1.0
implementation com.android.support.constraint:constraint-layout:1.0.2
testImplementation junit:junit:4.12
androidTestImplementation com.android.support.test:runner:1.0.1
androidTestImplementation com.android.support.test.espresso:espresso-core:3.0.1
//最主要的是下面這幾條
implementation "io.reactivex.rxjava2:rxjava:2.1.10"
implementation io.reactivex.rxjava2:rxandroid:2.0.2
implementation com.squareup.retrofit2:retrofit:2.4.0
implementation com.squareup.retrofit2:converter-gson:2.3.0
implementation com.squareup.retrofit2:adapter-rxjava2:2.3.0
implementation com.github.bumptech.glide:glide:4.6.1
其次來看看我們整個(gè)項(xiàng)目的一個(gè)結(jié)構(gòu)圖(MVP架構(gòu),直接用MVPHelper生成的):
現(xiàn)在來介紹一下整個(gè)項(xiàng)目結(jié)構(gòu):
contract包里面放的是接口,用來約束后面的開發(fā);
model里面就是實(shí)體類對(duì)象和提供后臺(tái)數(shù)據(jù)的請(qǐng)求接口;
presenter,很明顯就是MVP中的P,主持者類;
view里面本來我是要把a(bǔ)ctivity也放進(jìn)去的,但是沒放,里面就放了一個(gè)RecyclerView的適配器。
utils里面是一個(gè)創(chuàng)建Retrofit的工廠類
第二,在主界面布局文件中加入一個(gè)RcyclerView和一個(gè)ProgressBar,根布局我們直接使用的是ConstraintLayout:
"http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="top.cyixlq.rxtestapp.MainActivity">
"@+id/pro"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:visibility="gone"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
"@+id/rec"
android:layout_width="0dp"
android:layout_height="0dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
接著就是創(chuàng)建一個(gè)RecyclerView的單個(gè)條目布局文件joker_rec_item.xml,根布局我們用的是CardView:
"http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
xmlns:app="http://schemas.android.com/apk/res-auto"
app:cardCornerRadius="5dp"
app:cardElevation="3dp"
app:contentPadding="5dp"
android:layout_marginBottom="5dp">
"@+id/title"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="標(biāo)題"
android:textSize="15sp"/>
"@+id/content"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="10sp"
android:layout_marginTop="20dp"
android:text="內(nèi)容"/>
"@+id/img"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="40dp"/>
可以看出這個(gè)布局很簡(jiǎn)單,就是標(biāo)題內(nèi)容都是TextView,還有一個(gè)ImageView來展示圖片。 然后把約束類建立起來,JokerContract:
public interface JokerContract {
interface Model {
void getJokerList(String type,String page,Observer observer);
}
interface View {
void showJokerList(List list);
void getJokerListFinish();
void getJokerListErro(String msg);
}
interface Presenter {
void getJokerList(String type,String page);
}
}
第三,完成我們的Retrofit的工廠類RetrofitFactory:
public class RetrofitFactory {
private final static String BASE_URL="https://www.apiopen.top/";
private static final long TIMEOUT = 30;
private static JokerApiService jokerApiService=new Retrofit.Builder()
.baseUrl(BASE_URL)
//添加Gson轉(zhuǎn)換器
.addConverterFactory(GsonConverterFactory.create())
//// 添加Retrofit到RxJava的轉(zhuǎn)換器
.addCallAdapterFactory(RxJava2CallAdapterFactory.create())
.build()
.create(JokerApiService.class);
public static JokerApiService getJokerApiService(){
return jokerApiService;
}
}
第四,完成M層,先完成接口api的請(qǐng)求,在model包中的apiservices包中新建一個(gè)接口,JokerApiService:
public interface JokerApiService {
@GET("satinApi")
Observable getJokerList(@Query("type")String type,@Query("page")String page);
}
然后利用GsonFormat創(chuàng)建實(shí)體類Joker(內(nèi)容有點(diǎn)長(zhǎng),但是其實(shí)我們會(huì)用到的屬性不多):
public class Joker {
private int code;
private String msg;
private List data;
//get和set省略
public static class DataBean {
private String type;
private String text;
private String user_id;
private String name;
private String screen_name;
private String profile_image;
private String created_at;
private Object create_time;
private String passtime;
private String love;
private String hate;
private String comment;
private String repost;
private String bookmark;
private String bimageuri;
private Object voiceuri;
private Object voicetime;
private Object voicelength;
private String status;
private String theme_id;
private String theme_name;
private String theme_type;
private String videouri;
private int videotime;
private String original_pid;
private int cache_version;
private String playcount;
private String playfcount;
private String cai;
private Object weixin_url;
private String image1;
private String image2;
private boolean is_gif;
private String image0;
private String image_small;
private String cdn_img;
private String width;
private String height;
private String tag;
private int t;
private String ding;
private String favourite;
private Object top_cmt;
private Object themes;
//get和set省略
}
}
接著就是把Model類建起來,JokerModel:
public class JokerModel implements JokerContract.Model {
@Override
public void getJokerList(String type, String page, Observer observer) {
JokerApiService apiService=RetrofitFactory.getJokerApiService(); //獲取接口
Observable observable= apiService.getJokerList(type,page); //利用接口獲取數(shù)據(jù)
observable.subscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread())
.subscribe(observer); //在IO線程執(zhí)行,發(fā)送結(jié)果到主線程
}
}
第五,完成P層,新建JokerPresenter:
public class JokerPresenter implements JokerContract.Presenter {
JokerContract.Model mModel;
JokerContract.View mView;
public JokerPresenter(JokerContract.View view){
mModel=new JokerModel();
this.mView=view;
}
@Override
public void getJokerList(String type, String page) {
mModel.getJokerList(type,page,new Observer(){
@Override
public void onSubscribe(Disposable d) {
}
@Override
public void onNext(Joker joker) {
List list=joker.getData();
mView.showJokerList(list); //視圖層將列表結(jié)果展示出來
}
@Override
public void onError(Throwable e) {
mView.getJokerListErro(e.getMessage()); //視圖層將錯(cuò)誤信息顯示出來
}
@Override
public void onComplete() {
mView.getJokerListFinish(); //視圖層完成數(shù)據(jù)獲取狀態(tài)
}
});
}
}
第六,完成V層,也就是視圖層,在第一步中我們已經(jīng)把各種布局寫完了,這里我們主要寫activity和RecyclerView的適配器。先來寫適配器,JokerAdapter:
public class JokerAdapter extends RecyclerView.Adapter {
private List list;
private Context mContext;
private OnItemClickListener mOnItemClickListener;
public JokerAdapter(List list,Context context){
this.list=list;
this.mContext=context;
}
@Override
public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View view= LayoutInflater.from(parent.getContext()).inflate(R.layout.joker_rec_item,parent,false);
MyViewHolder viewHolder=new MyViewHolder(view);
return viewHolder;
}
@Override
public void onBindViewHolder(MyViewHolder holder, final int position) {
holder.txt_title.setText(list.get(position).getName());
holder.txt_content.setText(list.get(position).getText());
Glide.with(mContext).load(list.get(position).getBimageuri()).into(holder.img);
if(mOnItemClickListener!=null){
holder.itemView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
mOnItemClickListener.onClick(position);
}
});
}
}
@Override
public int getItemCount() {
return list.size();
}
//本來這個(gè)內(nèi)部類是沒有設(shè)置成靜態(tài)的,但是聽說不是靜態(tài)的會(huì)造成內(nèi)存泄漏?還望大神給我這個(gè)小白解答一下,感激不盡!
static class MyViewHolder extends RecyclerView.ViewHolder{
TextView txt_title;
TextView txt_content;
ImageView img;
public MyViewHolder(View itemView) {
super(itemView);
txt_title=itemView.findViewById(R.id.title);
txt_content=itemView.findViewById(R.id.content);
img=itemView.findViewById(R.id.img);
}
}
//點(diǎn)擊事件接口
public interface OnItemClickListener{
void onClick( int position);
}
//設(shè)置點(diǎn)擊事件
public void setOnItemClickListener(OnItemClickListener onItemClickListener ){
this.mOnItemClickListener=onItemClickListener;
}
}
以上就是我的適配器的所有代碼,其中有個(gè)問題想請(qǐng)教諸位大神,還請(qǐng)大神不吝賜教:ViewHolder那個(gè)內(nèi)部類是沒有設(shè)置成靜態(tài)的,但是聽說不是靜態(tài)的會(huì)造成內(nèi)存泄漏?還望大神給我這個(gè)小白解答一下,感激不盡! 接著就是MainActivity:
public class MainActivity extends AppCompatActivity implements JokerContract.View{
public static final String TAG="MainActivity";
JokerPresenter mPresenter;
JokerAdapter mAdapter;
List mList;
private RecyclerView mRecyclerView;
private ProgressBar mProgressBar;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mPresenter=new JokerPresenter(this);
mRecyclerView=findViewById(R.id.rec);
mProgressBar=findViewById(R.id.pro);
initData();
mPresenter.getJokerList("29","1"); //我們只獲取了第一頁的數(shù)據(jù)
}
private void initData(){
mList=new ArrayList<>();
mAdapter=new JokerAdapter(mList,this);
mAdapter.setOnItemClickListener(new JokerAdapter.OnItemClickListener() { //設(shè)置點(diǎn)擊事件
@Override
public void onClick(int position) {
String url=mList.get(position).getVideouri(); //獲取對(duì)應(yīng)的視頻鏈接,并且通過intent攜帶鏈接進(jìn)行跳轉(zhuǎn)
Intent intent=new Intent(MainActivity.this,VideoPlayActivity.class);
intent.putExtra("url",url);
startActivity(intent);
}
});
LinearLayoutManager manager=new LinearLayoutManager(MainActivity.this);
mRecyclerView.setLayoutManager(manager);
mRecyclerView.setAdapter(mAdapter);
}
@Override
public void showJokerList(List list) {
mProgressBar.setVisibility(View.VISIBLE);
mList.addAll(list);
mAdapter.notifyDataSetChanged();
}
@Override
public void getJokerListFinish() {
mProgressBar.setVisibility(View.GONE);
}
@Override
public void getJokerListErro(String msg) {
Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();
Log.e(TAG,msg);
}
}
第七,就是完成VideoActivity啦,我直接用的VideoView來播放網(wǎng)絡(luò)視頻,先把VideoActivity布局文件寫好,activity_video_play.xml:
"http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="top.cyixlq.rxtestapp.VideoPlayActivity">
"@+id/video"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
我就是直接放的一個(gè)VideoView。然后編寫activity代碼,讓VideoView播放網(wǎng)絡(luò)視頻:
public class VideoPlayActivity extends AppCompatActivity {
private VideoView mVideoVIew;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_video_play);
mVideoVIew=findViewById(R.id.video);
mVideoVIew.setMediaController(new MediaController(this));
mVideoVIew.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {
@Override
public void onCompletion(MediaPlayer mp) {
Toast.makeText(VideoPlayActivity.this, "播放完成了", Toast.LENGTH_SHORT).show();
}
});
startPlay();
}
private void startPlay(){
Intent intent=getIntent();
String url=intent.getStringExtra("url");
if(null!=url) {
mVideoVIew.setVideoURI(Uri.parse(url));
mVideoVIew.start();
}
}
}
這樣就能輕松實(shí)現(xiàn)VideoView播放網(wǎng)絡(luò)視頻啦。然后我們整個(gè)Demo也就這樣寫完了哦!
后記這只是一個(gè)簡(jiǎn)單的Demo,個(gè)人也是剛開始接觸不久,如果還有什么地方寫的不對(duì),還望各位大神指教,本人不勝感激,求大神帶飛!本Demo的GitHub地址:github.com/cyixlq/RxTe…
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/7058.html
摘要:完成網(wǎng)絡(luò)請(qǐng)求添加依賴。新的配置獲取對(duì)象請(qǐng)求在新的線程中執(zhí)行請(qǐng)求完成后在線程中執(zhí)行保存用戶信息到本地最后在主線程中執(zhí)行請(qǐng)求失敗請(qǐng)求成功形式的時(shí)候,把請(qǐng)求封裝進(jìn),在請(qǐng)求結(jié)束后調(diào)用或在請(qǐng)求失敗后調(diào)用。 showImg(https://segmentfault.com/img/remote/1460000018253018); 前言 本文基于RxJava、Retrofit的使用,若是對(duì)RxJa...
閱讀 740·2023-04-25 19:43
閱讀 3986·2021-11-30 14:52
閱讀 3816·2021-11-30 14:52
閱讀 3873·2021-11-29 11:00
閱讀 3809·2021-11-29 11:00
閱讀 3907·2021-11-29 11:00
閱讀 3584·2021-11-29 11:00
閱讀 6197·2021-11-29 11:00