朋友要求婚,请我帮忙做个小软件,希望他能控制周边朋友的手机,同时播放求婚用的歌曲。想法挺好的,奈何时间紧急,花了几个小时,帮他实现了一个粗糙的版本,可以控制连接上的手机同时播放、暂停、停止放歌。


        基本想法是使用C/S架构,一台手机做为服务器,其他手机做为客户端,所有客户端用socket连接上服务器,客户端不断读取socket中的数据,解析出命令后控制歌曲的播放。

        实现效果:


android 多设备播放同步 多个手机同步播放音乐_cs

android 多设备播放同步 多个手机同步播放音乐_cs_02


废话不多说,直接贴代码:

服务器:

布局文件:activity_main.xml


<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" 
    android:background="#FFFFFF">
	
    <LinearLayout 
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" 
        android:orientation="horizontal"
        android:layout_marginTop="20dp"
        >

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:textSize="15dp"
            android:textColor="#000000"
            android:text="服务器IP地址:" />

        <TextView
            android:id="@+id/tvIP"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="192.168.0.1" 
            android:textSize="20dp"
            android:textColor="#0000FF"/>
        
    </LinearLayout>
    
    <LinearLayout 
        android:layout_marginTop="20dp"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" 
        android:orientation="horizontal"
        >

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:textSize="15dp"
            android:textColor="#000000"
            android:text="连接的客户端数目:" />

        <TextView
            android:id="@+id/tvClientNum"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="0" 
            android:textSize="20dp"
            android:textColor="#FF0000"/>
        
    </LinearLayout>
    
        
    <Button 
        android:id="@+id/btn_start_play"
        android:layout_gravity="center"
        android:layout_width="250dp" 
        android:layout_height="80dp"
        android:textSize="22dp"
        android:text="开始播放音乐"
        android:layout_marginTop="50dp"
        android:layout_marginBottom="10dp"
        />
    
    <Button 
        android:id="@+id/btn_pause_play"
        android:layout_gravity="center"
        android:layout_width="250dp" 
        android:layout_height="50dp"
        android:textSize="18dp"
        android:text="暂停播放"
        android:layout_marginTop="20dp"
        android:layout_marginBottom="10dp"
        />
    
    <Button 
        android:id="@+id/btn_stop_play"
        android:layout_gravity="center"
        android:layout_width="250dp" 
        android:layout_height="50dp"
        android:textSize="18dp"
        android:text="停止播放"
        android:layout_marginTop="20dp"
        android:layout_marginBottom="10dp"
        />
    
</LinearLayout>


ServerThread.java


package ict.ldj.server;

import java.io.BufferedReader; 
import java.io.IOException; 
import java.io.InputStreamReader; 
import java.net.Socket; 

import android.util.Log;
 
public class ServerThread implements Runnable { 
    // 定义当前线程所处理的Socket  
    private Socket socket = null; 
    // 该线程所处理的Socket所对应的输入流  
    BufferedReader br = null; 
 
    public ServerThread(Socket socket) throws IOException { 
        this.socket = socket; 
        // 初始化该Socket对应的输入流  
        br = new BufferedReader(new InputStreamReader(socket.getInputStream(), 
                "utf-8")); 
    } 
 
    @Override 
    public void run() { 
        try { 
            String content = null; 
            // 采用循环不断从Socket中读取客户端发送过来的数据  
            while ((content = readFromClient()) != null) { 
                // 遍历socketList中的每个Socket,将读到的内容向每个Socket发送一次 
            	Log.w("client msg",content + "\t");
            	
            	if ("QUIT".equals(content)) {
					socket.close();
					ServerMainActivity.socketList.remove(socket);
				}
//                for (Socket s : ServerMainActivity.socketList) { 
//                    OutputStream os = s.getOutputStream(); 
//                    os.write((content + "\n").getBytes("utf-8")); 
//                } 
            } 
        } catch (Exception e) { 
            e.printStackTrace(); 
        } 
    } 
 
