Android面试题目基础总结(二)
怎么解决项目上的一个同步问题
- 多线程同步:在往届面试中,常被问到如何处理多线程同步资源访问问题。比如银行转账场景,多个线程可能同时操作账户余额。此时可使用
synchronized
关键字,它基于对象锁机制,确保同一时刻只有一个线程能进入同步代码块。以一个简单的账户类为例:
public class Account {private double balance;public Account(double initialBalance) {this.balance = initialBalance;}public synchronized void deposit(double amount) {balance += amount;}public synchronized void withdraw(double amount) {if (balance >= amount) {balance -= amount;}}public synchronized double getBalance() {return balance;}
}
但在高并发场景下,synchronized
可能性能不佳,面试官可能会进一步追问优化方案。这时可提及 java.util.concurrent.locks.ReentrantLock
,它相比 synchronized
更灵活,支持可中断的锁获取、公平锁等特性。例如:
import java.util.concurrent.locks.ReentrantLock;public class AccountWithLock {private double balance;private ReentrantLock lock = new ReentrantLock();public AccountWithLock(double initialBalance) {this.balance = initialBalance;}public void deposit(double amount) {lock.lock();try {balance += amount;} finally {lock.unlock();}}public void withdraw(double amount) {lock.lock();try {if (balance >= amount) {balance -= amount;}} finally {lock.unlock();}}public double getBalance() {lock.lock();try {return balance;} finally {lock.unlock();}}
}
- 数据同步(本地与服务器):面试中也常涉及本地数据与服务器数据同步问题。例如,电商应用中商品库存的同步。可使用
ContentProvider
结合SyncAdapter
(适用于较旧 Android 版本),ContentProvider
负责本地数据的存储与访问,SyncAdapter
处理与服务器的同步逻辑。而在较新的开发中,更多使用WorkManager
。WorkManager
内部会根据设备状态(如网络连接、电量等)智能调度同步任务。例如在同步任务中,通过OkHttp
进行网络请求,OkHttp
的Call
类负责执行请求,在RealCall
类的execute
方法里构建HttpCodec
处理请求与响应编码解码,同时通过RetryAndFollowUpInterceptor
类处理请求重试逻辑。
怎么使用 Retrofit 获取天气数据
- 基础使用:往届面试常考查 Retrofit 的基本使用流程。首先在
build.gradle
文件添加依赖:
implementation 'com.squareup.retrofit2:retrofit:2.9.0'
implementation 'com.squareup.retrofit2:converter - gson:2.9.0'
接着定义 API 接口,比如获取天气数据接口:
import retrofit2.Call;
import retrofit2.http.GET;
import retrofit2.http.Query;public interface WeatherApi {@GET("weather")Call<WeatherResponse> getWeather(@Query("q") String city,@Query("appid") String apiKey);
}
然后创建 Retrofit 实例:
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;public class RetrofitClient {private static Retrofit retrofit;private static final String BASE_URL = "https://api.openweathermap.org/data/2.5/";public static Retrofit getRetrofitInstance() {if (retrofit == null) {retrofit = new Retrofit.Builder().baseUrl(BASE_URL).addConverterFactory(GsonConverterFactory.create()).build();}return retrofit;}
}
最后发起请求:
WeatherApi weatherApi = RetrofitClient.getRetrofitInstance().create(WeatherApi.class);
Call<WeatherResponse> call = weatherApi.getWeather("Beijing", "YOUR_API_KEY");
call.enqueue(new Callback<WeatherResponse>() {@Overridepublic void onResponse(Call<WeatherResponse> call, Response<WeatherResponse> response) {if (response.isSuccessful()) {WeatherResponse weatherResponse = response.body();// 处理天气数据} else {// 处理错误}}@Overridepublic void onFailure(Call<WeatherResponse> call, Throwable t) {// 处理请求失败}
});
- 原理考查:面试可能深入到 Retrofit 原理。Retrofit 使用动态代理创建接口实例,在
Retrofit
的create
方法里,通过Proxy.newProxyInstance
创建代理对象,在代理对象的invoke
方法中,依据方法信息构建ServiceMethod
和OkHttpCall
,最终借助OkHttp
执行网络请求。当使用GsonConverterFactory
解析数据时,它在fromResponseBody
方法中利用Gson
库解析响应体,GsonResponseBodyConverter
的convert
方法调用Gson
的fromJson
方法将响应体字符串转为目标对象。
每五分钟自动刷新怎么做
- WorkManager 基本使用:面试常问如何使用
WorkManager
实现定时任务。首先定义Worker
类,例如:
import android.content.Context;
import android.util.Log;import androidx.annotation.NonNull;
import androidx.work.Worker;
import androidx.work.WorkerParameters;public class WeatherRefreshWorker extends Worker {public WeatherRefreshWorker(@NonNull Context context, @NonNull WorkerParameters workerParams) {super(context, workerParams);}@NonNull@Overridepublic Result doWork() {// 这里执行获取天气数据的操作,例如使用 RetrofitLog.d("WeatherRefresh", "Refreshing weather data...");return Result.success();}
}
然后创建并安排周期性任务:
import androidx.work.Constraints;
import androidx.work.NetworkType;
import androidx.work.PeriodicWorkRequest;
import androidx.work.WorkManager;import java.util.concurrent.TimeUnit;public class MainActivity {public void scheduleWeatherRefresh() {Constraints constraints = new Constraints.Builder().setRequiredNetworkType(NetworkType.CONNECTED).build();PeriodicWorkRequest workRequest = new PeriodicWorkRequest.Builder(WeatherRefreshWorker.class, 5, TimeUnit.MINUTES).setConstraints(constraints).build();WorkManager.getInstance(this).enqueue(workRequest);}
}
- 原理拓展:进一步可能问到
WorkManager
原理,它内部依据设备状态,利用JobScheduler
(API 21+)、AlarmManager
(API < 21)来调度任务。WorkManager
的enqueue
方法将任务添加到队列,然后根据任务约束条件和调度策略执行任务,PeriodicWorkRequest
按设定周期重复执行任务。