Android MVVM框架搭建(六)腾讯X5WebView + DrawerLayout + NavigationView

<?xml version="1.0" encoding="utf-8"?>

<layout 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”>

<androidx.constraintlayout.widget.ConstraintLayout

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:fitsSystemWindows=“true”

tools:context=“.ui.activity.WebActivity”>

<com.tencent.smtt.sdk.WebView

android:id=“@+id/webView”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:fitsSystemWindows=“true”

app:layout_constraintBottom_toBottomOf=“parent”

app:layout_constraintEnd_toEndOf=“parent”

app:layout_constraintStart_toStartOf=“parent”

app:layout_constraintTop_toTopOf=“parent” />

</androidx.constraintlayout.widget.ConstraintLayout>

然后在WebActivity中增加如下代码,用于配置WebView。

private final WebViewClient client = new WebViewClient() {

/**

  • 防止加载网页时调起系统浏览器

*/

@Override

public boolean shouldOverrideUrlLoading(WebView view, String url) {

view.loadUrl(url);

return true;

}

@Override

public void onReceivedHttpAuthRequest(WebView webview,

com.tencent.smtt.export.external.interfaces.HttpAuthHandler httpAuthHandlerhost, String host,

String realm) {

boolean flag = httpAuthHandlerhost.useHttpAuthUsernamePassword();

}

@Override

public void onPageFinished(WebView webView, String s) {

super.onPageFinished(webView, s);

}

@Override

public void onReceivedError(WebView webView, int i, String s, String s1) {

System.out.println(“***********onReceivedError ************”);

super.onReceivedError(webView, i, s, s1);

}

@Override

public void onReceivedHttpError(WebView webView, WebResourceRequest webResourceRequest, WebResourceResponse webResourceResponse) {

System.out.println(“***********onReceivedHttpError ************”);

super.onReceivedHttpError(webView, webResourceRequest, webResourceResponse);

}

};

当前的页面是需要网络请求的,因此就会有相应的ViewModel和Repository,因为聚合给的新闻数据里面有一个uniquekey,用于查询新闻的详情信息,然后再去返回的详情信息里面找到url通过WebView去加载。当然并不是每一条新闻都能够去显示的,有一些新闻是没有详情信息的,这在我们点击新闻的时候就要做处理。

这是我们下面要做的事情,现在对于X5WebView还需要进行一个初始化,这样做是方便使用的。在BaseApplication中增加如下代码:

private void initX5WebView() {

HashMap map = new HashMap(2);

map.put(TbsCoreSettings.TBS_SETTINGS_USE_SPEEDY_CLASSLOADER, true);

map.put(TbsCoreSettings.TBS_SETTINGS_USE_DEXLOADER_SERVICE, true);

QbSdk.initTbsSettings(map);

//搜集本地tbs内核信息并上报服务器,服务器返回结果决定使用哪个内核。

QbSdk.PreInitCallback cb = new QbSdk.PreInitCallback() {

@Override

public void onViewInitFinished(boolean arg0) {

//x5內核初始化完成的回调,为true表示x5内核加载成功,否则表示x5内核加载失败,会自动切换到系统内核。

Log.d(“app”, " onViewInitFinished is " + arg0);

}

@Override

public void onCoreInitFinished() {

}

};

//x5内核初始化接口

QbSdk.initX5Environment(getApplicationContext(), cb);

}

然后在onCreate中调用它。

在这里插入图片描述

下面关于WebView的使用就只有一步了,那就是加载url,现在还没有的,去获取它。

三、获取新闻详情


在聚合API中获取新闻详情是另一个接口,在写这个接口之前,先写一个返回的新闻详情数据。

① 新闻详情数据

在model包下新增一个NewsDetailResponse类,里面的代码如下:

public class NewsDetailResponse {

private String reason;

private ResultBean result;

private Integer error_code;

public String getReason() {

return reason;

}

public void setReason(String reason) {

this.reason = reason;

}

public ResultBean getResult() {

return result;

}

public void setResult(ResultBean result) {

this.result = result;

}

public Integer getError_code() {

return error_code;

}

public void setError_code(Integer error_code) {

this.error_code = error_code;

}

public static class ResultBean {

private String uniquekey;

private DetailBean detail;

private String content;

public String getUniquekey() {

return uniquekey;

}

public void setUniquekey(String uniquekey) {

this.uniquekey = uniquekey;

}

public DetailBean getDetail() {

return detail;

}

public void setDetail(DetailBean detail) {

this.detail = detail;

}

public String getContent() {

return content;

}

public void setContent(String content) {

this.content = content;

}

public static class DetailBean {

private String title;

private String date;

private String category;

private String author_name;

private String url;

private String thumbnail_pic_s;

private String thumbnail_pic_s02;

private String thumbnail_pic_s03;

public String getTitle() {

return title;

}

public void setTitle(String title) {

this.title = title;

}

public String getDate() {

return date;

}

public void setDate(String date) {

this.date = date;

}

public String getCategory() {

return category;

}

public void setCategory(String category) {

this.category = category;

}

public String getAuthor_name() {

return author_name;

}

public void setAuthor_name(String author_name) {

this.author_name = author_name;

}

public String getUrl() {

return url;

}

public void setUrl(String url) {

this.url = url;

}

public String getThumbnail_pic_s() {

return thumbnail_pic_s;

}

public void setThumbnail_pic_s(String thumbnail_pic_s) {

this.thumbnail_pic_s = thumbnail_pic_s;

}

public String getThumbnail_pic_s02() {

return thumbnail_pic_s02;

}

public void setThumbnail_pic_s02(String thumbnail_pic_s02) {

this.thumbnail_pic_s02 = thumbnail_pic_s02;

}

public String getThumbnail_pic_s03() {

return thumbnail_pic_s03;

}

public void setThumbnail_pic_s03(String thumbnail_pic_s03) {

this.thumbnail_pic_s03 = thumbnail_pic_s03;

}

}

}

}

② 新闻详情数据API

/**

  • 聚合新闻数据详情

*/

@GET(“/toutiao/content?key=99d3951ed32af2930afd9b38293a08a2”)

Observable newsDetail(@Query(“uniquekey”) String uniquekey);

这个接口用于请求新闻详情数据,返回值将会解析成NewsDetailResponse。

③ WebRepository

数据有了,API接口有了,下面就是去调用的地方了,在repository包下新增一个WebRepository类,里面的代码如下:

@SuppressLint(“CheckResult”)

public class WebRepository {

final MutableLiveData newsDetail = new MutableLiveData<>();

public final MutableLiveData failed = new MutableLiveData<>();

/**

  • 获取新闻详情数据

  • @param uniquekey 新闻ID

  • @return newsDetail

*/

public MutableLiveData getNewsDetail(String uniquekey) {

NetworkApi.createService(ApiService.class, 2).

newsDetail(uniquekey).compose(NetworkApi.applySchedulers(new BaseObserver() {

@Override

public void onSuccess(NewsDetailResponse newsDetailResponse) {

if (newsDetailResponse.getError_code() == 0) {

newsDetail.setValue(newsDetailResponse);

} else {

failed.postValue(newsDetailResponse.getReason());

}

}

@Override

public void onFailure(Throwable e) {

failed.postValue("NewsDetail Error: " + e.toString());

}

}));

return newsDetail;

}

}

很简单的代码,和之前的地方基本上没啥差别。

④ WebViewModel

数据获取有了,下面就是通过ViewModel去关联Activity。在viewmodels包下新建一个WebViewModel,里面的代码如下:

public class WebViewModel extends BaseViewModel {

public LiveData newsDetail;

public void getNewDetail(String uniquekey) {

WebRepository webRepository = new WebRepository();

failed = webRepository.failed;

newsDetail = webRepository.getNewsDetail(uniquekey);

}

}

下面就是在WebActivity中去观察这个网络返回的数据了。

⑤ 页面数据处理

打开WebActivity,实际上我们只需要修改onCreate中的代码就可以了,代码如下:

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

ActivityWebBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_web);

WebViewModel viewModel = new ViewModelProvider(this).get(WebViewModel.class);

binding.webView.setWebViewClient(client);

setStatusBar(true);

// 在调用TBS初始化、创建WebView之前进行如下配置

String uniquekey = getIntent().getStringExtra(“uniquekey”);

if (uniquekey != null) {

viewModel.getNewDetail(uniquekey);

viewModel.newsDetail.observe(context, newsDetailResponse ->

binding.webView.loadUrl(newsDetailResponse.getResult().getDetail().getUrl()));

viewModel.failed.observe(context, this::showMsg);

}

}