    /**
     * 定义读取客户端数据的方法
     * 
     * @return
     */ 
    private String readFromClient() { 
        try { 
            return br.readLine(); 
        } 
        // 如果捕捉到异常,表明该Socket对应的客户端已经关闭  
        catch (Exception e) { 
            // 删除该Socket  
            ServerMainActivity.socketList.remove(socket); 
            e.printStackTrace(); 
        } 
        return null; 
    } 
}

ServerMainActivity.java


package ict.ldj.server;

import java.io.IOException;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.ArrayList;

import android.app.Activity;
import android.app.AlertDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.net.wifi.WifiInfo;
import android.net.wifi.WifiManager;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.os.Process;
import android.util.Log;
import android.view.KeyEvent;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;

public class ServerMainActivity extends Activity {
	// 定义保存所有Socket的集合
	public static ArrayList<Socket> socketList = new ArrayList<Socket>();
	private static final int refresh_num_msd_code = 0x101;
	
	public Button btnStartPlay = null;
	public Button btnPausePlay = null;
	public Button btnStopPlay = null;
	
	public TextView tvIP = null;
	public TextView tvClientNum = null;
	
	//用来更新与服务器连接的客户端个数
	public Handler mHandler = new Handler(){
		public void handleMessage(Message msg) {
			if (msg.what == refresh_num_msd_code) { 
				tvClientNum.setText(socketList.size() + "");
			}
		}
	};
	
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		initWidget();

		new Thread() {
			public void run() {
				try {
					StartListenerSocket();
				} catch (IOException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
			}
		}.start();
	}

	private void initWidget() {
		btnStartPlay = (Button) findViewById(R.id.btn_start_play);
		btnPausePlay = (Button) findViewById(R.id.btn_pause_play);
		btnStopPlay = (Button) findViewById(R.id.btn_stop_play);
		
		tvIP = (TextView) findViewById(R.id.tvIP);
		tvClientNum = (TextView) findViewById(R.id.tvClientNum);
		
		btnStartPlay.setOnClickListener(myOnClickListener);
		btnPausePlay.setOnClickListener(myOnClickListener);
		btnStopPlay.setOnClickListener(myOnClickListener);

        tvIP.setText(getLocalIp());
        
        new Thread(){
        	public void run(){
        		while(true){
        			try {
        				mHandler.sendEmptyMessage(refresh_num_msd_code);
						this.sleep(2000);
					} catch (InterruptedException e) {
						// TODO Auto-generated catch block
						e.printStackTrace();
					}
        		}
        	}
        }.start();
	}
	
	/**
	 * 获取本机的IP地址
	 * @return 本机的IP地址,Sring
	 */
	private String getLocalIp(){
		// 获取wifi服务
		WifiManager wifiManager = (WifiManager) getSystemService(Context.WIFI_SERVICE);
		 //判断wifi是否开启  
        if (!wifiManager.isWifiEnabled()) {  
        	wifiManager.setWifiEnabled(true);    
        }  
        WifiInfo wifiInfo = wifiManager.getConnectionInfo(); 
        int ipAddress = wifiInfo.getIpAddress();   
        
        return intToIp(ipAddress); 
	}
	
	private String intToIp(int i) {       
        return (i & 0xFF ) + "." +       
	      ((i >> 8 ) & 0xFF) + "." +       
	      ((i >> 16 ) & 0xFF) + "." +       
	      ( i >> 24 & 0xFF) ;  
   }  
	
