首页
Web开发
Windows程序
编程语言
数据库
移动开发
系统相关
微信
其他好文
会员
首页
>
其他好文
> 详细
EventBus事件总线分发库
时间:
2015-08-14 01:23:12
阅读:
261
评论:
0
收藏:
0
[点我收藏+]
标签:
本文围绕以下六个部分展开:
一、事件总线管理
二、EventBus
三、EventBus与BroadcastReceiver的区别
案例一
案例二:一处点击发送数据,另一处或多处注册点可以及时获取更新传输过来的数据
案例三:Activity和Service之间互相发布与接收事件
一、事件总线管理
将事件放入队列里,用于管理和分发。
(1)保证应用的各个部分之间高效的通信及数据、事件分发。
(2)模块间解耦:通过事件的分发,可以让各个模块间关联程序变小。
当在开发一些庞大的的项目时,模块比较多,这个时候为了避免耦合度和保证 APP 的效率,会用到 EventBus 等类似的事件总线处理模型。这样可以简化一些数据传输操作,保证APP的简洁,做到高内聚、低耦合。
二、EventBus
1. 概念
EventBus是一个发布/订阅的事件总线。它可以让两个组件相互通信,但是它们之间并不相互知晓。
EventBus模式,也被称为 Message Bus模式,或者 发布者/订阅者(publisher/subscriber)模式。
事件响应有更多的线程选择,EventBus 可以向不同的线程中发布事件。
EventBus 支持 Sticky Event。
2. 有3个主要的元素:
(1)Event:事件。
Event可以是任意类型的对象。
(2)Subscriber:事件订阅者,接收特定的事件。
在EventBus中,使用约定来指定事件订阅者以简化使用。即,所有事件订阅都是以onEvent开头的函数。具体来说,函数的名字是onEventMainThread,onEventPostThread,onEventBackgroundThread,onEventAsync这四个,onEvent是默认的接收数据处理的方法。这四个事件订阅函数,每个都与一个“ThreadMode”相关联,ThreadMode指定了会调用的函数。有以下4个ThreadMode:
1)PostThread:事件的处理和事件的发送在相同的进程。
事件处理时间不应太长,不然影响事件的发送线程,而这个线程可能是UI线程。对应的函数名是onEvent。
2)MainThread:事件的处理会在UI线程中执行。
事件处理时间不应太长,长了会ANR的。对应的函数名是onEventMainThread。
3)BackgroundThread:事件的处理会在一个后台线程中执行。
虽然事件处理是在后台线程,但事件处理时间还是不应该太长,因为如果发送事件的线程是后台线程,会直接执行事件。如果当前线程是UI线程,事件会被加到一个队列中,由一个线程依次处理这些事件,如果某个事件处理时间太长,会阻塞后面的事件的派发或处理。对应的函数名是onEventBackgroundThread。
4)AsyncThread:事件处理会在单独的线程中执行。
主要用于在后台线程中执行耗时操作,每个事件会开启一个线程(有线程池),但最好限制线程的数目。
根据事件订阅的函数名称的不同,会使用不同的ThreadMode。比如在后台线程加载了数据想在UI线程显示,订阅者只需把函数命名为onEventMainThread。
3. 基本用法
分
注册
、
订阅
、
发布
、
取消注册
等步骤。使用时需先注册、订阅,然后向订阅者发布消息数据即可。
(1)注册(3个构造方法):
EventBus.getDefault().register(this);
EventBus.getDefault().register(new MyClass());
// 注册三个参数分别是:消息订阅者(接收者),接收方法名,事件类。
EventBus.getDefault().register(this,"setTextA",SetTextAEvent.class);
(2)注销注册:
EventBus.getDefault().unregister(this);
EventBus.getDefault().unregister(new MyClass());
(3)定义事件类型:
public class MyEvent {}
(4)订阅处理数据:
public void onEvent(AnyEventType event){}:默认的接收数据处理的方法
public void onEventMainThread{}
onEventPostThread
onEventBackgroundThread
onEventAsync
(5)发布:
EventBus.getDefault().postSticky(new SecondActivityEvent("Message from SecondActivity"));
EventBus.getDefault().post(new ChangeImgEvent(1));
三、EventBus与BroadcastReceiver的区别
1. 二者类似。
在Android中广播分为两个方面:广播发送者和广播接收者,通常情况下,BroadcastReceiver指的就是广播接收者(广播接收器)。
广播:在一个地方注册广播,在另一个地方针对action发送广播、传递数据,对应注册的地方就可以收到广播。
EventBus:基于订阅/发布的模式。在需要接收数据的地方,进行注册与订阅,在需要发布数据的地方发布,则在注册的地方就可以收到了。
简单点说,就是两人约定好怎么通信,一人发布消息,另外一个约定好的人立马接收到你发的消息。EventBus就可以帮助减少做很多事,不管你在任何地方任何位置发布一个事件,接收者都能立马接收到你的消息,不用你考虑android子线程操作UI线程的问题。
2. 二者区别。
(1)EventBus 有三个主要的元素:事件、订阅和发布。广播两个元素:订阅和发布,但是广播是针对整个App而言的。
(2)BroadcastReceiver是组件,需要在功能清单中注册。而EventBus 不需要注册。
(3)BroadcastReceiver只能做一件事情,而EventBus多事件处理比较好。
(4)在不同场景中的适用性:
1)同一App内部的同一组件内的消息通信(单个或多个线程之间),实际应用中肯定是不会用到广播机制的(虽然可以用),无论是使用扩展变量作用域、基于接口的回调还是Handler-post/Handler-Message等方式,都可以直接处理此类问题,若使用广播机制,显然有些“杀鸡牛刀”的感觉。
2)同一app内部的不同组件之间的消息通信(单个进程),对于此类需求,在有些教复杂的情况下单纯的依靠基于接口的回调等方式不好处理,此时可以直接使用EventBus等,相对而言,EventBus由于是针对同一进程,用于处理此类需求非常适合,且轻松。
3)其他情形,由于涉及不同进程间的消息通信,此时根据实际业务使用广播机制会显得非常适宜。
案例一
MainActivity
Java代码
public
class
MainActivity
extends
Activity {
@Override
protected
void
onCreate(Bundle savedInstanceState) {
super
.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 1. 注册
EventBus.getDefault().register(
this
);
}
/**
* 3. 发送数据消息事件
*/
private
void
postData() {
String string =
"我是消息"
;
EventBus.getDefault().post(string);
}
/**
* 2. 接收数据消息事件
*/
public
void
onEvent(String string) {
}
public
void
onEventMainThread(String string) {
}
public
void
onEventPostThread(String string) {
}
public
void
onEventBackgroundThread(String string) {
}
public
void
onEventAsync(String string) {
}
@Override
protected
void
onDestroy() {
super
.onDestroy();
// 4. 取消注册
EventBus.getDefault().unregister(
this
);
}
}
案例二:一处点击发送数据,另一处或多处注册点可以及时获取更新传输过来的数据
1. activity_main.xml。写一个按钮和一个文本框。
Xml代码
<
RelativeLayout
xmlns:android
=
"http://schemas.android.com/apk/res/android"
xmlns:tools
=
"http://schemas.android.com/tools"
android:layout_width
=
"match_parent"
android:layout_height
=
"match_parent"
android:paddingBottom
=
"@dimen/activity_vertical_margin"
android:paddingLeft
=
"@dimen/activity_horizontal_margin"
android:paddingRight
=
"@dimen/activity_horizontal_margin"
android:paddingTop
=
"@dimen/activity_vertical_margin"
tools:context
=
".MainActivity"
>
<
Button
android:id
=
"@+id/btnSend"
android:layout_width
=
"match_parent"
android:layout_height
=
"wrap_content"
android:text
=
"发送事件"
/>
<
TextView
android:id
=
"@+id/tvContent"
android:layout_width
=
"match_parent"
android:layout_height
=
"wrap_content"
android:layout_below
=
"@id/btnSend"
android:gravity
=
"center"
android:textSize
=
"20sp"
/>
</
RelativeLayout
>
2. MainActivity。声明和初始化两个控件。
Java代码
private
TextView tvContent;
private
Button btnSend;
Java代码
tvContent = (TextView) findViewById(R.id.tvContent);
btnSend = (Button) findViewById(R.id.btnSend);
3. MainActivity。onCreate()方法中注册。
Java代码
EventBus.getDefault().register(
this
);
4. MainActivity。onDestroy()方法中取消注册。
Java代码
@Override
protected
void
onDestroy() {
super
.onDestroy();
EventBus.getDefault().unregister(
this
);
}
5. MainActivity。通过事件监听器设置按钮的点击事件。
Java代码
btnSend.setOnClickListener(
new
View.OnClickListener() {
@Override
public
void
onClick(View v) {
}
});
6. 创建自定义的事件类:MyEvent。
Java代码
package
com.android.eventbustwo;
/**
* 自定义的事件类
*/
public
class
MyEvent {
private
String type;
private
String content;
public
String getType() {
return
type;
}
public
void
setType(String type) {
this
.type = type;
}
public
String getContent() {
return
content;
}
public
void
setContent(String content) {
this
.content = content;
}
}
7. MainActivity。在按钮点击事件中,创建自定义事件类MyEvent的实例事件,并当点击按钮的时候,发送事件。
Java代码
btnSend.setOnClickListener(
new
View.OnClickListener() {
@Override
public
void
onClick(View v) {
MyEvent event =
new
MyEvent();
event.setType(
"0"
);
event.setContent(
"0内容"
);
// 发送数据事件
EventBus.getDefault().post(event);
}
});
8. MainActivity。通过onEvent()订阅(接收)事件。
Java代码
public
void
onEvent(MyEvent event){
if
(event.getType().equals(
"0"
)){
tvContent.setText(event.getContent());
}
}
运行效果如下:
9. MainActivity。在按钮点击事件中,将事件类型和内容改变,并当点击按钮的时候,发送事件。
Java代码
btnSend.setOnClickListener(
new
View.OnClickListener() {
@Override
public
void
onClick(View v) {
MyEvent event =
new
MyEvent();
event.setType(
"1"
);
event.setContent(
"1内容"
);
// 发送数据事件
EventBus.getDefault().post(event);
}
});
10. MainActivity。通过onEventMainThread()同样可以订阅(接收)事件。
Java代码
public
void
onEventMainThread(MyEvent event){
if
(event.getType().equals(
"1"
)){
tvContent.setText(event.getContent());
}
}
运行效果如下:
代码补充:
1. activity_main.xml
2. MainActivity
Java代码
package
com.android.eventbustwo;
import
android.app.Activity;
import
android.os.Bundle;
import
android.view.Menu;
import
android.view.MenuItem;
import
android.view.View;
import
android.widget.Button;
import
android.widget.TextView;
import
de.greenrobot.event.EventBus;
public
class
MainActivity
extends
Activity {
// 2.
private
TextView tvContent;
private
Button btnSend;
@Override
protected
void
onCreate(Bundle savedInstanceState) {
super
.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 3.
tvContent = (TextView) findViewById(R.id.tvContent);
btnSend = (Button) findViewById(R.id.btnSend);
// 6.
btnSend.setOnClickListener(
new
View.OnClickListener() {
@Override
public
void
onClick(View v) {
// 8.
MyEvent event =
new
MyEvent();
event.setType(
"0"
);
event.setContent(
"0内容"
);
// 发送数据事件
EventBus.getDefault().post(event);
}
});
// 4.
EventBus.getDefault().register(
this
);
}
/**
* 9.
* @param event
*/
public
void
onEvent(MyEvent event){
if
(event.getType().equals(
"0"
)){
tvContent.setText(event.getContent());
}
}
/**
* 10.
* @param event
*/
public
void
onEventMainThread(MyEvent event){
if
(event.getType().equals(
"1"
)){
tvContent.setText(event.getContent());
}
}
/**
* 5.
*/
@Override
protected
void
onDestroy() {
super
.onDestroy();
EventBus.getDefault().unregister(
this
);
}
// -----------------------------------------------------------
@Override
public
boolean
onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.menu_main, menu);
return
true
;
}
@Override
public
boolean
onOptionsItemSelected(MenuItem item) {
// Handle action bar item clicks here. The action bar will
// automatically handle clicks on the Home/Up button, so long
// as you specify a parent activity in AndroidManifest.xml.
int
id = item.getItemId();
//noinspection SimplifiableIfStatement
if
(id == R.id.action_settings) {
return
true
;
}
return
super
.onOptionsItemSelected(item);
}
}
3. MyEvent
Java代码
package
com.android.eventbustwo;
/**
* 7. 自定义的事件类
*/
public
class
MyEvent {
private
String type;
private
String content;
public
String getType() {
return
type;
}
public
void
setType(String type) {
this
.type = type;
}
public
String getContent() {
return
content;
}
public
void
setContent(String content) {
this
.content = content;
}
}
案例三:Activity和Service之间互相发布与接收事件
主活动有4家公司名显示,当在文本框中输入“Amazon”,点击按钮后,将新的公司名显示在主活动界面上。
1. 创建一个Service: EventService。
2. EventService。将onBind中的异常去掉,改为 返回null。
Java代码
@Override
public
IBinder onBind(Intent intent) {
return
null
;
}
3. activity_main.xml。一个显示的文本框,用于显示Service发布的事件;一个可编辑文本框,用于输入字符串,并发布事件到Service;一个按钮,通过点击按钮,发布事件,然后Service进行订阅。
Xml代码
<
RelativeLayout
xmlns:android
=
"http://schemas.android.com/apk/res/android"
xmlns:tools
=
"http://schemas.android.com/tools"
android:layout_width
=
"match_parent"
android:layout_height
=
"match_parent"
android:paddingBottom
=
"@dimen/activity_vertical_margin"
android:paddingLeft
=
"@dimen/activity_horizontal_margin"
android:paddingRight
=
"@dimen/activity_horizontal_margin"
android:paddingTop
=
"@dimen/activity_vertical_margin"
tools:context
=
".MainActivity"
>
<
TextView
android:id
=
"@+id/tvCompany"
android:layout_width
=
"wrap_content"
android:layout_height
=
"wrap_content"
android:text
=
"@string/hello_world"
/>
<
EditText
android:id
=
"@+id/txtCompany"
android:layout_width
=
"match_parent"
android:layout_height
=
"wrap_content"
android:layout_below
=
"@id/tvCompany"
android:hint
=
"@string/txt_company"
/>
<
Button
android:id
=
"@+id/btnCompany"
android:layout_width
=
"wrap_content"
android:layout_height
=
"wrap_content"
android:layout_below
=
"@id/txtCompany"
android:text
=
"@string/btn_send_company"
/>
</
RelativeLayout
>
4. EventService。重写onCreate()、onStartCommand()、onDestroy()方法。
5. EventService。声明一个 List 集合,用于放公司名的集合,并在onCreate()方法中创建实例(ArrayList类型)。
Java代码
private
List<String> list;
Java代码
list =
new
ArrayList<>();
6. EventService。在onStartCommand()方法中,往List集合中添加元素(公司名)。销毁(onDestroy())的时候,集合设为null。
Java代码
list.add(
"Google"
);
list.add(
"Facebook"
);
list.add(
"Twitter"
);
list.add(
"Apple"
);
Java代码
list =
null
;
8. EventService。onCreate()中注册EventBus,onDestroy()销毁时注销注册。
Java代码
EventBus.getDefault().register(
this
);
Java代码
EventBus.getDefault().unregister(
this
);
9. MainActivity。onCreate()中同样注册EventBus,onDestroy()销毁时同样注销注册。
10. 新建一个目录:event,用来放事件类。
11. 在event目录下,新建一个用于从 Service 发布集合数据(公司名的集合)集合事件类:ListEvent。
Java代码
public
class
ListEvent {
// 集合
private
List<String> list;
// 含参构造方法
public
ListEvent(List<String> list) {
this
.list = list;
}
// getter方法(只需getter方法)
public
List<String> getList() {
return
list;
}
}
12. 在event目录下,新建一个用于从 Activity 发布字符串(公司名)的集合事件类:CompanyEvent。
Java代码
public
class
CompanyEvent {
// 字符串 company
private
String company;
// 含参构造方法
public
CompanyEvent(String company) {
this
.company = company;
}
// getter方法(只需getter方法)
public
String getCompany() {
return
company;
}
}
13. MainActivity。在按钮点击事件中,发布事件。
Java代码
@OnClick
(R.id.btnCompany)
public
void
btnCompanyClick() {
// 获得 在 txtCompany 文本框中输入的 company 字符串 的值
String company = txtCompany.getText().toString();
// Activity 发布事件 --- 将 在 txtCompany 文本框中输入的 company 字符串,
// 使用 CompanyEvent 发布过去
EventBus.getDefault().post(
new
CompanyEvent(company));
// 发布之后,将 文本框内容清空
txtCompany.setText(
""
);
}
14. EventService。Service订阅事件,将Activity发布的公司名加入List集合中。然后,再将最新的公司名List集合发布出去。
Java代码
/**
* Service 订阅事件
* @param event
*/
public
void
onEvent(CompanyEvent event){
// 将 事件中传过来的、在 txtCompany 文本框中输入的 company 字符串,放入 list集合中
list.add(event.getCompany());
// 再将 list集合,发布给 Activity,
// 这样,Activity订阅后,就可以在 控制台和界面上 显示最新的 list集合的值
EventBus.getDefault().postSticky(
new
ListEvent(list));
}
15. MainActivity。Activity 订阅了ListEvent事件,所以接收到了Service发布的公司名List集合的值,然后将其显示在控制台和界面的文本上。
Java代码
/**
* Activity 订阅事件
*
* @param event
*/
public
void
onEvent(ListEvent event) {
// 从Service发布的事件中获得 参数值(list集合)
List<String> list = event.getList();
// 日志输出 list集合的值 (控制台显示 list集合的值)
Log.v(TAG, list.toString());
// 将 list集合的值 显示在 tvCompany 文本中。(界面的文本上 显示 list集合的值)
tvCompany.setText(list.toString());
}
代码目录结构:
补充代码:
(1)EventService。
Java代码
package
com.android.eventbusservice;
import
android.app.Service;
import
android.content.Intent;
import
android.os.IBinder;
import
android.util.Log;
import
com.android.eventbusservice.event.CompanyEvent;
import
com.android.eventbusservice.event.ListEvent;
import
java.util.ArrayList;
import
java.util.List;
import
de.greenrobot.event.EventBus;
public
class
EventService
extends
Service {
private
static
final
String TAG =
"MainActivity"
;
// 声明一个 List 集合 -- 用于放公司名的集合
private
List<String> list;
public
EventService() {
}
@Override
public
void
onCreate() {
super
.onCreate();
Log.v(TAG,
"Service.onCreate"
);
// 当前组件注册事件
// (注册事件后,一定要有 onEvent() 方法,否则报错)
EventBus.getDefault().register(
this
);
// 创建实例 - ArrayList类型
list =
new
ArrayList<>();
}
/**
* 此方法系统自动调用,不写也可。可把里面的代码写入 onCreate()里面。
*/
@Override
public
int
onStartCommand(Intent intent,
int
flags,
int
startId) {
Log.v(TAG,
"Service.onStartCommand"
);
// 往集合中添加元素
list.add(
"Google"
);
list.add(
"Facebook"
);
list.add(
"Twitter"
);
list.add(
"Apple"
);
// Service 发布事件 -- 将 list 集合,使用 ListEvent 发送过去
// (ListEvent:用于从 Service 传事件到 Activity)
EventBus.getDefault().postSticky(
new
ListEvent(list));
// 改为:返回 START_STICKY
return
START_STICKY;
}
@Override
public
void
onDestroy() {
super
.onDestroy();
Log.v(TAG,
"Service.onDestroy"
);
// 销毁的时候,取消事件
EventBus.getDefault().unregister(
this
);
// 销毁的时候,集合设为空
list =
null
;
}
/**
* Service 订阅事件
* @param event
*/
public
void
onEvent(CompanyEvent event){
// 将 事件中传过来的、在 txtCompany 文本框中输入的 company 字符串,放入 list集合中
list.add(event.getCompany());
// 再将 list集合,发布给 Activity,
// 这样,Activity订阅后,就可以在 控制台和界面上 显示最新的 list集合的值
EventBus.getDefault().postSticky(
new
ListEvent(list));
}
@Override
public
IBinder onBind(Intent intent) {
return
null
;
}
}
(2)MainActivity。
Java代码
package
com.android.eventbusservice;
import
android.app.Activity;
import
android.content.Intent;
import
android.os.Bundle;
import
android.util.Log;
import
android.view.Menu;
import
android.view.MenuItem;
import
android.widget.EditText;
import
android.widget.TextView;
import
com.android.eventbusservice.event.CompanyEvent;
import
com.android.eventbusservice.event.ListEvent;
import
java.util.List;
import
butterknife.ButterKnife;
import
butterknife.InjectView;
import
butterknife.OnClick;
import
de.greenrobot.event.EventBus;
public
class
MainActivity
extends
Activity {
private
static
final
String TAG =
"MainActivity"
;
@InjectView
(R.id.tvCompany)
TextView tvCompany;
@InjectView
(R.id.txtCompany)
EditText txtCompany;
@Override
protected
void
onCreate(Bundle savedInstanceState) {
super
.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.inject(
this
);
// 在 Activity 中同样注册事件
EventBus.getDefault().register(
this
);
// 启动服务 -- 使用 startService 方式启动
// 启动服务后,activity与service脱钩了。
// 但是可以通过黄油刀进行联系。
startService(
new
Intent(
this
, EventService.
class
));
}
@Override
protected
void
onDestroy() {
super
.onDestroy();
// 在 Activity 中同样注销事件
EventBus.getDefault().unregister(
this
);
}
@OnClick
(R.id.btnCompany)
public
void
btnCompanyClick() {
// 获得 在 txtCompany 文本框中输入的 company 字符串 的值
String company = txtCompany.getText().toString();
// Activity 发布事件 --- 将 在 txtCompany 文本框中输入的 company 字符串,
// 使用 CompanyEvent 发布过去
EventBus.getDefault().post(
new
CompanyEvent(company));
// 发布之后,将 文本框内容清空
txtCompany.setText(
""
);
}
/**
* Activity 订阅事件
*
* @param event
*/
public
void
onEvent(ListEvent event) {
// 从Service发布的事件中获得 参数值(list集合)
List<String> list = event.getList();
// 日志输出 list集合的值 (控制台显示 list集合的值)
Log.v(TAG, list.toString());
// 将 list集合的值 显示在 tvCompany 文本中。(界面的文本上 显示 list集合的值)
tvCompany.setText(list.toString());
}
// ---------------------------------------------------------
@Override
public
boolean
onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.menu_main, menu);
return
true
;
}
@Override
public
boolean
onOptionsItemSelected(MenuItem item) {
// Handle action bar item clicks here. The action bar will
// automatically handle clicks on the Home/Up button, so long
// as you specify a parent activity in AndroidManifest.xml.
int
id = item.getItemId();
//noinspection SimplifiableIfStatement
if
(id == R.id.action_settings) {
return
true
;
}
return
super
.onOptionsItemSelected(item);
}
}
整理时重点参考:
http://www.jikexueyuan.com/course/933.html
EventBus事件总线分发库
标签:
原文地址:http://my.oschina.net/u/2273085/blog/492103
踩
(
0
)
赞
(
0
)
举报
评论
一句话评论(
0
)
登录后才能评论!
分享档案
更多>
2021年07月29日 (22)
2021年07月28日 (40)
2021年07月27日 (32)
2021年07月26日 (79)
2021年07月23日 (29)
2021年07月22日 (30)
2021年07月21日 (42)
2021年07月20日 (16)
2021年07月19日 (90)
2021年07月16日 (35)
周排行
更多
分布式事务
2021-07-29
OpenStack云平台命令行登录账户
2021-07-29
getLastRowNum()与getLastCellNum()/getPhysicalNumberOfRows()与getPhysicalNumberOfCells()
2021-07-29
【K8s概念】CSI 卷克隆
2021-07-29
vue3.0使用ant-design-vue进行按需加载原来这么简单
2021-07-29
stack栈
2021-07-29
抽奖动画 - 大转盘抽奖
2021-07-29
PPT写作技巧
2021-07-29
003-核心技术-IO模型-NIO-基于NIO群聊示例
2021-07-29
Bootstrap组件2
2021-07-29
友情链接
兰亭集智
国之画
百度统计
站长统计
阿里云
chrome插件
新版天听网
关于我们
-
联系我们
-
留言反馈
© 2014
mamicode.com
版权所有 联系我们:gaon5@hotmail.com
迷上了代码!