欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java通過控制鼠標實現(xiàn)屏幕廣播的方法

 更新時間:2014年12月15日 11:25:39   投稿:shichen2014  
這篇文章主要介紹了java通過控制鼠標實現(xiàn)屏幕廣播的方法,針對前面一篇Java屏幕共享功能進行了改進,實現(xiàn)了鼠標控制功能,具有一定的實用價值,需要的朋友可以參考下

本文實例講述了java通過控制鼠標實現(xiàn)屏幕廣播的方法。分享給大家供大家參考。具體分析如下:

在前面一篇《java實現(xiàn)屏幕共享功能實例分析》中提到截取屏幕時是沒鼠標,為了看到教師端界面上的鼠標,可以在截取屏幕的時候,把鼠標繪制到每一張截圖上去,但是由于截圖的時候是一張張截取的,所以看到的鼠標難免會有點卡,之前寫了java鼠標操控小程序,可以通過這種方式來看到鼠標的演示。

實現(xiàn)的方式也挺簡單的,前面兩篇文章分別實現(xiàn)了鼠標控制和不帶鼠標的屏幕分享功能,把這兩個結合一下就ok了,下面簡單分析下。

服務端,將SendScreenImg和SendMouseMessage看作兩個工具類,分別監(jiān)聽不同的端口,他們兩個都實現(xiàn)了Thread類,我們用線程池ExecutorService類控制他們。

使用了兩個端口,因為暫時還不知道該如何吧鼠標信息和圖片的信息一起發(fā)送,或許可以把圖片轉換成字節(jié)數(shù)組的形式,把鼠標的坐標放在數(shù)組前面,不過這樣的話鼠標可能也會不連貫,因為傳送鼠標坐標的速度會比傳圖片的要快一些,嗯,有空再試試。

客戶端類比上面就是了。

下面是代碼:

服務端:

主程序:

復制代碼 代碼如下:
/*
 * 屏幕廣播類,調用了兩個工具類:發(fā)送截屏信息的類和發(fā)送鼠標的信息類,利用了線程池。
 */
public class BroderCast {
    public  static void main(String[] args)
    {
        new BroderCast();
        System.out.println("開始");
    }
    public BroderCast()
    {
        ExecutorService exector = Executors.newFixedThreadPool(2);
        exector.execute(new SendScreenImg());
        exector.execute(new SendMouseMessage());
    }
}

發(fā)送截圖代碼:
復制代碼 代碼如下:
import java.awt.Dimension;
import java.awt.Rectangle;
import java.awt.Robot;
import java.awt.Toolkit;
import java.awt.image.BufferedImage;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream;
import javax.imageio.ImageIO;
 
/*
 * 工具類:發(fā)送教師端截屏信息給學生端,沒有鼠標信息,使用了8002號端口
 *      可以在發(fā)送的圖片上面組件繪制鼠標的信息,從而實現(xiàn)在學生端界面上見到鼠標信息,暫未實現(xiàn)該功能
 *             
 */
public  class SendScreenImg extends Thread
{
    public int serverPort=8002;
    private ServerSocket serverSocket;
    private Robot robot;
    public  Dimension screen;
    public Rectangle rect ;
    private Socket socket;
    public static void main(String args[])
    {
        new SendScreenImg().start();
    }
    
    public void changeServerPort(int serverPort)
    {
        if(this.serverPort == serverPort) return ;
        try{
            this.serverSocket.close();    //有必要先關閉當前端口
            this.serverPort = serverPort;
            serverSocket = new ServerSocket(this.serverPort);
            serverSocket.setSoTimeout(8000000);
        }catch(Exception e){}
        
    }
    
