java实现屏幕共享功能实例分析
本文实例讲述了java实现屏幕共享功能的方法。分享给大家供大家参考。具体分析如下:
最近在做软件软件工程的课程设计,做一个用于实验室的屏幕监控系统,参考各种前人代码,最后领悟之后要转换自己的代码,初学者都是这样模仿过来的。
说到屏幕监控系统,有教师端和学生端,教师端就是Server端,学生端就做Client端。系统里比较有趣的一个地方应该算是屏幕广播与屏幕监控吧,其余什么点名签到,锁屏,定时关机的,就相对来说简单点。
屏幕广播,在功能实现上面,说白了,就是教师端的机器不断截取屏幕信息,以图片的形式发送到每一个学生端的电脑上面,由此学生能够看见老师在电脑上的操作,这就是所谓的屏幕广播。
这里面有个麻烦的地方,就是截取屏幕图片的时候,是没有鼠标信息。不过有两种解决办法:
①在发送截图信息时,在图片上绘制一个鼠标,这样在学生端就会有两个鼠标,学生端可以移动自己电脑上的鼠标。
②发送教师端的鼠标坐标到学生端上,学生端的电脑鼠标根据坐标信息实时移动,这里其实是涉及到控制的功能了,学生端就不能移动鼠标了。
屏幕监控相对棘手点,其实这是这包含俩功能:
①教师监控所有学生电脑屏幕的功能;
②教师控制某一个学生的电脑;
因为涉及到并发,每个client都要实时的把屏幕信息发到教师端上,会有点麻烦,不过还是可以实现。
这里暂时实现了不带鼠标的屏幕共享功能,比较简单,有待完善,不过可以作为一个工具类在后面集成使用。
首先是教师端Server:
packageTest;
importjava.awt.Dimension;
importjava.awt.Rectangle;
importjava.awt.Robot;
importjava.awt.Toolkit;
importjava.awt.image.BufferedImage;
importjava.io.DataOutputStream;
importjava.io.IOException;
importjava.net.ServerSocket;
importjava.net.Socket;
importjava.util.zip.ZipEntry;
importjava.util.zip.ZipOutputStream;
importjavax.imageio.ImageIO;
/*
*ly 2014-11-20
*该类实时发送截屏消失,多线程实现,不包含鼠标信息,且没有做对每个Client做优化处理
*/
public classSendScreenImgextendsThread
{
publicstaticintSERVERPORT=8000;
privateServerSocketserverSocket;
privateRobotrobot;
public Dimensionscreen;
publicRectanglerect;
privateSocketsocket;
publicstaticvoidmain(Stringargs[])
{
newSendScreenImg(SERVERPORT).start();
}
//构造方法 开启套接字连接 机器人robot 获取屏幕大小
publicSendScreenImg(intSERVERPORT)
{
try{
serverSocket=newServerSocket(SERVERPORT);
serverSocket.setSoTimeout(864000000);
robot=newRobot();
}catch(Exceptione){
e.printStackTrace();
}
screen=Toolkit.getDefaultToolkit().getScreenSize(); //获取主屏幕的大小
rect=newRectangle(screen); //构造屏幕大小的矩形
}
@Override
publicvoidrun()
{
//实时等待接收截屏消息
while(true)
{
try{
socket=serverSocket.accept();
System.out.println("学生端口已经连接");
ZipOutputStreamzip=newZipOutputStream(newDataOutputStream(socket.getOutputStream()));
zip.setLevel(9); //设置压缩级别
BufferedImageimg=robot.createScreenCapture(rect);
zip.putNextEntry(newZipEntry("test.jpg"));
ImageIO.write(img,"jpg",zip);
if(zip!=null)zip.close();
System.out.println("Client正在实时连接");
}catch(IOExceptionioe){
System.out.println("连接断开");
}finally{
if(socket!=null){
try{
socket.close();
}catch(IOExceptione){e.printStackTrace();}
}
}
}
}
}
然后是学生端Client:
packageTest;
importjava.awt.Frame;
importjava.awt.Image;
importjava.awt.event.WindowAdapter;
importjava.awt.event.WindowEvent;
importjava.io.DataInputStream;
importjava.io.IOException;
importjava.net.Socket;
importjava.util.concurrent.TimeUnit;
importjava.util.zip.ZipInputStream;
importjavax.imageio.ImageIO;
importjavax.swing.ImageIcon;
importjavax.swing.JFrame;
importjavax.swing.JLabel;
/*
*ly 2014-11-20
*该类用于接收教师端的屏幕信息,不包括鼠标,待优化
*/
public classReceiveImagesextends Thread{
publicBorderInitframe;
publicSocketsocket;
publicStringIP;
publicstaticvoidmain(String[]args){
newReceiveImages(newBorderInit(),"127.0.0.1").start();
}
publicReceiveImages(BorderInitframe,StringIP)
{
this.frame=frame;
this.IP=IP;
}
publicvoidrun(){
while(frame.getFlag()){
try{
socket=newSocket(IP,8000);
DataInputStreamImgInput=newDataInputStream(socket.getInputStream());
ZipInputStreamimgZip=newZipInputStream(ImgInput);
imgZip.getNextEntry(); //到Zip文件流的开始处
Imageimg=ImageIO.read(imgZip); //按照字节读取Zip图片流里面的图片
frame.jlbImg.setIcon(newImageIcon(img));
System.out.println("连接第"+(System.currentTimeMillis()/1000)%24%60+"秒");
frame.validate();
TimeUnit.MILLISECONDS.sleep(50);//接收图片间隔时间
imgZip.close();
}catch(IOException|InterruptedExceptione){
System.out.println("连接断开");
}finally{
try{
socket.close();
}catch(IOExceptione){}
}
}
}
}
//Client端窗口辅助类,专门用来显示从教师端收到的屏幕信息
classBorderInitextendsJFrame
{
privatestaticfinallongserialVersionUID=1L;
publicJLabeljlbImg;
privatebooleanflag;
publicbooleangetFlag(){
returnthis.flag;
}
publicBorderInit()
{
this.flag=true;
this.jlbImg=newJLabel();
this.setTitle("远程监控--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(newWindowAdapter(){
publicvoidwindowClosing(WindowEvente){
flag=false;
BorderInit.this.dispose();
System.out.println("窗体关闭");
System.gc(); //垃圾回收
}
});
}
}
这里就从未成品中抽取了这么个小功能,距离成品还有很多要写,感兴趣的朋友可以在此基础上加以完善。
希望本文所述对大家的Java程序设计有所帮助。