这里的代码很常规,首先是绑定布局,然后是ViewModel,再设置webView的配置客户端,设置状态栏,然后就是获取其他页面传递过来的参数,通过这个参数去请求接口,观察返回值,最后加载返回的url。

这里都完成了,下一步就是传递这个参数了,什么时候传递呢?当然是点击的时候了。

四、传递新闻参数


在点击新闻列表中的某一项的时候传递参数到WebActivity中,在NewsAdapter类中添加如下代码:

public static class ClickBinding {

public void itemClick(NewsResponse.ResultBean.DataBean dataBean, View view) {

if(“1”.equals(dataBean.getIs_content())){

Intent intent = new Intent(view.getContext(), WebActivity.class);

intent.putExtra(“uniquekey”, dataBean.getUniquekey());

view.getContext().startActivity(intent);

} else {

Toast.makeText(view.getContext(), “没有详情信息”, Toast.LENGTH_SHORT).show();

}

}

}

当Is_content不为1的时候就表示没有详情信息,则提示一下即可。

然后在convert方法中添加一行代码,如下图所示:

在这里插入图片描述

最后就是修改item_newx.xml中的代码了,改动如下图所示:

在这里插入图片描述

由于我希望WebView可以沉浸式,因此我在AndroidManifest.xml中对这个WebActivity进行了主题设置,代码如下:

<activity

android:name=“.ui.activity.WebActivity”

android:theme=“@style/SplashTheme” />

下面就可以运行了。

在这里插入图片描述

下面就是点击视频item打开视频的播放地址了。

五、热门视频播放


这里首先要修改视频列表适配器中的内容,打开VideoAdapter,在里面增加如下代码:

public static class ClickBinding {

public void itemClick(@NotNull VideoResponse.ResultBean resultBean, View view) {

if (resultBean.getShare_url() != null) {

view.getContext().startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse(resultBean.getShare_url())));

} else {

Toast.makeText(view.getContext(), “视频地址为空”, Toast.LENGTH_SHORT).show();

}

}

}

然后convert方法中设置进去。

在这里插入图片描述

再修改item_video.xml的代码,如下图所示:

在这里插入图片描述

这里的视频链接地址实际上是抖音的视频地址,只不过聚合采集了数据,因此如果你的手机上有抖音,你点击之后会打开抖音播放这个视频,没有抖音会打开浏览器去播放这个视频,这里就不做演示了,自行去尝试。

六、侧滑抽屉


为了充分利用我们的屏幕控件,一些个人信息和设置是可以放到抽屉菜单里面的,就像QQ那样。那么怎么样做一个抽屉菜单呢?其实很简单。首先在layout下创建一个nav_header.xml布局,里面的代码如下:

<?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”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:orientation=“vertical”>

<RelativeLayout

android:layout_width=“match_parent”

android:layout_height=“120dp”

android:background=“@color/purple_500”>

<com.google.android.material.imageview.ShapeableImageView

android:id=“@+id/iv_avatar”

android:layout_width=“80dp”

android:layout_height=“80dp”

android:layout_centerVertical=“true”

android:layout_marginStart=“24dp”

android:layout_marginEnd=“24dp”

android:padding=“1dp”

android:scaleType=“centerCrop”

android:src=“@drawable/logo”

app:shapeAppearanceOverlay=“@style/circleImageStyle”

app:strokeColor=“@color/white”

app:strokeWidth=“2dp” />

<TextView

android:id=“@+id/tv_name”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_alignTop=“@+id/iv_avatar”

android:layout_marginTop=“16dp”

android:layout_toEndOf=“@+id/iv_avatar”

android:text=“初学者-Study”

android:textColor=“#FFF”

android:textSize=“16sp” />

<TextView

android:id=“@+id/tv_tip”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_below=“@+id/tv_name”

android:layout_marginTop=“8dp”

android:layout_toEndOf=“@+id/iv_avatar”

android:text=“Android | Java”

android:textColor=“#FFF”

android:textSize=“14sp” />

这里的图标就是我的博客头像,你可以到我的源码中去找,也可以用自己的图片。

然后在menu下创建一个nav_menu.xml,里面的代码如下:

<?xml version="1.0" encoding="utf-8"?>

<item

android:id=“@+id/item_setting”

android:icon=“@drawable/icon_settings”

android:title=“设置” />

<item

android:id=“@+id/item_logout”

