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

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”);

});

}

/**

  • 保存热门壁纸数据

*/

public void saveUser(User user) {

Completable deleteAll = BaseApplication.getDb().userDao().deleteAll();

CustomDisposable.addDisposable(deleteAll, () -> {

//保存到数据库

Completable insertAll = BaseApplication.getDb().userDao().insert(user);

//RxJava处理Room数据存储

CustomDisposable.addDisposable(insertAll, () -> failed.postValue(“200”));

});

}

}

这里有三个方法,这是我目前所想到的,后续可能会对方法已经修改,首先我们要完成用户的注册和登录,则就会用到获取用户和保存用户。

⑤ RegisterViewModel

这里我先写对应注册页面的ViewModel,然后再去写注册页面的代码,在viewmodels包下新建一个RegisterViewModel类,代码如下:

public class RegisterViewModel extends BaseViewModel {

public MutableLiveData user;

public MutableLiveData getUser(){

if(user == null){

user = new MutableLiveData<>();

}

return user;

}

/**

  • 注册

*/

public void register() {

UserRepository userRepository = new UserRepository();

failed = userRepository.failed;

user.getValue().setUid(1);

Log.d(“TAG”, "register: "+new Gson().toJson(user.getValue()));

userRepository.saveUser(user.getValue());

}

}

这里的核心功能就是注册了,这里的注册我只写了一个id,其他的数据需要从页面上去获取。

⑥ 注册页面

在activity包下新建一个RegisterActivity,对应的布局是activity_register.xml,布局的代码如下:

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

<variable

name=“register”

type=“com.llw.mvvm.viewmodels.RegisterViewModel” />

<LinearLayout

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:orientation=“vertical”>

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

app:navigationIcon=“@drawable/ic_back_white”>

<TextView

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>

<TextView

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:gravity=“center”

android:padding=“24dp”

android:text=“MVVM”

android:textColor=“@color/purple_500”

android:textSize=“48sp”

android:textStyle=“bold” />

<RelativeLayout

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:orientation=“vertical”

android:padding=“32dp”>

<com.google.android.material.textfield.TextInputLayout

android:id=“@+id/et_account_lay”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”>

<com.google.android.material.textfield.TextInputEditText

android:id=“@+id/et_account”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:background=“@color/white”

android:hint=“账号”

android:text=“@={register.user.account}” />

</com.google.android.material.textfield.TextInputLayout>

<TextView

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_alignTop=“@+id/et_account_lay”

android:layout_alignEnd=“@+id/et_account_lay”

android:layout_alignBottom=“@+id/et_account_lay”

android:gravity=“center”

android:text=“*”

android:textColor=“@color/purple_500”

android:textSize=“24sp” />

<com.google.android.material.textfield.TextInputLayout

android:id=“@+id/et_pwd_lay”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:layout_below=“@+id/et_account_lay”

android:layout_marginTop=“12dp”>

<com.google.android.material.textfield.TextInputEditText

android:id=“@+id/et_pwd”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:background=“@color/white”

android:hint=“密码”

android:inputType=“textPassword”

android:text=“@={register.user.pwd}” />

</com.google.android.material.textfield.TextInputLayout>

<TextView

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_alignTop=“@+id/et_pwd_lay”

android:layout_alignEnd=“@+id/et_pwd_lay”

android:layout_alignBottom=“@+id/et_pwd_lay”

android:gravity=“center”

android:text=“*”

android:textColor=“@color/purple_500”

android:textSize=“24sp” />

<com.google.android.material.textfield.TextInputLayout

android:id=“@+id/et_confirm_pwd_lay”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:layout_below=“@+id/et_pwd_lay”

android:layout_marginTop=“12dp”>

<com.google.android.material.textfield.TextInputEditText

android:id=“@+id/et_confirm_pwd”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:background=“@color/white”

android:hint=“确认密码”

android:inputType=“textPassword”

android:text=“@={register.user.confirmPwd}” />

</com.google.android.material.textfield.TextInputLayout>

<com.google.android.material.textfield.TextInputLayout

android:id=“@+id/et_nickname_lay”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:layout_below=“@+id/et_confirm_pwd_lay”

android:layout_marginTop=“12dp”>

<com.google.android.material.textfield.TextInputEditText

android:id=“@+id/et_nickname_pwd”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:background=“@color/white”

android:hint=“用户昵称”

android:text=“@={register.user.nickname}” />

</com.google.android.material.textfield.TextInputLayout>

<com.google.android.material.textfield.TextInputLayout

android:id=“@+id/et_introduction_lay”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:layout_below=“@+id/et_nickname_lay”

android:layout_marginTop=“12dp”>