    //構造方法  開啟套接字連接      機器人robot   獲取屏幕大小
    public SendScreenImg()
    {
        try {
            serverSocket = new ServerSocket(getServerPort());
            serverSocket.setSoTimeout(864000000);
            robot = new Robot();
        } catch (Exception e) {
            e.printStackTrace();
        }
        screen = Toolkit.getDefaultToolkit().getScreenSize();  //獲取主屏幕的大小
        rect = new Rectangle(screen);                          //構造相應大小的矩形
        
    }
    @Override
    public void run()
    {
        //實時等待接收截屏消息
        while(true){
            try {
                socket = serverSocket.accept();
                ZipOutputStream zip = new ZipOutputStream(new DataOutputStream(socket.getOutputStream()));
                zip.setLevel(9);     //設置壓縮級別
                try{
                    BufferedImage img = robot.createScreenCapture(rect);
                    zip.putNextEntry(new ZipEntry("test.jpg"));
                    ImageIO.write(img, "jpg", zip);
                    if(zip!=null)zip.close();
                    System.out.println("學生端口已經(jīng)連接");
                } catch (IOException ioe) {
                    System.out.println("被控端:disconnect");
                }
            } catch (IOException ioe) {
             System.out.println("連接出錯");
            } finally {
                if (socket != null) {
                    try {
                        socket.close();
                    } catch (IOException e) {
                    }
                }
            }
        }
    }
}

發(fā)送鼠標信息:
復制代碼 代碼如下:
/*
 * 工具類:獲取鼠標的信息并且發(fā)送給學生端
 */
public class SendMouseMessage extends Thread{
    private int OPERATE_PORT = 8001;
    private ServerSocket server;
    private Socket socket;
    private String operateStr;
    public static void main(String[] args)
    {
        new SendMouseMessage().start();
    }
    public SendMouseMessage(){
        try {
            server = new ServerSocket(OPERATE_PORT);
            //JOptionPane.showMessageDialog(null, "已經(jīng)開始監(jiān)聽");
        } catch (IOException e1) {
            e1.printStackTrace();
        }      
    }
    //多線程  在無線的循環(huán)中監(jiān)聽客戶端的
    public void run()
    {
        while(true){
            Point point = MouseInfo.getPointerInfo().getLocation();  //
            operateStr ="Movemouse,"+point.x+","+point.y;
            try {
                socket = server.accept();
                socket.setSoTimeout(1000000);
                DataOutputStream output =new DataOutputStream(socket.getOutputStream());
                output.write(operateStr.getBytes());
                output.flush();   //刷行輸出流,并且使所有緩沖的輸出字節(jié)寫出
                output.close();   //關閉輸出流且釋放資源
    
                System.out.println("INFO:  "+operateStr);
            } catch (IOException e) {
                System.out.println("已經(jīng)停止連接");
                break;   //斷開連接的時候就停止無線循環(huán)
            }
        }
    }
}

客戶端:

主程序:

復制代碼 代碼如下:
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
 
import com.Tool.OperateMouse;
import com.Tool.ReceiveImages;
 
public class ReceiveBroderCast {
    public ExecutorService exector;
    public static String IP="202.216.60.9";
    
    public static void main(String[] args)
    {
        new ReceiveBroderCast(IP);
    }
    
    public ReceiveBroderCast(String IP) {
        exector = Executors.newFixedThreadPool(2);
        exector.execute(new ReceiveImages(IP));
        exector.execute(new OperateMouse(IP));
    }
}

接收截圖代碼:
復制代碼 代碼如下:
/*
 * ly  2014-11-20
 * 該類用于接收教師端的屏幕信息,不包括鼠標
 * 使用socket()
 */
public  class ReceiveImages extends  Thread{
    public BorderInit frame ;
    public Socket socket;
    public String IP;
    