android:icon=“@drawable/icon_logout”

android:title=“退出” />

然后修改activity_home.xml,将根布局改成DrawerLayout,里面的代码如下:

<?xml version="1.0" encoding="utf-8"?>

<layout 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”>

<androidx.drawerlayout.widget.DrawerLayout

android:id=“@+id/drawer_layout”

android:layout_width=“match_parent”

android:layout_height=“match_parent”>

<RelativeLayout

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:fitsSystemWindows=“true”

tools:context=“.ui.activity.HomeActivity”>

<com.google.android.material.appbar.MaterialToolbar

android:id=“@+id/toolbar”

android:layout_width=“match_parent”

android:layout_height=“?attr/actionBarSize”

android:background=“@color/purple_500”>

<com.google.android.material.imageview.ShapeableImageView

android:id=“@+id/iv_avatar”

android:layout_width=“36dp”

android:layout_height=“36dp”

android:padding=“0.5dp”

android:scaleType=“centerCrop”

android:src=“@drawable/logo”

app:shapeAppearanceOverlay=“@style/circleImageStyle”

app:strokeColor=“@color/white”

app:strokeWidth=“1dp” />

<TextView

android:id=“@+id/tv_title”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_gravity=“center”

android:text=“头条新闻”

android:textColor=“@color/white”

android:textSize=“18sp”

android:textStyle=“bold” />

</com.google.android.material.appbar.MaterialToolbar>

<fragment

android:id=“@+id/nav_host_fragment”

android:name=“androidx.navigation.fragment.NavHostFragment”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:layout_above=“@+id/bottom_navigation”

android:layout_below=“@+id/toolbar”

app:navGraph=“@navigation/nav_graph” />

<com.google.android.material.bottomnavigation.BottomNavigationView

android:id=“@+id/bottom_navigation”

android:layout_width=“match_parent”

android:layout_height=“?attr/actionBarSize”

android:layout_alignParentBottom=“true”

android:background=“#FFF”

app:menu=“@menu/navigation_menu” />

<com.google.android.material.navigation.NavigationView

android:id=“@+id/nav_view”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:layout_gravity=“start”

app:headerLayout=“@layout/nav_header”

app:itemIconSize=“24dp”

app:itemIconTint=“#000”

app:itemTextColor=“#000”

app:menu=“@menu/nav_menu” />

</androidx.drawerlayout.widget.DrawerLayout>

这里主要就是通过NavigationView去加载刚才的两个布局xml,一个作为头部一个作为菜单。同时我在Toolbar上放了一个Image,当点击的时候就可以打开抽屉。

<com.google.android.material.navigation.NavigationView

android:id=“@+id/nav_view”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:layout_gravity=“start”

app:headerLayout=“@layout/nav_header”

app:itemIconSize=“24dp”

app:itemIconTint=“#000”

app:itemTextColor=“#000”

app:menu=“@menu/nav_menu” />

下面我们修改HomeActivity中的代码,在initView中增加如下代码:

在这里插入图片描述

然后运行一下:

在这里插入图片描述

嗯,这里我们的侧滑抽屉就完成了,当然后面还会对这个部分增加更多的功能使用,现在里面只有一个设置和一个退出。既然说到退出了,那么就来写一下退出这个功能吧。

七、应用退出


退出这是一个需要小心的功能,因为涉及到Activity的栈,当我们从一个Activity跳转到另一个Activity时,如果之前的Activity没有销毁掉,则它就在栈里,当前跳转的Activity在栈顶。而不可能每一次跳转页面都需要销毁之前的页面。因此当应用需要退出时,首先我们应该销毁掉所有的Activity,然后再去关掉进程,这样你的程序才算是完整退出了。这里我们需要一个ActivityManager,在activity包下新建一个ActivityManager类,里面的代码如下:

public class ActivityManager {

//保存所有创建的Activity

private final List activityList = new ArrayList<>();

public static ActivityManager mInstance;

public static ActivityManager getInstance() {

if (mInstance == null) {

synchronized (ActivityManager.class) {

if (mInstance == null) {

mInstance = new ActivityManager();

}

}

}

return mInstance;

}

/**

  • 添加Activity

  • @param activity

*/

public void addActivity(Activity activity){

if(activity != null){

activityList.add(activity);

}

}

/**

  • 移除Activity

  • @param activity

*/

public void removeActivity(Activity activity){

if(activity != null){

activityList.remove(activity);

}

}

/**

  • 关闭所有Activity

*/

public void finishAllActivity(){

for (Activity activity : activityList) {

activity.finish();

}

}

}