	private OnClickListener myOnClickListener = new OnClickListener() {
		@Override
		public void onClick(View v) {
			// TODO Auto-generated method stub
			switch (v.getId()) {
			case R.id.btn_start_play:
				try {
					for (Socket s : socketList) {
						OutputStream os = s.getOutputStream();
						if (os != null)
							os.write(("START PLAY MUSIC\n").getBytes("utf-8"));
					}
				} catch (Exception e) {
					e.printStackTrace();
				}
				break;
			case R.id.btn_pause_play:
				try {
					for (Socket s : socketList) {
						OutputStream os = s.getOutputStream();
						if (os != null)
							os.write(("PAUSE PLAY MUSIC\n").getBytes("utf-8"));
					}
				} catch (Exception e) {
					e.printStackTrace();
				}
				break;
			case R.id.btn_stop_play:
				try {
					for (Socket s : socketList) {
						OutputStream os = s.getOutputStream();
						if (os != null)
							os.write(("STOP PLAY MUSIC\n").getBytes("utf-8"));
					}
				} catch (Exception e) {
					e.printStackTrace();
				}
				break;
			default:
				break;
			}
		}
	};

	public void StartListenerSocket() throws IOException {
		ServerSocket ss = new ServerSocket(20000);
		System.out.println("服务器创建成功!");
		System.out.println("等待客戶端的连接。。。");
		while (true) {
			// 此行代码会阻塞,等待用户的连接
			Socket socket = ss.accept();
			System.out.println("有客户端连接进来!");
			Log.w("server log", "有客户端连接进来!");
			socketList.add(socket);
			// 每当客户端连接后启动一条ServerThread线程为该客户端服务
			new Thread(new ServerThread(socket)).start();
		}
	}

	
    @Override
	protected void onDestroy() {
		// TODO Auto-generated method stub
		super.onDestroy();
		closeAllSocket();
	}
    
    /**
     * 关闭服务器连接的所有Socket
     */
    private void closeAllSocket(){
    	try {
			for (Socket s : socketList) {
				OutputStream os = s.getOutputStream();
				if(os != null)
					os.write(("QUIT\n").getBytes("utf-8"));
				s.close();
				ServerMainActivity.socketList.remove(s);
			}
		} catch (Exception e) {
			e.printStackTrace();
		}
    }

	@Override
	public boolean onKeyDown(int keyCode, KeyEvent event) {
		// TODO Auto-generated method stub
		if (keyCode == KeyEvent.KEYCODE_BACK) {

			new AlertDialog.Builder(this)
					.setIcon(R.drawable.menu_exit)
					.setTitle("Notice")
					.setMessage("Do you want to quit?")
					.setNegativeButton("No",
							new DialogInterface.OnClickListener() {
								@Override
								public void onClick(DialogInterface dialog,
										int which) {
								}
							})
					.setPositiveButton("Yes",
							new DialogInterface.OnClickListener() {
								public void onClick(DialogInterface dialog,
										int whichButton) {
									closeAllSocket();
									
									ServerMainActivity.this.finish();
									Process.killProcess(Process.myPid());
								}
							}).show();
			return true;
		} else {
			return super.onKeyDown(keyCode, event);
		}
	}
}



客户端:

布局文件activity_main.xml


<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="#FFFFFF"
    android:orientation="vertical" >

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="20dp"
        android:orientation="horizontal" >

        <EditText
            android:id="@+id/edtServerIp"
            android:layout_width="240dp"
            android:layout_height="wrap_content"
            android:layout_gravity="center"
            android:hint="如:192.168.0.101"
            android:paddingLeft="10dp"
            />

        <Button
            android:id="@+id/btnSetAndEditIP"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:paddingLeft="10dp"
            android:paddingRight="10dp"
            android:text="设置\n服务器IP" />
    </LinearLayout>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="20dp"
        android:gravity="center"
        android:orientation="horizontal" >

        <Button
            android:id="@+id/btnConnectServer"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_gravity="center"
            android:layout_weight="3"
            android:enabled="false"
            android:paddingLeft="10dp"
            android:paddingRight="10dp"
            android:text="连接服务器" />

        <Button
            android:id="@+id/btnDisConnectServer"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_gravity="center"
            android:layout_weight="2"
            android:enabled="false"
            android:paddingLeft="10dp"
            android:paddingRight="10dp"
            android:text="断开服务器连接" />
    </LinearLayout>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="20dp"
        android:gravity="center"
        android:visibility="gone"
        android:orientation="horizontal" >

        <Button
            android:id="@+id/start"
            android:layout_width="0dp"
            android:layout_weight="2"
            android:layout_height="wrap_content"
            android:text="start" />

        <Button
            android:id="@+id/pause"
            android:layout_width="0dp"
            android:layout_weight="2"
            android:layout_height="wrap_content"
            android:text="pause" />

        <Button
            android:id="@+id/stop"
            android:layout_width="0dp"
            android:layout_weight="2"
            android:layout_height="wrap_content"
            android:text="stop" />
    </LinearLayout>