<com.google.android.material.textfield.TextInputEditText

android:id=“@+id/et_introduction”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:background=“@color/white”

android:hint=“用户简介”

android:text=“@={register.user.introduction}” />

</com.google.android.material.textfield.TextInputLayout>

<TextView

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_alignTop=“@+id/et_confirm_pwd_lay”

android:layout_alignEnd=“@+id/et_confirm_pwd_lay”

android:layout_alignBottom=“@+id/et_confirm_pwd_lay”

android:gravity=“center”

android:text=“*”

android:textColor=“@color/purple_500”

android:textSize=“24sp” />

<com.google.android.material.button.MaterialButton

android:id=“@+id/btn_register”

android:layout_width=“match_parent”

android:layout_height=“48dp”

android:layout_below=“@+id/et_introduction_lay”

android:layout_margin=“48dp”

android:insetTop=“0dp”

android:insetBottom=“0dp”

android:text=“注 册”

app:cornerRadius=“12dp” />

<TextView

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_alignParentBottom=“true”

android:layout_centerHorizontal=“true”

android:text=“注册信息中 * 为必填项” />

这里的布局里面就是五个输入框一个按钮,其中有一些信息是必须要输入的,有一些信息不是必须的。下面我们修改RegisterActivity的代码,如下所示:

public class RegisterActivity extends BaseActivity {

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

private ActivityRegisterBinding binding;

private RegisterViewModel registerViewModel;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

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

registerViewModel = new ViewModelProvider(this).get(RegisterViewModel.class);

registerViewModel.getUser().setValue(new User(0, “”, “”, “”, “”, “”));

binding.setRegister(registerViewModel);

initView();

}

private void initView() {

back(binding.toolbar);

binding.btnRegister.setOnClickListener(v -> {

if (registerViewModel.user.getValue().getAccount().isEmpty()) {

showMsg(“请输入账号”);

return;

}

if (registerViewModel.user.getValue().getPwd().isEmpty()) {

showMsg(“请输入密码”);

return;

}

if (registerViewModel.user.getValue().getConfirmPwd().isEmpty()) {

showMsg(“请确认密码”);

return;

}

if (!registerViewModel.user.getValue().getPwd().equals(registerViewModel.user.getValue().getConfirmPwd())) {

showMsg(“两次输入密码不一致”);

return;

}

registerViewModel.register();

registerViewModel.failed.observe(this, failed -> {

showMsg(“200”.equals(failed) ? “注册成功” : failed);

if (“200”.equals(failed)) {

finish();

}

});

});

}

}

这里我们可以利用failed返回的内容作为注册成功和失败的标准,注册成功则返回之前的登录页面,虽然我们还没有在登录页面中写跳转到注册页面的代码。下面我们就来修改登录页面的代码。

⑦ 修改登录页面

首先是从布局上修改,修改activity_login.xml的代码如下:

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

<variable

name=“viewModel”

type=“com.llw.mvvm.viewmodels.LoginViewModel” />

<LinearLayout

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:orientation=“vertical”>

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

android:layout_width=“match_parent”

android:layout_height=“?attr/actionBarSize”

android:background=“@color/purple_500”>

<TextView

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>

<LinearLayout

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:gravity=“center”

android:orientation=“vertical”

android:padding=“32dp”>

<TextView

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:gravity=“center”

android:padding=“24dp”

android:text=“MVVM”

android:textColor=“@color/purple_500”

android:textSize=“48sp”

android:textStyle=“bold” />

<TextView

android:id=“@+id/tv_account”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:text=“@{viewModel.user.account}”

android:visibility=“gone” />

<TextView

android:id=“@+id/tv_pwd”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_marginBottom=“24dp”

android:text=“@{viewModel.user.pwd}”

android:visibility=“gone” />

<com.google.android.material.textfield.TextInputLayout

android:layout_width=“match_parent”

android:layout_height=“wrap_content”>

<com.google.android.material.textfield.TextInputEditText

android:id=“@+id/et_account”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:background=“@color/white”

android:hint=“账号”

android:text=“@={viewModel.user.account}” />

</com.google.android.material.textfield.TextInputLayout>

<com.google.android.material.textfield.TextInputLayout

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:layout_marginTop=“12dp”>

<com.google.android.material.textfield.TextInputEditText

android:id=“@+id/et_pwd”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:background=“@color/white”

android:hint=“密码”

android:inputType=“textPassword”

android:text=“@={viewModel.user.pwd}” />

</com.google.android.material.textfield.TextInputLayout>

<com.google.android.material.button.MaterialButton

android:id=“@+id/btn_login”

android:layout_width=“match_parent”