然后要使我们的每一个Activity在创建的时候都添加到ActivityManager中,我们需要现在BaseApplication中添加如下代码:

public static ActivityManager getActivityManager() {

return ActivityManager.getInstance();

}

然后在BaseActivity中的onCreate中增加如下代码即可。

BaseApplication.getActivityManager().addActivity(this);

然后这样有一个前提,就是你所有的Activity都要继承自BaseActivity。然后我们在HomeActivity中新增一个退出登录方法。

/**

  • 退出登录

*/

private void logout() {

showMsg(“退出登录”);

MVUtils.put(Constant.IS_LOGIN,false);

jumpActivityFinish(LoginActivity.class);

}

在点击菜单的时候调用它。

在这里插入图片描述

然后我们会回到登录页面,在登录页面中增加一个两次返回表示退出应用的功能,在LoginActivity中增加如下代码:

private long timeMillis;

/**

  • Add a prompt to exit the application

*/

@Override

public boolean onKeyDown(int keyCode, KeyEvent event) {

if (keyCode == KeyEvent.KEYCODE_BACK && event.getAction() == KeyEvent.ACTION_DOWN) {

if ((System.currentTimeMillis() - timeMillis) > 2000) {

showMsg(“再次按下退出应用程序”);

timeMillis = System.currentTimeMillis();

} else {

exitTheProgram();

}

return false;

}

return super.onKeyDown(keyCode, event);

}

这里的exitTheProgram()方法,要写在BaseAcctivity中,方法如下:

protected void exitTheProgram() {

BaseApplication.getActivityManager().finishAllActivity();

}

那么我们再运行一下,看看效果。

在这里插入图片描述

八、登录注册


你可能会很疑惑,不是已经有一个登录了吗?为啥还有登录注册?等会儿,注册?注册到哪里去?又没有服务器数据库,这里我是都使用本地数据库,也从本地数据库去做校验。也就是说,后面你使用这个软件你需要先手动去注册一个用户,然后再去登录这个用户,我这么做的目的是希望更接近实际开发中的需求设计。写代码就讲究一个真听真看真实现。所以我们先来完成一个注册的功能,只不过我们的注册只是本地有效,请注意这一点。

① 建表

既然是保存用户信息到本地数据库里,则我们需要有一个表来操作,在bean包下新建一个User类,代码如下:

@Entity(tableName = “user”)

public class User extends BaseObservable {

@PrimaryKey

private int uid;

private String account;

private String pwd;

@Ignore

private String confirmPwd;

private String nickname;

private String introduction;

public int getUid() {

return uid;

}

public void setUid(int uid) {

this.uid = uid;

}

@Bindable

public String getAccount() {

return account;

}

public void setAccount(String account) {

this.account = account;

notifyPropertyChanged(BR.account);

}

@Bindable

public String getPwd() {

return pwd;

}

public void setPwd(String pwd) {

this.pwd = pwd;

notifyPropertyChanged(BR.pwd);

}

@Bindable

public String getConfirmPwd() {

return confirmPwd;

}

public void setConfirmPwd(String confirmPwd) {

this.confirmPwd = confirmPwd;

notifyPropertyChanged(BR.confirmPwd);

}

@Bindable

public String getNickname() {

return nickname;

}

public void setNickname(String nickname) {

this.nickname = nickname;

notifyPropertyChanged(BR.nickname);

}

@Bindable

public String getIntroduction() {

return introduction;

}

public void setIntroduction(String introduction) {

this.introduction = introduction;

notifyPropertyChanged(BR.introduction);

}

public User() {}

@Ignore

public User(int uid, String account, String pwd, String confirmPwd, String nickname, String introduction) {

this.uid = uid;

this.account = account;

this.pwd = pwd;

this.confirmPwd = confirmPwd;

this.nickname = nickname;

this.introduction = introduction;

}

}

这里是一个User表,它里面有6个属性,uid可以不用管它,依次看下来就是账号,密码,确认密码,昵称,简介,其中确认密码这个字段只是用作校验的,因此不需要放入数据表中,所以我用@Ignore注解了,下面创建相关的Dao类。

② 表操作接口