    public static void main(String[] args){
        new ReceiveImages("202.216.60.7").start();
    }
    public ReceiveImages(String IP)
    {
        frame=new BorderInit();
        this.IP=IP;
        
    }
    public void run() {
        while(frame.getFlag()){
            System.out.println("已經(jīng)連接"+(System.currentTimeMillis()/1000)%24%60+"秒");
            try {
                socket = new Socket(IP,8002);
                DataInputStream ImgInput = new DataInputStream(socket.getInputStream());
                ZipInputStream imgZip = new ZipInputStream(ImgInput);
                Image img = null;
                try{
                    imgZip.getNextEntry();   //到Zip文件流的開始處
                    img = ImageIO.read(imgZip); //按照字節(jié)讀取Zip圖片流里面的圖片
                    frame.jlbImg.setIcon(new ImageIcon(img));
                    frame.validate();
                }catch (IOException e) {e.printStackTrace();}
                
                try{
                    imgZip.close();
                } catch (IOException e) {
                    System.out.println("連接斷開");
                }
                try {
                    TimeUnit.MILLISECONDS.sleep(50);// 接收圖片間隔時間
                } catch (InterruptedException ie) {
                    ie.printStackTrace();
                }      
            } catch (IOException e) {
                e.printStackTrace();
            }finally{
                try {
                    socket.close();
                } catch (IOException e) {} 
            }      
        }  
    }
}
class BorderInit extends JFrame
{
    private static final long serialVersionUID = 1L;
    public JLabel jlbImg;
    private boolean flag;
    public boolean getFlag(){
        return this.flag;
    }
    public BorderInit()
    {
        this.flag=true;
        this.jlbImg = new JLabel();
        this.setTitle("遠程監(jiān)控--IP:"  + "--主題:" );
        this.setSize(400, 400);
        //this.setUndecorated(true);
        //this.setAlwaysOnTop(true);  //始終在最前面
        this.add(jlbImg);
        this.setLocationRelativeTo(null);
        this.setExtendedState(Frame.MAXIMIZED_BOTH);
        this.setDefaultCloseOperation(DISPOSE_ON_CLOSE);
        this.setVisible(true);
        this.validate();
    
        //窗口關閉事件
        this.addWindowListener(new WindowAdapter() {
            public void windowClosing(WindowEvent e) {
                flag=false;
                BorderInit.this.dispose();
                System.out.println("窗體關閉");
                System.gc();    //垃圾回收
            }
        });
    }
}

接收鼠標信息并控制鼠標移動:
復制代碼 代碼如下:
import java.awt.AWTException;
import java.awt.Robot;
import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.io.DataInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.Socket;
 
import javax.swing.JOptionPane;
 
/*
 * 學生端   控制鼠標和教師端一致
 * 該類  負責接收鼠標的信息  并且用robot.mouseMove()函數(shù)控制鼠標移動 
 */
public class OperateMouse extends Thread{
    public static void main(String[] args)
    {
        new OperateMouse("202.116.60.7").start();
    }
    private Socket socket;
    public String IP;
    private int OPERATE_PORT = 8001;
    private Robot robot;
    
    public OperateMouse(String IP)
    {
        this.IP = IP;
    }
    public void run() {
        while(true){
            try {
                socket = new Socket(IP,OPERATE_PORT);
                robot = new Robot();
                //獲取鼠標移動的信息
                DataInputStream dataIn = new DataInputStream(socket.getInputStream());       
                String info="";
                int r;
                while((r=dataIn.read()) != -1){
                    info +=""+(char)r;   //把字節(jié)數(shù)組中所有元素都變?yōu)樽址?br />                 }
                dataIn.close();
                System.out.println("數(shù)據(jù)流斷開"+info);
                if(info!=null){
                        String s[] = info.trim().split(",");
                        if("Movemouse".equals(s[0].trim()));
                        {
                            if (s.length == 3) {
                                int x = Integer.parseInt(s[1].trim());
                                int y = Integer.parseInt(s[2].trim());
                                System.out.println("輸出鼠標的信息"+x+"  "+ y);
                                robot.mouseMove(x, y);
                            }
                        }
                    }
            } catch (IOException e) {
                System.out.println("已斷開連接");
                break;
            } catch (AWTException e) {
                e.printStackTrace();
            }
        }
    }
}

希望本文所述對大家的Java程序設計有所幫助。

相關文章

