290dp
291dp
292dp
293dp
294dp
295dp
296dp
297dp
298dp
299dp
300dp
301dp
302dp
303dp
304dp
305dp
306dp
307dp
308dp
309dp
310dp
311dp
312dp
313dp
314dp
315dp
316dp
317dp
318dp
319dp
320dp
321dp
322dp
323dp
324dp
325dp
326dp
327dp
328dp
329dp
330dp
331dp
332dp
333dp
334dp
335dp
336dp
337dp
338dp
339dp
340dp
341dp
342dp
343dp
344dp
345dp
346dp
347dp
348dp
349dp
350dp
351dp
352dp
353dp
354dp
355dp
356dp
357dp
358dp
359dp
360dp
365dp
370dp
399dp
400dp
410dp
420dp
412dp
422dp
472dp
500dp
520dp
550dp
560dp
600dp
640dp
720dp
6sp
7sp
8sp
9sp
10sp
11sp
12sp
13sp
14sp
15sp
16sp
17sp
18sp
19sp
20sp
21sp
22sp
23sp
24sp
25sp
26sp
27sp
28sp
29sp
30sp
31sp
32sp
33sp
34sp
35sp
36sp
37sp
38sp
40sp
42sp
48sp
12dp
在修改布局之前,先准备好需要的资源,首先是两个图标
在drawable中新建search_et_bg.xml,里面的代码如下:
<?xml version="1.0" encoding="utf-8"?>这是设置输入框的背景
然后再新建一个cursor_style.xml,里面的代码如下:
<?xml version="1.0" encoding="utf-8"?><shape xmlns:android=“http://schemas.android.com/apk/res/android”
android:shape=“rectangle”>
这是设置输入框的光标颜色。
下面可以写布局了。
<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android=“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”
android:fitsSystemWindows=“true”
android:orientation=“vertical”
tools:context=“.MainActivity”>
<com.google.android.material.appbar.MaterialToolbar
android:id=“@+id/toolbar”
android:layout_width=“match_parent”
android:layout_height=“?attr/actionBarSize”
android:background=“@color/colorPrimaryDark”>
<TextView
android:layout_width=“wrap_content”
android:layout_height=“wrap_content”
android:layout_gravity=“center”
android:text=“垃圾分类”
android:textColor=“@color/white”
android:textSize=“18sp” />
</com.google.android.material.appbar.MaterialToolbar>
<LinearLayout
android:layout_width=“match_parent”
android:layout_height=“wrap_content”
android:background=“@color/white”>
<RelativeLayout
android:layout_width=“match_parent”
android:layout_height=“36dp”
android:layout_marginStart=“@dimen/sp_16”
android:layout_marginTop=“@dimen/dp_12”
android:layout_marginEnd=“@dimen/dp_16”
android:layout_marginBottom=“@dimen/dp_12”
android:background=“@drawable/search_et_bg”
android:orientation=“horizontal”>
<ImageView
android:id=“@+id/iv_search”
android:layout_width=“@dimen/dp_16”
android:layout_height=“@dimen/dp_16”
android:layout_centerVertical=“true”
android:layout_marginStart=“@dimen/dp_12”
android:layout_marginEnd=“@dimen/dp_8”
app:srcCompat=“@mipmap/icon_search” />
<EditText
android:id=“@+id/et_goods”
android:layout_width=“wrap_content”
android:layout_height=“match_parent”
android:layout_toStartOf=“@+id/iv_clear”
android:layout_toEndOf=“@+id/iv_search”
android:background=“@null”
android:hint=“请输入物品”
android:imeOptions=“actionSearch”
android:singleLine=“true”
android:textColor=“@color/black”
android:textColorHint=“@color/hint_color”
android:textCursorDrawable=“@drawable/cursor_style”
android:textSize=“@dimen/sp_14” />
<ImageView
android:id=“@+id/iv_clear”
android:layout_width=“@dimen/dp_36”
android:layout_height=“@dimen/dp_36”
android:layout_alignParentEnd=“true”
android:padding=“@dimen/dp_10”
android:src=“@mipmap/icon_clear”
android:visibility=“invisible” />
<androidx.recyclerview.widget.RecyclerView
android:id=“@+id/rv_result”
android:layout_width=“match_parent”
android:layout_height=“match_parent”
android:overScrollMode=“never”
android:scrollbars=“none” />
下面设置接口的访问地址,打开mvplibrary下的NetworkApi类,修改正式环境的访问地址
mBaseUrl = “http://api.tianapi.com”;
如下图所示
下面回到app模块,在com.llw.goodtrash包下新建NetworkRequiredInfo类,里面代码如下;
package com.llw.goodtrash;
import android.app.Application;
import com.llw.mvplibrary.BuildConfig;
import com.llw.mvplibrary.network.INetworkRequiredInfo;
/**
-
网络访问信息
-
@author llw
*/
public class NetworkRequiredInfo implements INetworkRequiredInfo {
private Application application;
public NetworkRequiredInfo(Application application){
this.application = application;
}
/**
- 版本名
*/
@Override
public String getAppVersionName() {
return BuildConfig.VERSION_NAME;
}
/**
- 版本号
*/
@Override
public String getAppVersionCode() {
return String.valueOf(BuildConfig.VERSION_CODE);
}
/**
- 是否为debug
*/
@Override
public boolean isDebug() {
return BuildConfig.DEBUG;
}
/**
- 应用全局上下文
*/
@Override
public Application getApplicationContext() {
return application;
}
}
之后再新建一个TrashApplication类,代码如下:
package com.llw.goodtrash;
import com.llw.mvplibrary.BaseApplication;
import com.llw.mvplibrary.network.NetworkApi;
/**
- 自定义Application
*/
public class TrashApplication extends BaseApplication {
@Override
public void onCreate() {
super.onCreate();
//初始化网络框架
NetworkApi.init(new NetworkRequiredInfo(this));
}
}
自定义的Application需要到AndroidManifest.xml中去配置才会生效,如下图所示:
现在你就可以访问网络了,这虽然看起来是单独使用Okhttp要麻烦,但实际上是要方便很多的,你可能现在不明白,以后你会懂得。
我在天行API中请求接口使用的KEY,这里我新建了一个全局常量类,放在里面,在com.llw.goodtrash包下新建一个utils包。包下的代码如下:
package com.llw.goodtrash.utils;
/**
-
全局常量
-
@author llw
-
@date 2021/3/30 15:14
*/
public class Constant {
/**
- 垃圾分类的key 请替换为自己的
*/
public static final String KEY = “783da68c7ea7e10fcd259db651cc550b”;
/**
- 请求成功码
*/
public static final int SUCCESS_CODE = 200;
}
之前通过返回的数据可以生成一个返回数据实体,在com.llw.goodtrash包下新建一个model包,包下新建一个TrashResponse类,代码如下:
package com.llw.goodtrash.model;
import java.util.List;
/**
- 垃圾分类返回数据
*/
public class TrashResponse {
/**
-
code : 200
-
msg : success
-
newslist : [{“name”:“羽毛球”,“type”:3,“aipre”:0,“explain”:“干垃圾即其它垃圾,指除可回收物、有害垃圾、厨余垃圾(湿垃圾)以外的其它生活废弃物。”,“contain”:“常见包括砖瓦陶瓷、渣土、卫生间废纸、猫砂、污损塑料、毛发、硬壳、一次性制品、灰土、瓷器碎片等难以回收的废弃物”,“tip”:“尽量沥干水分;难以辨识类别的生活垃圾都可以投入干垃圾容器内”},{“name”:“羽毛球拍”,“type”:0,“aipre”:0,“explain”:“可回收垃圾是指适宜回收、可循环利用的生活废弃物。”,“contain”:“常见包括各类废金属、玻璃瓶、易拉罐、饮料瓶、塑料玩具、书本、报纸、广告单、纸板箱、衣服、床上用品、电子产品等”,“tip”:“轻投轻放;清洁干燥,避免污染,费纸尽量平整;立体包装物请清空内容物,清洁后压扁投放;有尖锐边角的、应包裹后投放”}]
*/
private int code;
private String msg;
private List newslist;
public int getCode() {
return code;
}
public void setCode(int code) {
this.code = code;
}
public String getMsg() {
return msg;
}
public void setMsg(String msg) {
this.msg = msg;
}
public List getNewslist() {
return newslist;
}
public void setNewslist(List newslist) {
this.newslist = newslist;
}
public static class NewslistBean {
/**
-
name : 羽毛球
-
type : 3
-
aipre : 0
-
explain : 干垃圾即其它垃圾,指除可回收物、有害垃圾、厨余垃圾(湿垃圾)以外的其它生活废弃物。
-
contain : 常见包括砖瓦陶瓷、渣土、卫生间废纸、猫砂、污损塑料、毛发、硬壳、一次性制品、灰土、瓷器碎片等难以回收的废弃物
-
tip : 尽量沥干水分;难以辨识类别的生活垃圾都可以投入干垃圾容器内
*/
private String name;
private int type;
private int aipre;
private String explain;
private String contain;
private String tip;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getType() {
return type;
}
public void setType(int type) {
this.type = type;
}
public int getAipre() {
return aipre;
}
public void setAipre(int aipre) {
this.aipre = aipre;
}
public String getExplain() {
return explain;
}
public void setExplain(String explain) {
this.explain = explain;
}
public String getContain() {
return contain;
}
public void setContain(String contain) {
this.contain = contain;
}
public String getTip() {
return tip;
}
public void setTip(String tip) {
this.tip = tip;
}
}
}
下面设置api接口,在com.llw.goodtrash包下新建一个api包,包下新建一个ApiService接口。
package com.llw.goodtrash.api;
import com.llw.goodtrash.model.TrashResponse;
import io.reactivex.Observable;
import retrofit2.http.GET;
import retrofit2.http.Query;
import static com.llw.goodtrash.utils.Constant.KEY;
/**
-
API接口
-
@author llw
-
@date 2021/3/30 15:13
*/
public interface ApiService {
/**
-
垃圾分类
-
@param word 物品名
-
@return TrashResponse 结果实体
*/
@GET(“/txapi/lajifenlei/index?key=” + KEY)
Observable searchGoods(@Query(“word”) String word);
}
这里通过Observable观察TrashResponse,当获取到返回数据时,会解析成TrashResponse,就不需要手动去转了,而searchGoods标识方法名,后面的@Query(“word”) String word表示请求API时会把word作为参数拼接到请求地址后面。
下面构建页面访问网络的订阅类
在com.llw.goodtrash包下新建一个contract包,这个包下新建一个MainContract类。
package com.llw.goodtrash.contract;
import android.annotation.SuppressLint;
import com.llw.goodtrash.api.ApiService;
import com.llw.goodtrash.model.TrashResponse;
import com.llw.mvplibrary.base.BasePresenter;
import com.llw.mvplibrary.base.BaseView;
import com.llw.mvplibrary.network.NetworkApi;
import com.llw.mvplibrary.network.observer.BaseObserver;
/**
- 主页面访问网络
*/
public class MainContract {
public static class MainPresenter extends BasePresenter {
/**
-
搜索物品
-
@param word 物品名
*/
@SuppressLint(“CheckResult”)
public void searchGoods(String word) {
ApiService service = NetworkApi.createService(ApiService.class);
service.searchGoods(word).compose(NetworkApi.applySchedulers(new BaseObserver() {
@Override
public void onSuccess(TrashResponse groupResponse) {
if (getView() != null) {
getView().getSearchResponse(groupResponse);
}
}
@Override
public void onFailure(Throwable e) {
if (getView() != null) {
getView().getSearchResponseFailed(e);
}
}
}));
}
}
public interface MainView extends BaseView {
/**
-
搜索物品返回
-
@param response
*/
void getSearchResponse(TrashResponse response);
/**
-
搜索物品异常返回
-
@param throwable
*/
void getSearchResponseFailed(Throwable throwable);
}
}
下面回到MainActivity,先删掉上一篇添加的代码,然后修改代码如下图所示:
之后你需要重写五个方法。
@Override
public void initData(Bundle savedInstanceState) {
}
@Override
public int getLayoutId() {
return 0;
}
@Override
protected MainContract.MainPresenter createPresenter() {
return null;
}
@Override
public void getSearchResponse(TrashResponse response) {
}
@Override
public void getSearchResponseFailed(Throwable throwable) {
}
下面会一一修改这五个方法。
在上一篇文章中说到返回的是一个可变数组数据,那么是采用列表来显示,有列表,自然要有列表适配器,而列表适配器里面使用了item的布局。在layout下创建item_search_rv.xml,里面的代码如下:
<?xml version="1.0" encoding="utf-8"?><RelativeLayout xmlns:android=“http://schemas.android.com/apk/res/android”
android:id=“@+id/item_search_goods”
android:layout_width=“match_parent”
android:layout_height=“wrap_content”
android:layout_marginTop=“@dimen/dp_1”
android:background=“@color/white”
android:foreground=“?attr/selectableItemBackground”
android:orientation=“vertical”
android:padding=“@dimen/dp_12”>
<TextView
android:id=“@+id/tv_name”
android:layout_width=“wrap_content”
android:layout_height=“wrap_content”
android:text=“物品名称”
android:textColor=“@color/black”
android:textSize=“@dimen/sp_16” />
<TextView
android:id=“@+id/tv_type”
android:layout_width=“wrap_content”
android:layout_height=“wrap_content”
android:layout_alignParentEnd=“true”
android:text=“垃圾类型”
android:textColor=“@color/black”
android:textSize=“@dimen/sp_14” />
<TextView
android:id=“@+id/tv_explain”
android:layout_width=“wrap_content”
android:layout_height=“wrap_content”
android:layout_below=“@+id/tv_name”
android:layout_marginTop=“@dimen/dp_8”
android:text=“解释”
android:textColor=“@color/hint_color”
android:textSize=“@dimen/sp_14” />
然后写适配器,在com.llw.goodtrash包下新建一个adapter包,包下新建一个SearchGoodsAdapter类,代码如下:
package com.llw.goodtrash.adapter;
import android.widget.TextView;
import androidx.annotation.Nullable;
import com.chad.library.adapter.base.BaseQuickAdapter;
import com.chad.library.adapter.base.BaseViewHolder;
import com.llw.goodtrash.R;
import com.llw.goodtrash.model.TrashResponse;
import java.util.List;
/**
- 搜索物品结果列表适配器
*/
public class SearchGoodsAdapter extends BaseQuickAdapter<TrashResponse.NewslistBean, BaseViewHolder> {
public SearchGoodsAdapter(int layoutResId, @Nullable List<TrashResponse.NewslistBean> data) {
super(layoutResId, data);
}
@Override
protected void convert(BaseViewHolder helper, TrashResponse.NewslistBean item) {
helper.setText(R.id.tv_name, item.getName())
.setText(R.id.tv_explain, item.getExplain())
.addOnClickListener(R.id.item_search_goods);
TextView tvType = helper.getView(R.id.tv_type);
switch (item.getType()) {
case 0:
tvType.setText(“可回收垃圾”);
break;
case 1:
tvType.setText(“有害垃圾”);
break;
case 2:
tvType.setText(“厨余垃圾”);
break;
case 3:
//干垃圾即其他垃圾
tvType.setText(“干垃圾”);
break;
default:
break;
}
}
}
当我们进入这个页面时要完成页面的初始化,之前是写在onCreate中完成的,现在用initData中完成。而布局id在getLayoutId中返回。
@Override
public int getLayoutId() {
return R.layout.activity_main;
}
你的onCreate方法可以删掉了,之后创建变量
private static final String TAG = “MainActivity”;
private EditText etGoods;//输入框
private ImageView ivClear;//清空输入框
private RecyclerView rvResult;//结果显示列表
private List<TrashResponse.NewslistBean> newslistBeanList = new ArrayList<>();//数据列表
private SearchGoodsAdapter searchGoodsAdapter;//结果列表适配器
然后创建一个initView方法。
/**
- 页面初始化
*/
private void initView() {
etGoods = findViewById(R.id.et_goods);
ivClear = findViewById(R.id.iv_clear);
rvResult = findViewById(R.id.rv_result);
//配置适配器 设置布局和数据源
searchGoodsAdapter = new SearchGoodsAdapter(R.layout.item_search_rv, newslistBeanList);
//设置列表的布局管理器
rvResult.setLayoutManager(new LinearLayoutManager(this));
//列表item点击事件
searchGoodsAdapter.setOnItemChildClickListener((adapter, view, position) -> {
showMsg(“点击了” + newslistBeanList.get(position).getName());
});
//设置列表适配器
rvResult.setAdapter(searchGoodsAdapter);
//设置输入监听
etGoods.addTextChangedListener(new TextWatcher() {
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
if (s.toString().length() < 1) {
ivClear.setVisibility(View.INVISIBLE);
} else {
ivClear.setVisibility(View.VISIBLE);
}
}
@Override
public void afterTextChanged(Editable s) {
}
});
//设置动作监听
etGoods.setOnEditorActionListener((v, actionId, event) -> {
if (actionId == EditorInfo.IME_ACTION_SEARCH) {
String word = etGoods.getText().toString().trim();
if (word.isEmpty()) {
showMsg(“请输入物品名”);
} else {
//显示加载弹窗
showLoadingDialog();
//控制输入法
controlInputMethod();
//请求接口
mPresenter.searchGoods(word);
}
return true;
}
return false;
});
//清空输入框内容
ivClear.setOnClickListener(v -> {
controlInputMethod();
etGoods.setText(“”);
});
}
这里面有一个方法controlInputMethod,用于控制输入法是否显示,代码如下:
/**
-
控制输入法
-
当输入法打开时关闭,关闭时弹出
*/
private void controlInputMethod() {
InputMethodManager inputMethodManager = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
inputMethodManager.toggleSoftInput(0, InputMethodManager.HIDE_NOT_ALWAYS);
}
别忘记在initData()方法中调用initView();
返回新的订阅
@Override
protected MainContract.MainPresenter createPresenter() {
return new MainContract.MainPresenter();
}
/**
-
搜索物品返回数据
-
@param response
*/
@Override
学习福利
【Android 详细知识点思维脑图(技能树)】
其实Android开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。
虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。
这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。
由于篇幅有限,这里以图片的形式给大家展示一小部分。
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!
lse {
//显示加载弹窗
showLoadingDialog();
//控制输入法
controlInputMethod();
//请求接口
mPresenter.searchGoods(word);
}
return true;
}
return false;
});
//清空输入框内容
ivClear.setOnClickListener(v -> {
controlInputMethod();
etGoods.setText(“”);
});
}
这里面有一个方法controlInputMethod,用于控制输入法是否显示,代码如下:
/**
-
控制输入法
-
当输入法打开时关闭,关闭时弹出
*/
private void controlInputMethod() {
InputMethodManager inputMethodManager = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
inputMethodManager.toggleSoftInput(0, InputMethodManager.HIDE_NOT_ALWAYS);
}
别忘记在initData()方法中调用initView();
返回新的订阅
@Override
protected MainContract.MainPresenter createPresenter() {
return new MainContract.MainPresenter();
}
/**
-
搜索物品返回数据
-
@param response
*/
@Override
学习福利
【Android 详细知识点思维脑图(技能树)】
[外链图片转存中…(img-vpLhohGE-1714679233945)]
其实Android开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。
虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。
这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。
由于篇幅有限,这里以图片的形式给大家展示一小部分。
[外链图片转存中…(img-C0B0gGRL-1714679233946)]
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!