在dao包下新建一个UserDao接口,里面的代码如下:

@Dao

public interface UserDao {

@Query(“SELECT * FROM user”)

Flowable<List> getAll();

@Update

Completable update(User user);

@Insert(onConflict = OnConflictStrategy.REPLACE)

Completable insert(User user);

@Query(“DELETE FROM user”)

Completable deleteAll();

}

这里没啥好说的,就是操作用户表的方法。

③ 数据库升级

之前的数据库版本是3,现在我新增了用户表,则需要对数据库进行一个升级迁移,在AppDatabase中增加如下代码:

/**

  • 版本升级迁移到4 新增用户表

*/

static final Migration MIGRATION_3_4 = new Migration(3, 4) {

@Override

public void migrate(SupportSQLiteDatabase database) {

//创建用户表

database.execSQL("CREATE TABLE user " +

"(uid INTEGER NOT NULL, " +

"account TEXT, " +

"pwd TEXT, " +

“nickname TEXT,” +

“introduction TEXT,” +

“PRIMARY KEY(uid))”);

}

};

新增一个抽象方法。

public abstract UserDao userDao();

然后如下图所示修改一下,数据库的升级迁移就完成了。

在这里插入图片描述

④ 数据储存库

现在关于数据库的部分就已经弄完了,接下来就是涉及到数据的保存和操作了,因为我们的用户表涉及到的页面可能不止一个,所以用户的存储库就不以页面所命名,因此在repository包下新建一个UserRepository,里面的代码如下:

public class UserRepository {

private static final String TAG = UserRepository.class.getSimpleName();

private final MutableLiveData userMutableLiveData = new MutableLiveData<>();

public final MutableLiveData failed = new MutableLiveData<>();

public MutableLiveData getUser() {

Flowable<List> listFlowable = BaseApplication.getDb().userDao().getAll();

CustomDisposable.addDisposable(listFlowable, users -> {

if (users.size() > 0) {

for (User user : users) {

if (user.getUid() == 1) {

userMutableLiveData.postValue(user);

break;

}

}

} else {

failed.postValue(“你还没有注册过吧,去注册吧!”);

}

});

return userMutableLiveData;

}

/**

  • 更新用户信息

  • @param user

*/

public void updateUser(User user) {

Completable update = BaseApplication.getDb().userDao().update(user);

CustomDisposable.addDisposable(update, () -> {

failed.postValue(“200”);

});

}

最后

最后这里放上我这段时间复习的资料,这个资料也是偶然一位朋友分享给我的,里面包含了腾讯、字节跳动、阿里、百度2019-2021面试真题解析,并且把每个技术点整理成了视频和PDF(知识脉络 + 诸多细节)。

还有 高级架构技术进阶脑图、高级进阶架构资料 帮助大家学习提升进阶,也可以分享给身边好友一起学习。

一起互勉~

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

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

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

了。

在这里插入图片描述

④ 数据储存库

现在关于数据库的部分就已经弄完了,接下来就是涉及到数据的保存和操作了,因为我们的用户表涉及到的页面可能不止一个,所以用户的存储库就不以页面所命名,因此在repository包下新建一个UserRepository,里面的代码如下:

public class UserRepository {

private static final String TAG = UserRepository.class.getSimpleName();

private final MutableLiveData userMutableLiveData = new MutableLiveData<>();

public final MutableLiveData failed = new MutableLiveData<>();

public MutableLiveData getUser() {

Flowable<List> listFlowable = BaseApplication.getDb().userDao().getAll();

CustomDisposable.addDisposable(listFlowable, users -> {

if (users.size() > 0) {

for (User user : users) {

if (user.getUid() == 1) {

userMutableLiveData.postValue(user);

break;

}

}

} else {

failed.postValue(“你还没有注册过吧,去注册吧!”);

}

});

return userMutableLiveData;

}

/**

  • 更新用户信息

  • @param user

*/

public void updateUser(User user) {

Completable update = BaseApplication.getDb().userDao().update(user);

CustomDisposable.addDisposable(update, () -> {

failed.postValue(“200”);

});

}

最后

最后这里放上我这段时间复习的资料,这个资料也是偶然一位朋友分享给我的,里面包含了腾讯、字节跳动、阿里、百度2019-2021面试真题解析,并且把每个技术点整理成了视频和PDF(知识脉络 + 诸多细节)。

还有 高级架构技术进阶脑图、高级进阶架构资料 帮助大家学习提升进阶,也可以分享给身边好友一起学习。

[外链图片转存中…(img-FCsJwLWw-1714285524868)]

[外链图片转存中…(img-YQDSBXkr-1714285524869)]

一起互勉~

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

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

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

