Andriod的联通性---Wi-Fi Direct

Wi-Fi Direct允许Android4.0(API Level 14)以后的设备,使用相应的硬件通过Wi-Fi直接的彼此相连,而不需要中间访问点。当每个都设备支持Wi-Fi Direct时,使用这些API就能够发现并连接另一个对等设备,而且通信距离要远远超过蓝牙连接。这对于要在用户间共享应用程序的数据是有用的,如多人游戏或图片共享应用程序等。

Wi-Fi Direct由以下主要部分组成:

  1. 在WifiP2pManager类中定义了用于彼此对等发现、请求、连接的方法;

  2. 用于获取WifiP2pManager方法调用结果(成功或失败)通知的监听器。当调用WifiP2pManager方法时,每个方法都能够接收一个作为参数传入的特殊监听器;

  3. 用于通知你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接收器的基本步骤:

  1. 创建一个继承BroadcastReceiver类的类。这个类的构造器,要有WifiP2pManager、WifiP2pManager.Channel和Activity类型的参数。这样就允许广播接收器把更新发送给Activity以及要访问的Wi-Fi硬件和通信通道。

  2. 在广播接收器的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。

  1. 在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"/>

  1. 检查是否支持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{

    }
}
...

}

  1. 在你的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);
...
}

  1. 创建一个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);
...
}

  1. 在你的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){

}

});

传输数据

一旦建立了连接,就能够使用套接字在设备之间传输数据。基本的步骤如下:

  1. 创建一个ServerSocket对象。这个套接字会在指定的端口上等待来自客户端的连接,并且要一直阻塞到连接发生,因此要在后台线程中做这件事。

  2. 创建一个客户端的Socket对象。该客户端要使用服务套接字的IP地址和端口来连接服务端设备。

  3. 把数据从客户端发送给服务端。当客户端套接字跟服务端套接字成功的建立了连接,你就能够以字节流的形式,把数据从客户端发送给服务端了。

  4. 服务套接字等待客户端的连接(用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){

        }
    }
}

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容