java實現(xiàn)屏幕共享功能實例分析
本文實例講述了java實現(xiàn)屏幕共享功能的方法。分享給大家供大家參考。具體分析如下:
最近在做軟件軟件工程的課程設(shè)計,做一個用于實驗室的屏幕監(jiān)控系統(tǒng),參考各種前人代碼,最后領(lǐng)悟之后要轉(zhuǎn)換自己的代碼,初學(xué)者都是這樣模仿過來的。
說到屏幕監(jiān)控系統(tǒng),有教師端和學(xué)生端,教師端就是Server端,學(xué)生端就做Client端。系統(tǒng)里比較有趣的一個地方應(yīng)該算是屏幕廣播與屏幕監(jiān)控吧,其余什么點名簽到,鎖屏,定時關(guān)機(jī)的,就相對來說簡單點。
屏幕廣播,在功能實現(xiàn)上面,說白了,就是教師端的機(jī)器不斷截取屏幕信息,以圖片的形式發(fā)送到每一個學(xué)生端的電腦上面,由此學(xué)生能夠看見老師在電腦上的操作,這就是所謂的屏幕廣播。
這里面有個麻煩的地方,就是截取屏幕圖片的時候,是沒有鼠標(biāo)信息。不過有兩種解決辦法:
①在發(fā)送截圖信息時,在圖片上繪制一個鼠標(biāo),這樣在學(xué)生端就會有兩個鼠標(biāo),學(xué)生端可以移動自己電腦上的鼠標(biāo)。
②發(fā)送教師端的鼠標(biāo)坐標(biāo)到學(xué)生端上,學(xué)生端的電腦鼠標(biāo)根據(jù)坐標(biāo)信息實時移動,這里其實是涉及到控制的功能了,學(xué)生端就不能移動鼠標(biāo)了。
屏幕監(jiān)控相對棘手點,其實這是這包含倆功能:
①教師監(jiān)控所有學(xué)生電腦屏幕的功能;
②教師控制某一個學(xué)生的電腦;
因為涉及到并發(fā),每個client都要實時的把屏幕信息發(fā)到教師端上,會有點麻煩,不過還是可以實現(xiàn)。
這里暫時實現(xiàn)了不帶鼠標(biāo)的屏幕共享功能,比較簡單,有待完善,不過可以作為一個工具類在后面集成使用。
首先是教師端Server:
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;
/*
* ly 2014-11-20
* 該類實時發(fā)送截屏消失,多線程實現(xiàn),不包含鼠標(biāo)信息,且沒有做對每個Client做優(yōu)化處理
*/
public class SendScreenImg extends Thread
{
public static int SERVERPORT=8000;
private ServerSocket serverSocket;
private Robot robot;
public Dimension screen;
public Rectangle rect ;
private Socket socket;
public static void main(String args[])
{
new SendScreenImg(SERVERPORT).start();
}
//構(gòu)造方法 開啟套接字連接 機(jī)器人robot 獲取屏幕大小
public SendScreenImg(int SERVERPORT)
{
try {
serverSocket = new ServerSocket(SERVERPORT);
serverSocket.setSoTimeout(864000000);
robot = new Robot();
} catch (Exception e) {
e.printStackTrace();
}
screen = Toolkit.getDefaultToolkit().getScreenSize(); //獲取主屏幕的大小
rect = new Rectangle(screen); //構(gòu)造屏幕大小的矩形
}
@Override
public void run()
{
//實時等待接收截屏消息
while(true)
{
try{
socket = serverSocket.accept();
System.out.println("學(xué)生端口已經(jīng)連接");
ZipOutputStream zip = new ZipOutputStream(new DataOutputStream(socket.getOutputStream()));
zip.setLevel(9); //設(shè)置壓縮級別
BufferedImage img = robot.createScreenCapture(rect);
zip.putNextEntry(new ZipEntry("test.jpg"));
ImageIO.write(img, "jpg", zip);
if(zip!=null)zip.close();
System.out.println("Client正在實時連接");
} catch (IOException ioe) {
System.out.println("連接斷開");
} finally {
if (socket != null) {
try {
socket.close();
} catch (IOException e) {e.printStackTrace();}
}
}
}
}
}
然后是學(xué)生端Client:
import java.awt.Frame;
import java.awt.Image;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.io.DataInputStream;
import java.io.IOException;
import java.net.Socket;
import java.util.concurrent.TimeUnit;
import java.util.zip.ZipInputStream;
import javax.imageio.ImageIO;
import javax.swing.ImageIcon;
import javax.swing.JFrame;
import javax.swing.JLabel;
/*
* ly 2014-11-20
* 該類用于接收教師端的屏幕信息,不包括鼠標(biāo),待優(yōu)化
*/
public class ReceiveImages extends Thread{
public BorderInit frame ;
public Socket socket;
public String IP;
public static void main(String[] args){
new ReceiveImages(new BorderInit(), "127.0.0.1").start();
}
public ReceiveImages(BorderInit frame,String IP)
{
this.frame = frame;
this.IP=IP;
}
public void run() {
while(frame.getFlag()){
try {
socket = new Socket(IP,8000);
DataInputStream ImgInput = new DataInputStream(socket.getInputStream());
ZipInputStream imgZip = new ZipInputStream(ImgInput);
imgZip.getNextEntry(); //到Zip文件流的開始處
Image img = ImageIO.read(imgZip); //按照字節(jié)讀取Zip圖片流里面的圖片
frame.jlbImg.setIcon(new ImageIcon(img));
System.out.println("連接第"+(System.currentTimeMillis()/1000)%24%60+"秒");
frame.validate();
TimeUnit.MILLISECONDS.sleep(50);// 接收圖片間隔時間
imgZip.close();
} catch (IOException | InterruptedException e) {
System.out.println("連接斷開");
}finally{
try {
socket.close();
} catch (IOException e) {}
}
}
}
}
//Client端窗口輔助類,專門用來顯示從教師端收到的屏幕信息
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("遠(yuǎn)程監(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();
//窗口關(guān)閉事件
this.addWindowListener(new WindowAdapter() {
public void windowClosing(WindowEvent e) {
flag=false;
BorderInit.this.dispose();
System.out.println("窗體關(guān)閉");
System.gc(); //垃圾回收
}
});
}
}
這里就從未成品中抽取了這么個小功能,距離成品還有很多要寫,感興趣的朋友可以在此基礎(chǔ)上加以完善。
希望本文所述對大家的Java程序設(shè)計有所幫助。
- Java 存儲模型和共享對象詳解
- Java讀寫Windows共享文件夾的方法實例
- Java線程重復(fù)執(zhí)行以及操作共享變量的代碼示例
- Java中tomcat memecached session 共享同步問題的解決辦法
- Java多線程編程之ThreadLocal線程范圍內(nèi)的共享變量
- Java多線程編程之訪問共享對象和數(shù)據(jù)的方法
- Java設(shè)計模式之共享模式/享元模式(Flyweight模式)介紹
- java多線程并發(fā)中使用Lockers類將多線程共享資源鎖定
- java通過共享變量結(jié)束run停止線程的方法示例
- Java使用wait() notify()方法操作共享資源詳解
相關(guān)文章
spring帶bean和config如何通過main啟動測試
這篇文章主要介紹了spring帶bean和config,通過main啟動測試,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-07-07Jvisualvm監(jiān)控遠(yuǎn)程SpringBoot項目的過程詳解
這篇文章主要介紹了Jvisualvm監(jiān)控遠(yuǎn)程SpringBoot項目,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-04-04SpringBoot使用Prometheus實現(xiàn)監(jiān)控
在當(dāng)今的軟件開發(fā)世界中,監(jiān)控是至關(guān)重要的一部分,本文主要介紹了如何在Spring Boot應(yīng)用程序中使用Prometheus進(jìn)行監(jiān)控,以幫助大家更好地理解和管理您的應(yīng)用程序,有需要的可以參考下2023-10-10淺談SpringMVC HandlerInterceptor詭異問題排查
這篇文章主要介紹了淺談SpringMVC HandlerInterceptor詭異問題排查,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-05-05