  • 10
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Android MVVM框架搭建需要使用Java语言。MVVM框架是一种基于模型-视图-视图模型的设计模式,它可以帮助开发者更好地组织和管理Android应用程序的代码。在搭建MVVM框架时,需要使用一些常用的开源框架,如Data Binding、LiveData、ViewModel等。同时,还需要了解一些基本的Android开发知识和Java语言的基础知识。搭建MVVM框架需要一定的技术水平和经验,但是它可以提高应用程序的可维护性和可扩展性,让开发者更加高效地开发Android应用程序。 ### 回答2: 在Android开发过程中,MVVM框架已经成为了一种非常流行的设计模式,因为它能够很好地解决传统的MVC模式中存在的问题。这篇文章将着重讲解如何使用Java来构建MVVM框架MVVM框架主要由三个部分组成:View、ViewModel和Model。其中,View指的是用户界面;ViewModel则是View和Model之间的中间层,它包含View所需的数据和业务逻辑;而Model则是数据层,负责从数据源中获取和管理数据。 在Java中构建MVVM框架,可以使用以下几个关键技术: 1. Data Binding:Data Binding是一种新兴的技术,可以轻松地将数据绑定到用户界面上。在Android中,Data Binding库已经成为了MVVM框架构建中不可少的一部分。 2. LiveData:LiveData是Jetpack组件库中的一部分,它是用于构建响应式应用程序的一种强大的工具,也是MVVM模式的关键组成部分。LiveData可以监听数据源中的更改,并在数据发生变化时通知ViewModel。 3. ViewModel:ViewModel是MVVM中的重要组成部分,主要用于保存状态、管理数据和处理业务逻辑。它允许View对Model进行观察,从而实现MVVM框架的完整性和数据驱动。 4. RxJava:RxJava是一种Reactive编程框架,可以用于处理异步事件流,并在应用程序中实现非阻塞的、响应式编程。RxJava可以与MVVM框架结合使用,使架构更加灵活和响应。 搭建MVVM框架的过程中,首先需要创建View、ViewModel和Model三个关键组件。ViewModel将负责将Model的数据更新到View上,并处理业务逻辑。此外,使用Data Binding和LiveData建立数据流,实现双向数据绑定,保持ViewModel和View的同步。 最终,使用RxJava将ViewModel和View解耦,实现响应性和灵活性。在整个MVVM框架中,使用Java构建时需要注意的重要细节包括内存泄漏问题、持久性存储问题等等,尽可能地在代码编写时去考虑这些问题,以确保框架的可靠性和稳定性。 总而言之,使用Java搭建MVVM框架需要一定的技术积累和编码能力,但这种框架Android开发中已经被广泛应用,相信有机会成为您的一个不错的选择。 ### 回答3: MVVM是Model-View-ViewModel的缩写,是一种软件架构模式。在安卓应用开发中,我们可以利用MVVM框架搭建Java开发项目,实现数据和用户交互分离,让代码更加清晰易懂,可读性更高。 MVVM架构中,Model代表实体模型,View代表图形界面,ViewModel作为桥梁连接Model和View,负责将Model转化成View的形式,以及监听View的变化反向更新Model。 在搭建Android MVVM框架时,我们需要先定义布局文件,将View的控件和ViewModel关联。然后定义ViewModel类,在类中添加模型数据和业务逻辑。最后,在Activity或Fragment中实现ViewModel的数据绑定和监听,将数据和页面连接起来。 在Java开发中,我们可以利用一些主流的MVVM框架,例如Google官方提供的Android Architecture Components框架,它包含LiveData、ViewModel、Room和DataBinding四个组件,能够快速搭建MVVM架构,实现组件之间的数据通信和交互。 除此之外,还有Databinding框架,它可以在布局文件中直接绑定数据,省去了一些繁琐的代码,但是需要一定的学习成本。 总而言之,MVVM框架搭建Java项目是非常有利于提高代码可维护性和可读性,通过桥梁连接View与Model,让开发更加高效。不同的情况需要使用不同的框架,开发者可以根据自己的需求进行选择。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值