</LinearLayout>

ClientThread.java


package ict.ldj.client;

import java.io.BufferedReader; 
import java.io.IOException; 
import java.io.InputStreamReader; 
import java.net.Socket; 
 
import android.os.Handler; 
import android.os.Message; 
import android.util.Log;
 
public class ClientThread implements Runnable { 
    private Handler handler; 
    // 该线程所处理的Socket所对应的输入流  
    private BufferedReader br = null; 
 
    public ClientThread(Socket socket, Handler handler) throws IOException { 
        this.handler = handler; 
        br = new BufferedReader(new InputStreamReader(socket.getInputStream())); 
    }
 
    @Override 
    public void run() { 
        try { 
            String content = null; 
            // 不断读取Socket输入流的内容  
            while ((content = br.readLine()) != null) { 
                // 每当读到来自服务器的数据之后,发送消息通知程序界面显示该数据  
               
                if ("START PLAY MUSIC".equals(content)) {
                	handler.sendEmptyMessage(ClientMainActivity.socket_start_play);
				}else if ("PAUSE PLAY MUSIC".equals(content)) {
					handler.sendEmptyMessage(ClientMainActivity.socket_pause_play);
				}else if ("STOP PLAY MUSIC".equals(content)) {
					handler.sendEmptyMessage(ClientMainActivity.socket_stop_play);
				}else if ("QUIT".equals(content)) {
					handler.sendEmptyMessage(ClientMainActivity.socket_quit);
				}else{
					Message msg = new Message(); 
	                msg.what = ClientMainActivity.socket_msg; 
	                msg.obj = content; 
	                handler.sendMessage(msg);
				}
                Log.w("msg", content);
            } 
        } catch (Exception e) { 
            e.printStackTrace(); 
        } 
    } 
 
}

ClientMainActivity.java


package ict.ldj.client;

import java.io.IOException;
import java.io.OutputStream; 
import java.net.Socket; 
 
import android.app.Activity; 
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.content.res.AssetFileDescriptor;
import android.media.MediaPlayer;
import android.os.Bundle; 
import android.os.Handler; 
import android.os.Message;
import android.os.Process;
import android.view.KeyEvent;
import android.view.View; 
import android.view.View.OnClickListener; 
import android.widget.Button; 
import android.widget.EditText; 
import android.widget.Toast;
 
public class ClientMainActivity extends Activity { 
	public static final int socket_quit = 0;			// socket断开
	public static final int socket_start_play = 1;	// 播放音乐
	public static final int socket_pause_play = 2;	// 暂停播放
	public static final int socket_stop_play = 3;		// 停止播放
	public static final int socket_msg = 0x234;		// 用来聊天用的信息,该程序中未使用
	
    private EditText edtServerIp = null; 
    private Button btnSetAndEditIP = null; 
    private Button btnConnectServer = null; 
    private Button btnDisConnectServer = null; 
    
    private String serverIp = "";
    private OutputStream os = null; 
    private Handler handler = null; 
    private Socket socket = null;
    
    private MediaPlayer mp3 = null;
    private Boolean isPlaying = false; //设置标记,false表示正在播放
    