  • java打印指定年月的日歷

    java打印指定年月的日歷

    這篇文章主要為大家詳細介紹了Java如何打印指定年月的日歷,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-07-07
  • Spring?MVC基于注解的使用之JSON數(shù)據(jù)處理的方法

    Spring?MVC基于注解的使用之JSON數(shù)據(jù)處理的方法

    這篇文章主要介紹了Spring?MVC基于注解的使用JSON數(shù)據(jù)處理,json是一種輕量級的數(shù)據(jù)交換格式,是一種理想的數(shù)據(jù)交互語言,它易于閱讀和編寫,同時也易于機器解析和生成,本文通過實例代碼給大家介紹的非常詳細,需要的朋友可以參考下
    2022-05-05
  • IDEA Error:java: 無效的源發(fā)行版: 17錯誤

    IDEA Error:java: 無效的源發(fā)行版: 17錯誤

    本文主要介紹了IDEA Error:java: 無效的源發(fā)行版: 17錯誤,這個錯誤是因為您的IDEA編譯器不支持Java 17版本,您需要更新您的IDEA編譯器或者將您的Java版本降級到IDEA支持的版本,本文就來詳細的介紹一下
    2023-08-08
  • Java?Lambda表達式常用的函數(shù)式接口

    Java?Lambda表達式常用的函數(shù)式接口

    這篇文章主要介紹了Java?Lambda表達式常用的函數(shù)式接口,文章基于Java?Lambda表達式展開對常用的函數(shù)式接口的介紹,具有一的的參考價值需要的小伙伴可以參考一下
    2022-04-04
  • 帶你快速搞定java數(shù)組

    帶你快速搞定java數(shù)組

    數(shù)組是相同類型數(shù)據(jù)的有序集合數(shù)組描述的是相同類型的若干個數(shù)據(jù),按照一定的先后次序排列組合而成。其中,每一個數(shù)據(jù)稱作一個數(shù)組元素,每個數(shù)組元素可以通過一個下標來訪問它們數(shù)組的聲明創(chuàng)建
    2021-07-07
  • Java正則表達式_動力節(jié)點Java學院整理

    Java正則表達式_動力節(jié)點Java學院整理

    什么是正則表達式,正則表達式的作用是什么?這篇文章主要為大家詳細介紹了Java正則表達式的相關資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • Java實現(xiàn)常見的排序算法代碼實例

    Java實現(xiàn)常見的排序算法代碼實例

    這篇文章主要介紹了Java實現(xiàn)常見的排序算法代碼實例,按照思路實現(xiàn)了以下幾個排序算法(冒泡排序、直接插入排序、直接選擇排序、快速排序),方便日后用到,特此記錄一下,需要的朋友可以參考下
    2023-11-11
  • java 實現(xiàn)單鏈表逆轉詳解及實例代碼

    java 實現(xiàn)單鏈表逆轉詳解及實例代碼

    這篇文章主要介紹了java 實現(xiàn)單鏈表逆轉實例代碼的相關資料,需要的朋友可以參考下
    2017-02-02
  • Java檢測線程中斷狀態(tài)的方法示例

    Java檢測線程中斷狀態(tài)的方法示例

    這篇文章主要介紹了Java檢測線程中斷狀態(tài)的方法,結合實例形式分析了java針對線程中斷狀態(tài)檢測的相關實現(xiàn)技巧,需要的朋友可以參考下
    2019-10-10
  • spring boot定時器實現(xiàn)定時同步數(shù)據(jù)的操作步驟

    spring boot定時器實現(xiàn)定時同步數(shù)據(jù)的操作步驟

    在Java中,@Scheduled注解是用于指定定時任務的執(zhí)行規(guī)則的,這篇文章給大家介紹spring boot定時器實現(xiàn)定時同步數(shù)據(jù)的操作步驟,感興趣的朋友一起看看吧
    2023-12-12

最新評論