android:layout_height=“48dp”

android:layout_margin=“24dp”

android:insetTop=“0dp”

android:insetBottom=“0dp”

android:text=“登 录”

app:cornerRadius=“12dp” />

<LinearLayout

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”>

<TextView

android:text=“没有账号?去”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”/>

<TextView

android:text=“注册”

android:onClick=“toRegister”

android:textColor=“@color/purple_500”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

tools:ignore=“UsingOnClickInXml” />

然后就是修改LoginViewModel的代码了,这里要注意,之前用的是model包下的User类,现在用的是bean包下的User类,有本质的区别,修改LoginViewModel的代码如下所示:

public class LoginViewModel extends BaseViewModel {

public MutableLiveData user;

public MutableLiveData getUser(){

if(user == null){

user = new MutableLiveData<>();

}

return user;

}

public LiveData<com.llw.mvvm.db.bean.User> localUser;

public void getLocalUser(){

UserRepository userRepository = new UserRepository();

localUser = userRepository.getUser();

failed = userRepository.failed;

}

}

下面修改LoginActivity,修改代码如下所示:

public class LoginActivity extends BaseActivity {

private ActivityLoginBinding dataBinding;

private LoginViewModel loginViewModel;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

//数据绑定视图

dataBinding = DataBindingUtil.setContentView(this, R.layout.activity_login);

loginViewModel = new LoginViewModel();

//Model → View

User user = new User(“”, “”);

loginViewModel.getUser().setValue(user);

//获取观察对象

MutableLiveData user1 = loginViewModel.getUser();

user1.observe(this, user2 -> {

Log.d(“LoginActivity”, "onCreate: " + user2.getAccount());

dataBinding.setViewModel(loginViewModel);

});

dataBinding.btnLogin.setOnClickListener(v -> {

if (loginViewModel.user.getValue().getAccount().isEmpty()) {

showMsg(“请输入账号”);

return;

}

if (loginViewModel.user.getValue().getPwd().isEmpty()) {

showMsg(“请输入密码”);

return;

}

//检查输入的账户和密码是否是注册过的。

checkUser();

});

}

private void checkUser() {

loginViewModel.getLocalUser();

loginViewModel.localUser.observe(this, localUser -> {

if (!loginViewModel.user.getValue().getAccount().equals(localUser.getAccount()) ||

!loginViewModel.user.getValue().getPwd().equals(localUser.getPwd())) {

showMsg(“账号或密码错误”);

return;

}

//记录已经登录过

MVUtils.put(Constant.IS_LOGIN, true);

showMsg(“登录成功”);

jumpActivity(MainActivity.class);

});

loginViewModel.failed.observe(this, this::showMsg);

}

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);

}

public void toRegister(View view) {

jumpActivity(RegisterActivity.class);

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
img

总结:

各行各样都会淘汰一些能力差的,不仅仅是IT这个行业,所以,不要被程序猿是吃青春饭等等这类话题所吓倒,也不要觉得,找到一份工作,就享受安逸的生活,你在安逸的同时,别人正在奋力的向前跑,这样与别人的差距也就会越来越遥远,加油,希望,我们每一个人,成为更好的自己。

  • BAT大厂面试题、独家面试工具包,

  • 资料包括 数据结构、Kotlin、计算机网络、Framework源码、数据结构与算法、小程序、NDK、Flutter

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

}

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);

}

public void toRegister(View view) {

jumpActivity(RegisterActivity.class);

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
[外链图片转存中…(img-0yl3tUje-1712767813526)]
[外链图片转存中…(img-sMny6HO5-1712767813527)]
[外链图片转存中…(img-oulvwD3r-1712767813527)]
[外链图片转存中…(img-MtPbfHt6-1712767813527)]
[外链图片转存中…(img-BvABl35v-1712767813528)]
[外链图片转存中…(img-O5X1zE5R-1712767813528)]
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
[外链图片转存中…(img-g6K5N767-1712767813528)]

总结:

各行各样都会淘汰一些能力差的,不仅仅是IT这个行业,所以,不要被程序猿是吃青春饭等等这类话题所吓倒,也不要觉得,找到一份工作,就享受安逸的生活,你在安逸的同时,别人正在奋力的向前跑,这样与别人的差距也就会越来越遥远,加油,希望,我们每一个人,成为更好的自己。

  • BAT大厂面试题、独家面试工具包,

  • 资料包括 数据结构、Kotlin、计算机网络、Framework源码、数据结构与算法、小程序、NDK、Flutter
    [外链图片转存中…(img-t99bQoQL-1712767813528)]

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
[外链图片转存中…(img-5szbckqT-1712767813529)]

  • 25
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值