Original text: medium.com/gitconnecte…
When I started Android development, there were no Fragments, RecyclerViews, ViewModels, or Coroutines. The technical components are updated, but the only thing that stays the same is BaseClass, the various base classes that exist in the project 🥶.
If you haven’t seen BaseActivity, BaseFragment, or BaseViewModel, you are in luck 🌟.
These BaseClass base classes misapply the “(Don’t Repeat Yourself) Don’t Repeat Yourself” principle.
“Don’t Repeat Yourself” (DRY) is a principle in software development that uses abstraction and data standardization to reduce rework in software development.
A common scenario is when you need to do tracking log analysis on your App, or when you need to register a BroadcastReceiver with each AcElasticity App to send notifications.
class BaseActivity: AppCompatActivity() { private lateinit var logoutReceiver: BroadcastReceiver override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) logoutReceiver = LogoutReceiver() registerReceiver(logoutReceiver, IntentFilter("ACTION_LOGOUT")) } override fun onStart() { super.onStart() traceEvent("Activity started") } override fun onStop() { traceEvent("Activity stopped") super.onStop() } private fun traceEvent(event: String) { //MyAnalytics.newEvent(event) } inner class LogoutReceiver: BroadcastReceiver() { override fun onReceive(context: Context? , intent: Intent?) { finish() } } }Copy the code
This code looks fine. Each BaseActivity subclass does not need to write new code, but inherits the ability to track logs and log out from its parent class.
But not every Activity requires logging and logging out, and these functions are redundant for these activities.
Some people say that I can flag whether these two functions are enabled by a flag, which is wrong ❌ approach.
We can use Delegation, take advantage of Kotlin (Java also works, with some extra code to write) and the Android Lifecycle component to get rid of the BaseClass nightmare.
Delegation Pattern is a basic technique in software design patterns. In the delegate pattern, two objects participate in processing the same request, and the receiving object delegates the request to the other object. The delegate pattern is a basic skill, and many other patterns, such as the state pattern, the policy pattern, and the visitor pattern, essentially use the delegate pattern for more specific situations. The delegate pattern allows us to substitute aggregation for inheritance, and it also allows us to simulate mixins. — Wikipedia
We can create an interface and hand the actual implementation to the delegate class. Thanks to the Lifecycle component, we can bind our logic to the Lifecycle.
We only need 2 steps:
- Create an interface and implementation class for each Feature. (Follow the principle of single responsibility)
- The Activity/Fragment implements the interface, passing the implementation of the interface to the delegate implementation class and binding the lifecycle
class MyActivity:
AppCompatActivity(),
AnalyticsDelegate by AnalyticsDelegateImpl(),
LogoutDelegate by LogoutDelegateImpl()
{
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
registerAnalytics(lifecycle)
registerLogout(this)
}
}
Copy the code
Interface and delegate implementation classes:
interface AnalyticsDelegate {
fun registerAnalytics(lifecycle: Lifecycle)
}
class AnalyticsDelegateImpl: AnalyticsDelegate, DefaultLifecycleObserver {
override fun registerAnalytics(lifecycle: Lifecycle) {
lifecycle.addObserver(this)
}
override fun onStart(owner: LifecycleOwner) {
traceEvent("Activity started")
}
override fun onStop(owner: LifecycleOwner) {
traceEvent("Activity stopped")
}
private fun traceEvent(event: String) {
//MyAnalytics.newEvent(event)
}
}
Copy the code
interface LogoutDelegate { fun registerLogout(activity: AppCompatActivity) } class LogoutDelegateImpl: LogoutDelegate, DefaultLifecycleObserver { private lateinit var activity: AppCompatActivity override fun registerLogout(activity: AppCompatActivity) { this.activity = activity this.activity.lifecycle.addObserver(this) } private lateinit var logoutReceiver: BroadcastReceiver override fun onCreate(owner: LifecycleOwner) { logoutReceiver = LogoutReceiver() activity.registerReceiver(logoutReceiver, IntentFilter("ACTION_LOGOUT")) //unregister missing for keeping the sample code smaller! } inner class LogoutReceiver: BroadcastReceiver() { override fun onReceive(context: Context? , intent: Intent?) { activity.finish() } } }Copy the code
Have you ever used delegate mode? Let me know in the comments if you’ve ever suffered from BaseActivity! 😎