Wi-Fi Direct允许Android4.0(API Level 14)以后的设备,使用相应的硬件通过Wi-Fi直接的彼此相连,而不需要中间访问点。当每个都设备支持Wi-Fi Direct时,使用这些API就能够发现并连接另一个对等设备,而且通信距离要远远超过蓝牙连接。这对于要在用户间共享应用程序的数据是有用的,如多人游戏或图片共享应用程序等。
Wi-Fi Direct由以下主要部分组成:
在WifiP2pManager类中定义了用于彼此对等发现、请求、连接的方法;
用于获取WifiP2pManager方法调用结果(成功或失败)通知的监听器。当调用WifiP2pManager方法时,每个方法都能够接收一个作为参数传入的特殊监听器;
用于通知你Wi-Fi Direct框架所检测到的特定事件的Intent对象,如删除链接或发现新设备。
这个三个主要的API组件经常要一起使用。例如,给discoverPeers()方法提供一个WifiP2pManager.ActionListener监听器,以便能够用ActionListener.onSuccess()和ActionListener.onFailure()方法来获取通知。如果discoverPeers()方法发现对等设备列表发生改变,还会发出一个WIFI_P2P_PEERS_CHANGED_ACTION类型的Intent广播。
API概要
WifiP2pManager提供了一些用于跟设备的Wi-Fi硬件相互作用的方法,这些方法完成设备的彼此发现和连接。以下是操作是有效的:
表1.Wi-Fi Direct方法
方法 介绍
initialize()
把应用程序注册到Wi-Fi框架中,它必须在调用其他Wi-Fi Direct方法之前调用。
connect()
用指定的配置来启动设备间的对等连接。
cancelConnect()
取消任何进行中的对等设备间连接请求。
requestConnectInfo()
请求设备的连接信息。
createGroup()
用当前设备作为组管理员来创建一个对等组。
removeGroup()
删除当期对等设备组。
requestGroupInfo()
请求对等组信息。
discoverPeers()
启动对等点的发现。
requestPeers()
请求当前发现的对等点的列表。
WifiP2pManager类的方法会让你传入一个监听器,以便Wi-Fi Direct框架能够通知你的Activity该调用的状态。下表介绍了WifiP2pManager类的方法调用时可用的监听器接口:
表2.Wi-Fi Direct监听器
监听器接口
关联的操作
WifiP2pManager.ActionListener
connect(),cancelConnect(),createGroup(),removeGroup(), and discoverPeers()
WifiP2pManager.ChannelListener
initialize()
WifiP2pManager.ConnectionInfoListener
requestConnectInfo()
WifiP2pManager.GroupInfoListener
requestGroupInfo()
WifiP2pManager.PeerListListener
requestPeers()
Wi-Fi Direct API还定义了一些在某些Wi-Fi Direct事件发生时,用广播的形式发出的Intent对象,如在发现对等设备或设备的Wi-Fi状态发生变化时。你能够在你的应用程序中注册接收这些Intent对象,并通过创建广播接收器来处理这些Intent对象:
表3.Wi-Fi Direct Intent
Intent 介绍
WIFI_P2P_CONNECTION_CHANGED_ACTION
在设备的Wi-Fi连接状态变化时,发出这个广播。
WIFI_P2P_PEERS_CHANGED_ACTION
在调用discoverPeers()方法时,发出这个广播,如果你要在应用程序中处理这个Intent,通常是希望调用requestPeers()方法来获取对等设备的更新列表。
WIFI_P2P_STATE_CHANGED_ACTION
当启用或禁用设备上的Wi-Fi Direct时,发出这个广播。
WIFI_P2P_THIS_DEVICE_CHANGED_ACTION
当设备的细节(如设备的名称)发生变化时,发出这个广播。
给Wi-Fi Direct的Intent创建广播接收器
广播接收器允许你接收由Android系统发出的Intent广播,以便你的应用程序能够响应你感兴趣的事件。以下是创建处理Wi-Fi Direct的Intent接收器的基本步骤:
创建一个继承BroadcastReceiver类的类。这个类的构造器,要有WifiP2pManager、WifiP2pManager.Channel和Activity类型的参数。这样就允许广播接收器把更新发送给Activity以及要访问的Wi-Fi硬件和通信通道。
在广播接收器的onReceive()方法中检查你感兴趣的Intent对象,根据接收到的Intent来执行必要的操作。例如,如果广播接收器接收到一个WIFI_P2P_PEERS_CHANGED_ACTION类型的Intent,就可以调用requestPeers()方法来获取当前发现的对等设备的列表。
下列代码显示了如何创建一个典型的广播接收器。该广播接收器需要一个WifiP2pManager对象和一个Activity对象作为参数,并且使用这两个类在广播接收器接收到Intent时,执行合适的需要的操作:
/**
- A BroadcastReceiver that notifies of important Wi-Fi p2p events.
*/
public class WiFiDirectBroadcastReceiver extends BroadcastReceiver {
private WifiP2pManager manager;
private Channel channel;
private MyWiFiActivity activity;
public WiFiDirectBroadcastReceiver(WifiP2pManager manager, Channel channel,MyWifiActivity activity) {
super();
this.manager = manager;
this.channel = channel;
this.activity = activity;
}
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
if (WifiP2pManager.WIFI_P2P_STATE_CHANGED_ACTION.equals(action)) {
} else if (WifiP2pManager.WIFI_P2P_PEERS_CHANGED_ACTION.equals(action)) {
} else if (WifiP2pManager.WIFI_P2P_CONNECTION_CHANGED_ACTION.equals(action)) {
} else if (WifiP2pManager.WIFI_P2P_THIS_DEVICE_CHANGED_ACTION.equals(action)) {
}
}
}
创建Wi-Fi Direct应用程序
创建Wi-Fi Direct应用程序涉及到给应用程序创建和注册广播接收器、发现对等设备、连接对等设备和把数据传送给对等设备。下面会介绍如何完成这些事情。
初始安装
在使用Wi-Fi Direct API之前,必须确保你的应用程序能够访问硬件,并且该设备要支持Wi-Fi Direct协议。如果支持Wi-Fi Direct,你就可以获得一个WifiP2pManager实例,然后创建和注册你的广播接收器,开始使用Wi-Fi Direct API。
- 在Android清单中申请使用设备上Wi-Fi硬件的权限,并声明要使用的最小的SDK版本:
<uses-sdkandroid:minSdkVersion="14"/>
<uses-permissionandroid:name="android.permission.ACCESS_WIFI_STATE"/>
<uses-permissionandroid:name="android.permission.CHANGE_WIFI_STATE"/>
<uses-permissionandroid:name="android.permission.CHANGE_NETWORK_STATE"/>
<uses-permissionandroid:name="android.permission.INTERNET"/>
<uses-permissionandroid:name="android.permission.ACCESS_NETWORK_STATE"/>
- 检查是否支持Wi-Fi Direct。做这项检查的一个好的位置是,接收WIFI_P2P_STATE_CHANGED_ACTION类型的Intent的广播接收器中,把Wi-Fi Direct的状态通知给你的Activity,并作出相应的反应:
@Override
publicvoid onReceive(Context context,Intent intent){
...
String action = intent.getAction();
if(WifiP2pManager.WIFI_P2P_STATE_CHANGED_ACTION.equals(action)){
int state = intent.getIntExtra(WifiP2pManager.EXTRA_WIFI_STATE,-1);
if(state ==WifiP2pManager.WIFI_P2P_STATE_ENABLED){
}else{
}
}
...
}
- 在你的Activity的onCreate()方法中,获得一个WifiP2pManager的实例,并且要调用initialize()方法把你的应用程序注册到Wi-Fi Direct框架中。这个方法会返回一个WifiP2pManager.Channel对象,它用于把应用程序连接到Wi-Fi Direct框架。你还应该创建一个带有WifiP2pManager和WifiP2pManager.Channel对象以及你的Activity的引用的广播接收器实例。这样就允许你的广播接收器根据变化把你感兴趣的事件通知给你的Activity。如果需要,你还可以维护设备的Wi-Fi状态:
WifiP2pManager mManager;
Channel mChannel;
BroadcastReceiver mReceiver;
...
@Override
protectedvoid onCreate(Bundle savedInstanceState){
...
mManager =(WifiP2pManager) getSystemService(Context.WIFI_P2P_SERVICE);
mChannel = mManager.initialize(this, getMainLooper(),null);
mReceiver =newWiFiDirectBroadcastReceiver(manager, channel,this);
...
}
- 创建一个Intent过滤器,并给这个Intent添加你的广播接收器要检查操作:
IntentFilter mIntentFilter;
...
@Override
protectedvoid onCreate(Bundle savedInstanceState){
...
mIntentFilter =newIntentFilter();
mIntentFilter.addAction(WifiP2pManager.WIFI_P2P_STATE_CHANGED_ACTION);
mIntentFilter.addAction(WifiP2pManager.WIFI_P2P_PEERS_CHANGED_ACTION);
mIntentFilter.addAction(WifiP2pManager.WIFI_P2P_CONNECTION_CHANGED_ACTION);
mIntentFilter.addAction(WifiP2pManager.WIFI_P2P_THIS_DEVICE_CHANGED_ACTION);
...
}
- 在你的Activity的onResume()方法中注册广播接收器,并且要在你的Activity的onPause()方法注销它:
@Override
protectedvoid onResume(){
super.onResume();
registerReceiver(mReceiver, mIntentFilter);
}
@Override
protectedvoid onPause(){
super.onPause();
unregisterReceiver(mReceiver);
}
当你已经获得了一个WifiP2pManager.Channel对象并建立一个广播接收器时,你的应用程序就能够调用Wi-Fi Direct方法和接收Wi-Fi Direct的Intent对象了。
现在,通过调用WifiP2pManager对象中的方法,你能够使用Wi-Fi Direct功能了。接下来向你介绍如何使用发现和连接对等设备等通用操作。
发现对等设备
要发现有效的可连接的对等设备,就要调用discoverPeers()方法,在一定的范围内检查有效的对等设备。这个功能调用是异步的,如果你创建了WifiP2pManager.ActionListener监听器,那么你的应用程序就会使用onSuccess()和onFailure()方法来完成成功或失败的传递。onSuccess()方法只会通知你,发现处理成功了,它并不提供发现的相关实际对等设备的任何信息:
manager.discoverPeers(channel,newWifiP2pManager.ActionListener(){
@Override
publicvoid onSuccess(){
...
}
@Override
publicvoid onFailure(int reasonCode){
...
}
});
如果发现处理成功,并检测到对等设备,系统会广播WIFI_P2P_PEERS_CHANGED_ACTION类型的Intent,你能够在一个广播接收器中监听这个Intent,以便获得对等设备的列表。当你的应用程序接收到这个Intent时,你能够使用requestPeers()方法来请求被发现的对等设备的列表。下列代码显示了如何做这件事:
PeerListListener myPeerListListener;
...
if(WifiP2pManager.WIFI_P2P_PEERS_CHANGED_ACTION.equals(action)){
if(manager !=null){
manager.requestPeers(channel, myPeerListListener);
}
}
requestPeers()方法也是异步的,有效的对等设备列表是使用onPeersAvailable()回调来通知你的Activity的,这个回调方法是在WifiP2pManager.PeerListListener接口中定义的。onPeersAvailable()方法会给你提供一个WifiP2pDeviceList对象,通过迭代该对象就能够找到你想要连接的对等设备。
连接对等设备
在获得可能的对等设备列表之后,并从中找到了你想要连接的设备时,就要调用connect()方法来连接设备。调用这个方法需要一个WifiP2pConfig对象,该对象包含了要连接的设备的信息。通过WifiP2pManager.ActionListener监听器,你能够获得连接成功或失败的通知。下列代码显示了如何创建跟期望的设备的连接:
WifiP2pDevice device;
WifiP2pConfig config =newWifiP2pConfig();
config.deviceAddress = device.deviceAddress;
manager.connect(channel, config,newActionListener(){
@Override
publicvoid onSuccess(){
}
@Override
publicvoid onFailure(int reason){
}
});
传输数据
一旦建立了连接,就能够使用套接字在设备之间传输数据。基本的步骤如下:
创建一个ServerSocket对象。这个套接字会在指定的端口上等待来自客户端的连接,并且要一直阻塞到连接发生,因此要在后台线程中做这件事。
创建一个客户端的Socket对象。该客户端要使用服务套接字的IP地址和端口来连接服务端设备。
把数据从客户端发送给服务端。当客户端套接字跟服务端套接字成功的建立了连接,你就能够以字节流的形式,把数据从客户端发送给服务端了。
服务套接字等待客户端的连接(用accept()方法)。这个调用会一直阻塞到客户端的连接发生,因此这个调用要放到另外一个线程中。当连接发生时,服务端能够接收来自客户端的数据。并对这个数据执行一些操作,如保存到文件或展现给用户。
下例来自Wi-Fi Direct Demo示例,向你展示了如何创建这种客户-服务套接字的通信,并从客户端把JPEG图片传输给服务端。完整的示例请编译和运行Wi-Fi Direct Demo示例:
public static class FileServerAsyncTask extends AsyncTask {
private Context context;
private TextView statusText;
public FileServerAsyncTask(Context context, View statusText) {
this.context = context;
this.statusText = (TextView) statusText;
}
@Override
protected String doInBackground(Void... params) {
try {
ServerSocket serverSocket = new ServerSocket(8888);
Socket client = serverSocket.accept();
final File f = new File(Environment.getExternalStorageDirectory() + "/"
+ context.getPackageName() + "/wifip2pshared-" + System.currentTimeMillis()
+ ".jpg");
File dirs = new File(f.getParent());
if (!dirs.exists())
dirs.mkdirs();
f.createNewFile();
InputStream inputstream = client.getInputStream();
copyFile(inputstream, new FileOutputStream(f));
serverSocket.close();
return f.getAbsolutePath();
} catch (IOException e) {
Log.e(WiFiDirectActivity.TAG, e.getMessage());
return null;
}
}
@Override
protected void onPostExecute(String result) {
if (result != null) {
statusText.setText("File copied - " + result);
Intent intent = new Intent();
intent.setAction(android.content.Intent.ACTION_VIEW);
intent.setDataAndType(Uri.parse("file://" + result), "image/*");
context.startActivity(intent);
}
}
}
在客户端,客户套接字连接到服务套接字,并传输数据。这个例子是把客户端设备的文件系统上的一个JPEG文件传输给服务端。
Context context =this.getApplicationContext();
String host;
int port;
int len;
Socket socket =newSocket();
byte buf[] =newbyte[1024];
...
try{
socket.bind(null);
socket.connect((newInetSocketAddress(host, port)),500);
OutputStream outputStream = socket.getOutputStream();
ContentResolver cr = context.getContentResolver();
InputStream inputStream =null;
inputStream = cr.openInputStream(Uri.parse("path/to/picture.jpg"));
while((len = inputStream.read(buf))!=-1){
outputStream.write(buf,0, len);
}
outputStream.close();
inputStream.close();
}catch(FileNotFoundException e){
}catch(IOException e){
}
finally{
if(socket !=null){
if(socket.isConnected()){
try{
socket.close();
}catch(IOException e){
}
}
}
}