橘子味的心
标题:8.13 WIFI Direct开发

创建一个 WIFI Direct 应用程序,包括发现连接点、请求连接、建立连接、发送数据,以及建立对该应用程序广播的 Intent 进行接收的 BroadcastReceiver,需要经过以下步骤。

1. 创建 BroadcastReceiver

需要注意的是,要在 BroadcastReceiver 的构造方法中传入 WifiP2pManager、WifiP2pManager.Channel 以及注册该 BroadcastReceiver 的 Activity 的对象,以便在 BroadcastReceiver 中访问 WIFI 硬件设备并对 Activity 进行更新。

创建 BroadcastReceiver 的代码如下:
  1. public class WiFiDirectBroadcastReceiver extends BroadcastReceiver {
  2.  
  3. private WifiP2pManager manager;
  4. private Channel channel;
  5. private MyWiFiActivity activity;
  6.  
  7. public WiFiDirectBroadcastReceiver (WifiP2pManager manager, Channel channel,MyWifiActivity activity) {
  8. super();
  9. this.manager=manager;
  10. this.channel=channel;
  11. this.activity=activity;
  12. }
  13. @Override
  14. public void onReceive(Context context, Intent intent) {
  15. String action=intent.getAction();
  16.  
  17. if (WifiP2pManager.WIFI_P2P_STATE_CHANGED_ACTI0N.equals (action)) {
  18. //检测 WIFI 功能是否被打开
  19. } else if (WifiP2pManager.WIFI_P2P_PEERS_CHANGED_ACTION.equals (action)) {
  20. //获取当前可用连接点的列表
  21. } else if (WifiP2pManager.WIFI_P2P_CONNECTION_CHANGED_ACTION.equals (action)) {
  22. //建立或者断开连接
  23. } else if (WifiP2pManager.WIFI_P2P_THIS_DEVICE_CHANGED_ACTION.equals (action)) {
  24. //当前设备的 WIFI 状态发生变化
  25. }
  26. }
  27. }

2. 初始化操作

1)修改 AndroidManifest.xml 文件

指定支持 WIFI Direct 的 Android SDK 的最小版本并增加使用 WIFI Direct 的相应权限,代码如下:
  1. <uses-sdk android:minSdkVersion="14" />
  2. <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
  3. <uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
  4. <uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" />
  5. <uses-permission android:name="android.permission.INTERNET" />
  6. <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
2)确认当前设备是否支持并且打开了 WIFI Direct 功能

相关代码应该被放在 BroadcastReceiver 的 onReceive() 方法中。实例代码如下:
  1. public void onReceive (Context context, Intent intent) {
  2. ...
  3. String action=intent.getAction();
  4.  
  5. if (WifiP2pManager.WIFI_P2P_STATE_CHANGED_ACTI0N.equals (action)) {
  6. int state=intent.getIntExtra (WifiP2pManager.EXTRA_WIFI_STATE, -1);
  7. if (state==WifiP2pManager.WIFI_P2P_STATE_ENABLED) {
  8. // Wifi Direct is enabled
  9. } else {
  10. // Wi-Fi Direct is not enabled
  11. }
  12. }
  13. ...
  14. }
3)在 Activity 的 onCreate() 方法中创建对象

创建 WifiP2pManager 和 Channel 对象,并创建 BroadcastReceiver 对象,代码如下:
  1. WifiP2pManager mManager;
  2. Channel mChannel;
  3. BroadcastReceiver mReceiver;
  4. ...
  5.  
  6. @Override
  7. protected void onCreate (Bundle savedInstanceState) {
  8. ...
  9. mManager= (WifiP2pManager) getSystemService (Context.WIFI_P2P_SERVICE);
  10. mChannel=mManager.initialize (this, getMainLooper(), null);
  11. mReceiver=new WiFiDirectBroadcastReceiver (manager, channel, this);
  12. ...
  13. }
4)创建 BroadcastReceiver 要使用的 IntentFilter 对象

代码如下:
  1. IntentFilter mIntentFilter;
  2. ...
  3. @Override
  4. protected void onCreate (Bundle savedInstanceState) {
  5. ...
  6. mIntentFilter=new IntentFilter();
  7. mIntentFilter.addAction (WifiP2pManager.WIFI_P2P_STATE_CHANGED_ACTI0N);
  8. mIntentFilter.addAction (WifiP2pManager.WIFI_P2P_PEERS_CHANGED_ACTI0N);
  9. mIntentFilter.addAction (WifiP2pManager.WIFI_P2P_C0NNECTI0N_CHANGED_ACTI0N);
  10. mIntentFilter.addAction (WifiP2pManager.WIFI_P2P_THIS_DEVICE_CHANGED_ACTI0N);
  11. ...
  12. }
5)在 Activity 的 onResume() 方法中注册 BroadcastReceiver 对象,在 onPause() 方法中注销对象

代码如下:
  1. @Override
  2. protected void onResume(){
  3. super.onResume();
  4. registerReceiver(mReceiver, mIntentFilter);
  5. }
  6.  
  7. /*unregister the broadcast receiver */
  8. @Override
  9. protected void onPause(){
  10. super.onPause();
  11. unregisterReceiver(mReceiver);
  12. }