    @Override 
    protected void onCreate(Bundle savedInstanceState) { 
        super.onCreate(savedInstanceState); 
        setContentView(R.layout.activity_main); 
        
        initWidget();
    } 
    
    private void initWidget(){
    	edtServerIp = (EditText) findViewById(R.id.edtServerIp); 
        btnSetAndEditIP = (Button) findViewById(R.id.btnSetAndEditIP); 
        btnConnectServer = (Button) findViewById(R.id.btnConnectServer); 
        btnDisConnectServer = (Button) findViewById(R.id.btnDisConnectServer); 
        
        btnSetAndEditIP.setOnClickListener(myOnClickListener);
        btnConnectServer.setOnClickListener(myOnClickListener);
        btnDisConnectServer.setOnClickListener(myOnClickListener);
        findViewById(R.id.start).setOnClickListener(myOnClickListener);
        findViewById(R.id.stop).setOnClickListener(myOnClickListener);
        findViewById(R.id.pause).setOnClickListener(myOnClickListener);
        
        initMediaPlayer();
        
        handler = new Handler() {
            public void handleMessage(Message msg) { 
                // 如果消息来自子线程  
                if (msg.what == socket_msg) { 
                    // 将读取的内容追加显示在文本框中  
                	String mString = msg.obj.toString();
                }else if (msg.what == socket_start_play) {
                	showMessage("开始播放音乐");
                	// 开始播放音乐
                	startPlayMusic();
                }else if (msg.what == socket_pause_play) {
                	showMessage("暂停播放");
                	pausePlayMusic();
                }else if (msg.what == socket_stop_play) {
                	showMessage("停止播放");
                	stopPlayMusic();
                }else if(msg.what == socket_quit){
                	showAlertDialog("与服务器的连接已断开!!");
				}
            } 
        };
    }
    
    /**
     * 初始化播放器句柄
     */
    private void initMediaPlayer(){
    	mp3 = new MediaPlayer();
        // mp3 = MediaPlayer.create(ClientMainActivity.this,R.raw.music);
         try {
 			AssetFileDescriptor afd = this.getAssets().openFd("music.mp3");
 			mp3.setDataSource(afd.getFileDescriptor(),afd.getStartOffset(),afd.getLength());
 			if (mp3 != null) {
 				mp3.prepare();
 			}
         } catch (IOException e) {
 			// TODO Auto-generated catch block
 			e.printStackTrace();
 		}
    }
    
    /**
     * 开始播放
     */
    private void startPlayMusic(){
    	try {
			if (mp3 != null) {
				mp3.stop();
			}
			mp3.prepare();
			mp3.start();
			isPlaying = true;
		} catch (Exception e) {
			// TODO: handle exception
			e.printStackTrace();
		}
    }
    
    /**
     * 停止播放
     */
    private void stopPlayMusic(){
    	try {
			if (mp3 != null) {
				mp3.stop();
				mp3.reset();
				initMediaPlayer();
			}
		} catch (Exception e) {
			// TODO: handle exception
			e.printStackTrace();
		}
    }
    
    /**
     * 暂停播放
     */
    private void pausePlayMusic(){
    	try {
			if (isPlaying) {
				mp3.pause();
				isPlaying = false;
			}else {
				mp3.start();
				isPlaying = true;
			}
		} catch (Exception e) {
			// TODO: handle exception
			e.printStackTrace();
		}
    }
    