3. 使用 WifiP2pManager.discoverPeers() 方法获取可以连接点的列表

示例代码如下:
  1. manager.discoverPeers (channel, new WifiP2pManager.ActionListener() {
  2. @Override
  3. public void onSuccess(){
  4. ...
  5. }
  6.  
  7. @Override
  8. public void onFailure (int reasonCode) {
  9. ...
  10. }
  11. });
若成功搜寻到可以连接的点,则 WIFI Direct 系统框架会广播一个带有 WIFI_P2P_ PEERS_CHANGED_ACTION 信息的 Intent,该 Intent 会被之前定义的 BoradcastReceiver 接收,并获得可以连接点的列表。示例代码如下:
  1. PeerListListener myPeerListListener;
  2. ...
  3. if (WifiP2pManager.WIFI_P2P_PEERS_CHANGED_ACTION.equals (action)) {
  4. if (manager !=null) {
  5. manager.requestPeers (channel, myPeerListListener);
  6. }
  7. }

4. WifiP2pManager.connect() 方法可以与列表中的某个连接点设备建立连接,该方法通过 WifiP2pConfig 对象获得连接设备的相关信息

示例代码如下:
  1. WifiP2pDevice device;
  2. WifiP2pConfig config=new WifiP2pConfig();
  3. config.deviceAddress=device.deviceAddress;
  4. manager.connect (channel, config, new ActionListener () {
  5.  
  6. @Override
  7. public void onSuccess(){
  8. //success logic
  9. }
  10.  
  11. @Override
  12. public void onFailure (int reason) {
  13. //failure logic
  14. }
  15. });

5. 连接建立后,就可以用两个设备直接通过 Socket 进行数据传输

其传输过程与之前讲解的 Socket 通信完全相同,基本步骤如下:

1)在其中一个设备上建立 ServerSocket 对象,监听特定端口,并堵塞应用程序,直到有连接请求。

2)在另一个设备上建立 Socket 对象,通过 IP 地址和端口向 ServerSocket 发出连接请求。

3)ServerSocket 监听到连接请求后,调用 accept() 方法建立连接。

4)连接建立后,Socket 对象可以通过字节流在两个设备间直接进行数据传递。

下面的示例代码演示了通过 ServerSocket 和 Socket 在客户端和服务器间直接传递 JPG 图像的过程。

服务器代码如下:
  1. public static class FileServerAsyncTask extends AsyncTask {
  2. private Context context;
  3. private TextView statusText;
  4.  
  5. public FileServerAsyncTask(Context context, View statusText) {
  6. this.context = context;
  7. this.statusText = (TextView) statusText;
  8. }
  9.  
  10. @Override
  11. protected String doInBackground(Void... params) {
  12. try {
  13. //创建 ServerSocket 对象,监听 8888 端口,等待客户连接
  14. ServerSocket serverSocket = new ServerSocket(8888);
  15. Socket client = serverSocket.accept();
  16. //建立连接成功,开始传送数据
  17. final File f = new File(Environment.getExternalStorageDirectory() + "/"
  18. + context.getPackageName() + "/wifip2pshared-" + System.currentTimeMillis() + ".jpg");
  19. File dirs = new File(f.getParent());
  20. if (!dirs.exists())
  21. dirs.mkdirs();
  22. f.createNewFile();
  23. InputStream inputstream = client.getlnputStream();
  24. copyFile(inputstream, new FileOutputStream(f));
  25. ServerSocket.close();
  26. return f.getAbsolutePath();
  27. } catch (IOException e) {
  28. Log.e(WiFiDirectActivity.TAG, e.getMessage());
  29. return null;
  30. }
  31. }
  32.  
  33. //启动用于显示图像的Activity
  34. @Override
  35. protected void onPostExecute(String result) {
  36. if (result != null) {
  37. statusText.setText("File copied - " + result);
  38. Intent intent = new Intent();
  39. intent.setAction(android.content.Intent.ACTION_VIEW);
  40. intent.setDataAndType(Uri.parse("file://" + result),"image/*");
  41. context.startActivity(intent);
  42. }
  43. }
  44. }
客户端的相关代码如下:
  1. Context context=this.getApplicationContext();
  2. String host;
  3. int port;
  4. int len;
  5. Socket socket=new Socket();
  6. byte buf[]=new byte[1024];
  7. ...
  8. try{
  9. //创建Socket对象,并请求连接
  10. socket.bind (null);
  11. socket.connect((new InetSocketAddress(host,port)),500);
  12. //连接建立成功,开始传输数据
  13. OutputStream outputStream=socket.getOutputStream();
  14. ContentResolver cr=context.getContentResolver();
  15. Inputstream inputStream=null;
  16. inputstream=cr.openlnputStream(Uri.parse("path/to/picture.jpg"))while((len=inputStream.read(buf))!=-l){outputStream.write(buf,0,len);
  17. outputStream.close();
  18. inputstream.close();
  19. }catch(FileNotFoundException e){
  20. //catch logic
  21. } catch (IOException e) {
  22. //catch logic
  23. }
  24. //关闭连接
  25. finally{
  26. if(socket!=null){
  27. if(socket.isConnected()){
  28. try{
  29. socket.close();
  30. }catch(IOException e){
  31. //catch logic
  32. }
  33. }
  34. }
  35. }