    private OnClickListener myOnClickListener = new OnClickListener() {
		@Override
		public void onClick(View v) {
			// TODO Auto-generated method stub
			switch (v.getId()) {
			case R.id.btnSetAndEditIP:
				if (btnSetAndEditIP.getText().toString().startsWith("设置")) {
					serverIp = edtServerIp.getText().toString();
					edtServerIp.setEnabled(false);
					btnConnectServer.setEnabled(true);
					btnSetAndEditIP.setText("修改\n服务器IP");
				}else {		// 修改\n服务器IP
					edtServerIp.setEnabled(true);
					btnConnectServer.setEnabled(false);
					btnSetAndEditIP.setText("设置\n服务器IP");
				}
				break;
			case R.id.btnConnectServer:	
				// 连接服务器
				if (serverIp.length() <= 7) {
					showMessage("服务器 IP 地址输入不合法");
				}else {
					if (socket != null) {
						showMessage("服务器已连接!!");
					}else{
				        try { 
				            socket = new Socket(serverIp, 20000); 
				            // 客户端启动ClientThread线程不断读取来自服务器的数据  
				            new Thread(new ClientThread(socket, handler)).start(); 
				            os = socket.getOutputStream();
				            showMessage("服务器连接成功!!");
				            btnDisConnectServer.setEnabled(true);
				        } catch (Exception e) { 
				        	showAlertDialog("服务器连接失败!!");
				            e.printStackTrace(); 
				        }
					}
				}
				break;
			case R.id.btnDisConnectServer:
				if (socket != null) {
					if(os != null){
						try {
							os.write(("QUIT\n").getBytes());
							socket.close();
							btnDisConnectServer.setEnabled(false);
							socket = null;
						} catch (IOException e) {
							// TODO Auto-generated catch block
							e.printStackTrace();
						}
					}
				}
				break;
			case R.id.start:
				startPlayMusic();
				break;
			case R.id.stop:
				stopPlayMusic();
				break;
			case R.id.pause:
				pausePlayMusic();
				break;
			default:
				break;
			}
		}
	};
    
	/**
	 * 弹出信息显示框,如果点击Yes按钮,则向服务器端发送一个QUIT标志,向服务器提示Socket要释放
	 * @param str 弹出框中输出的信息
	 */
    public void showAlertDialog(String str){
    	new AlertDialog.Builder(this)
			.setIcon(R.drawable.menu_exit)
			.setTitle("Notice")
			.setMessage(str)
			.setPositiveButton("Yes",
				new DialogInterface.OnClickListener() {
					public void onClick(DialogInterface dialog,
							int whichButton) {
						try {
							if(os != null)
								os.write(("QUIT\n").getBytes());
						} catch (IOException e) {
							// TODO Auto-generated catch block
							e.printStackTrace();
						} 
					}
				})
			.show();
    }
    
    @Override
	protected void onDestroy() {
		// TODO Auto-generated method stub
    	super.onDestroy();
    	try {
			if(os != null)
				os.write(("QUIT\n").getBytes());
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} 
    	
    	if (mp3 != null) {
			mp3.stop();
			mp3.release();
			mp3 = null;
		}
	}

	@Override
	public boolean onKeyDown(int keyCode, KeyEvent event) {
		// TODO Auto-generated method stub
		if (keyCode == KeyEvent.KEYCODE_BACK) {
			new AlertDialog.Builder(this)
				.setIcon(R.drawable.menu_exit)
				.setTitle("Notice")
				.setMessage("Do you want to quit?")
				.setNegativeButton("No",
					new DialogInterface.OnClickListener() {
						@Override
						public void onClick(DialogInterface dialog,
								int which) {
						}
					})
				.setPositiveButton("Yes",
					new DialogInterface.OnClickListener() {
						public void onClick(DialogInterface dialog,
								int whichButton) {
							try {
								if(os != null)
									os.write(("QUIT\n").getBytes());
							} catch (IOException e) {
								// TODO Auto-generated catch block
								e.printStackTrace();
							} 
							//关闭程序
							ClientMainActivity.this.finish();
							Process.killProcess(Process.myPid());
						}
					})
				.show();
			return true;
		} else {
			return super.onKeyDown(keyCode, event);
		}
	}
	
	private void showMessage(String str){
		Toast.makeText(getApplicationContext(), str, Toast.LENGTH_SHORT).show();
	}
}



大家求婚如有需要,可以尽情使用,O(∩_∩)O哈哈~

 源码下载请点击。