Android静态图片人脸识别的完整demo(附完整源码)

 

Demo功能:利用android自带的人脸识别进行识别,标记出眼睛和人脸位置。点击按键后进行人脸识别,完毕后显示到imageview上。

第一部分:布局文件activity_main.xml

 

[html] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:id="@+id/layout_main"  
  4.     android:layout_width="match_parent"  
  5.     android:layout_height="match_parent"  
  6.     android:paddingBottom="@dimen/activity_vertical_margin"  
  7.     android:paddingLeft="@dimen/activity_horizontal_margin"  
  8.     android:paddingRight="@dimen/activity_horizontal_margin"  
  9.     android:paddingTop="@dimen/activity_vertical_margin"  
  10.     tools:context=".MainActivity" >  
  11.   
  12.     <TextView  
  13.         android:id="@+id/textview_hello"  
  14.         android:layout_width="wrap_content"  
  15.         android:layout_height="wrap_content"  
  16.         android:text="@string/hello_world" />  
  17.   
  18.     <ImageView  
  19.         android:id="@+id/imgview"  
  20.         android:layout_width="wrap_content"  
  21.         android:layout_height="wrap_content"  
  22.         android:layout_below="@id/textview_hello" />  
  23.   
  24.     <Button  
  25.         android:id="@+id/btn_detect_face"  
  26.         android:layout_width="wrap_content"  
  27.         android:layout_height="wrap_content"  
  28.         android:layout_below="@id/imgview"  
  29.         android:layout_centerHorizontal="true"  
  30.         android:text="检测人脸" />  
  31.   
  32. </RelativeLayout>  


注意:ImageView四周的padding由布局文件里的这四句话决定:

 

[html] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. android:paddingBottom="@dimen/activity_vertical_margin"  
  2. android:paddingLeft="@dimen/activity_horizontal_margin"  
  3. android:paddingRight="@dimen/activity_horizontal_margin"  
  4. android:paddingTop="@dimen/activity_vertical_margin"  


而上面的两个margin定义在dimens.xml文件里:

 

[html] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. <resources>  
  2.   
  3.     <!-- Default screen margins, per the Android Design guidelines. -->  
  4.     <dimen name="activity_horizontal_margin">16dp</dimen>  
  5.     <dimen name="activity_vertical_margin">16dp</dimen>  
  6.   
  7. </resources>  


这里采用的都是默认的,可以忽略!

第二部分:MainActivity.java

 

[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. package org.yanzi.testfacedetect;  
  2.   
  3. import org.yanzi.util.ImageUtil;  
  4. import org.yanzi.util.MyToast;  
  5.   
  6. import android.app.Activity;  
  7. import android.graphics.Bitmap;  
  8. import android.graphics.Bitmap.Config;  
  9. import android.graphics.BitmapFactory;  
  10. import android.graphics.Canvas;  
  11. import android.graphics.Color;  
  12. import android.graphics.Paint;  
  13. import android.graphics.Point;  
  14. import android.graphics.PointF;  
  15. import android.graphics.Rect;  
  16. import android.media.FaceDetector;  
  17. import android.media.FaceDetector.Face;  
  18. import android.os.Bundle;  
  19. import android.os.Handler;  
  20. import android.os.Message;  
  21. import android.util.DisplayMetrics;  
  22. import android.util.Log;  
  23. import android.view.Menu;  
  24. import android.view.View;  
  25. import android.view.View.OnClickListener;  
  26. import android.view.ViewGroup;  
  27. import android.view.ViewGroup.LayoutParams;  
  28. import android.widget.Button;  
  29. import android.widget.ImageView;  
  30. import android.widget.ProgressBar;  
  31. import android.widget.RelativeLayout;  
  32.   
  33. public class MainActivity extends Activity {  
  34.     static final String tag = "yan";  
  35.     ImageView imgView = null;  
  36.     FaceDetector faceDetector = null;  
  37.     FaceDetector.Face[] face;  
  38.     Button detectFaceBtn = null;  
  39.     final int N_MAX = 2;  
  40.     ProgressBar progressBar = null;  
  41.   
  42.     Bitmap srcImg = null;  
  43.     Bitmap srcFace = null;  
  44.     Thread checkFaceThread = new Thread(){  
  45.   
  46.         @Override  
  47.         public void run() {  
  48.             // TODO Auto-generated method stub  
  49.             Bitmap faceBitmap = detectFace();  
  50.             mainHandler.sendEmptyMessage(2);  
  51.             Message m = new Message();  
  52.             m.what = 0;  
  53.             m.obj = faceBitmap;  
  54.             mainHandler.sendMessage(m);  
  55.               
  56.         }  
  57.   
  58.     };  
  59.      Handler mainHandler = new Handler(){  
  60.   
  61.         @Override  
  62.         public void handleMessage(Message msg) {  
  63.             // TODO Auto-generated method stub  
  64.             //super.handleMessage(msg);  
  65.             switch (msg.what){  
  66.             case 0:  
  67.                 Bitmap b = (Bitmap) msg.obj;  
  68.                 imgView.setImageBitmap(b);  
  69.                 MyToast.showToast(getApplicationContext(), "检测完毕");  
  70.                 break;  
  71.             case 1:  
  72.                 showProcessBar();  
  73.                 break;  
  74.             case 2:  
  75.                 progressBar.setVisibility(View.GONE);  
  76.                 detectFaceBtn.setClickable(false);  
  77.                 break;  
  78.             default:  
  79.                 break;  
  80.             }  
  81.         }  
  82.   
  83.     };  
  84.     @Override  
  85.     protected void onCreate(Bundle savedInstanceState) {  
  86.         super.onCreate(savedInstanceState);  
  87.         setContentView(R.layout.activity_main);  
  88.         initUI();   
  89.         initFaceDetect();  
  90.         detectFaceBtn.setOnClickListener(new OnClickListener() {  
  91.   
  92.             @Override  
  93.             public void onClick(View v) {  
  94.                 // TODO Auto-generated method stub  
  95.                 mainHandler.sendEmptyMessage(1);  
  96.                 checkFaceThread.start();  
  97.                   
  98.             }  
  99.         });  
  100.   
  101.   
  102.   
  103.     }  
  104.   
  105.     @Override  
  106.     public boolean onCreateOptionsMenu(Menu menu) {  
  107.         // Inflate the menu; this adds items to the action bar if it is present.  
  108.         getMenuInflater().inflate(R.menu.main, menu);  
  109.         return true;  
  110.     }  
  111.     public void initUI(){  
  112.   
  113.         detectFaceBtn = (Button)findViewById(R.id.btn_detect_face);  
  114.         imgView = (ImageView)findViewById(R.id.imgview);  
  115.         LayoutParams params = imgView.getLayoutParams();  
  116.         DisplayMetrics dm = getResources().getDisplayMetrics();  
  117.         int w_screen = dm.widthPixels;  
  118.         //      int h = dm.heightPixels;  
  119.   
  120.         srcImg = BitmapFactory.decodeResource(getResources(), R.drawable.kunlong);  
  121.         int h = srcImg.getHeight();  
  122.         int w = srcImg.getWidth();  
  123.         float r = (float)h/(float)w;  
  124.         params.width = w_screen;  
  125.         params.height = (int)(params.width * r);  
  126.         imgView.setLayoutParams(params);  
  127.         imgView.setImageBitmap(srcImg);  
  128.     }  
  129.   
  130.     public void initFaceDetect(){  
  131.         this.srcFace = srcImg.copy(Config.RGB_565, true);  
  132.         int w = srcFace.getWidth();  
  133.         int h = srcFace.getHeight();  
  134.         Log.i(tag, "待检测图像: w = " + w + "h = " + h);  
  135.         faceDetector = new FaceDetector(w, h, N_MAX);  
  136.         face = new FaceDetector.Face[N_MAX];  
  137.     }  
  138.     public boolean checkFace(Rect rect){  
  139.         int w = rect.width();  
  140.         int h = rect.height();  
  141.         int s = w*h;  
  142.         Log.i(tag, "人脸 宽w = " + w + "高h = " + h + "人脸面积 s = " + s);  
  143.         if(s < 10000){  
  144.             Log.i(tag, "无效人脸,舍弃.");  
  145.             return false;  
  146.         }  
  147.         else{  
  148.             Log.i(tag, "有效人脸,保存.");  
  149.             return true;      
  150.         }  
  151.     }  
  152.     public Bitmap detectFace(){  
  153.         //      Drawable d = getResources().getDrawable(R.drawable.face_2);  
  154.         //      Log.i(tag, "Drawable尺寸 w = " + d.getIntrinsicWidth() + "h = " + d.getIntrinsicHeight());  
  155.         //      BitmapDrawable bd = (BitmapDrawable)d;  
  156.         //      Bitmap srcFace = bd.getBitmap();  
  157.   
  158.         int nFace = faceDetector.findFaces(srcFace, face);  
  159.         Log.i(tag, "检测到人脸:n = " + nFace);  
  160.         for(int i=0; i<nFace; i++){  
  161.             Face f  = face[i];  
  162.             PointF midPoint = new PointF();  
  163.             float dis = f.eyesDistance();  
  164.             f.getMidPoint(midPoint);  
  165.             int dd = (int)(dis);  
  166.             Point eyeLeft = new Point((int)(midPoint.x - dis/2), (int)midPoint.y);  
  167.             Point eyeRight = new Point((int)(midPoint.x + dis/2), (int)midPoint.y);  
  168.             Rect faceRect = new Rect((int)(midPoint.x - dd), (int)(midPoint.y - dd), (int)(midPoint.x + dd), (int)(midPoint.y + dd));  
  169.             Log.i(tag, "左眼坐标 x = " + eyeLeft.x + "y = " + eyeLeft.y);  
  170.             if(checkFace(faceRect)){  
  171.                 Canvas canvas = new Canvas(srcFace);  
  172.                 Paint p = new Paint();  
  173.                 p.setAntiAlias(true);  
  174.                 p.setStrokeWidth(8);  
  175.                 p.setStyle(Paint.Style.STROKE);  
  176.                 p.setColor(Color.GREEN);  
  177.                 canvas.drawCircle(eyeLeft.x, eyeLeft.y, 20, p);  
  178.                 canvas.drawCircle(eyeRight.x, eyeRight.y, 20, p);  
  179.                 canvas.drawRect(faceRect, p);  
  180.             }  
  181.   
  182.         }  
  183.         ImageUtil.saveJpeg(srcFace);  
  184.         Log.i(tag, "保存完毕");  
  185.           
  186.         //将绘制完成后的faceBitmap返回  
  187.         return srcFace;  
  188.   
  189.     }  
  190.     public void showProcessBar(){  
  191.         RelativeLayout mainLayout = (RelativeLayout)findViewById(R.id.layout_main);  
  192.         progressBar = new ProgressBar(MainActivity.this, null, android.R.attr.progressBarStyleLargeInverse); //ViewGroup.LayoutParams.WRAP_CONTENT  
  193.         RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);  
  194.         params.addRule(RelativeLayout.ALIGN_PARENT_TOP, RelativeLayout.TRUE);  
  195.         params.addRule(RelativeLayout.CENTER_HORIZONTAL, RelativeLayout.TRUE);  
  196.         progressBar.setVisibility(View.VISIBLE);  
  197.         //progressBar.setLayoutParams(params);  
  198.         mainLayout.addView(progressBar, params);  
  199.           
  200.     }  
  201.   
  202.   
  203. }  


关于上述代码,注意以下几点:

1、 在initUI()函数里初始化UI布局,主要是将ImageView的长宽比设置。根据srcImg的长宽比及屏幕的宽度,设置ImageView的宽 度为屏幕宽度,然后根据比率得到ImageView的高。然后将Bitmap设置到ImageView里。一旦设置了ImageView的长和 宽,Bitmap会自动缩放填充进去,所以对Bitmap就无需再缩放了。

2、 initFaceDetect()函数里初始化人脸识别所需要的变量。首先将Bitmap的ARGB格式转换为RGB_565格式,这是android自 带人脸识别要求的图片格式,必须进行此转化:this.srcFace = srcImg.copy(Config.RGB_565, true);

然后实例化这两个变量:

FaceDetector faceDetector = null;
FaceDetector.Face[] face;

faceDetector = new FaceDetector(w, h, N_MAX);
face = new FaceDetector.Face[N_MAX];

FaceDetector就是用来进行人脸识别的类,face是用来存放识别得到的人脸信息。N_MAX是允许的人脸个数最大值。

3、真正的人脸识别在自定义的方法detectFace()里,核心代码:faceDetector.findFaces(srcFace, face)。在识别后,通过Face f  = face[i];得到每个人脸f,通过 float dis = f.eyesDistance();得到两个人眼之间的距离,f.getMidPoint(midPoint);得到人脸中心的坐标。下面这两句话得到左右人眼的坐标:

 

[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. Point eyeLeft = new Point((int)(midPoint.x - dis/2), (int)midPoint.y);  
  2. Point eyeRight = new Point((int)(midPoint.x + dis/2), (int)midPoint.y);  


下面是得到人脸的矩形:

 

[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. Rect faceRect = new Rect((int)(midPoint.x - dd), (int)(midPoint.y - dd), (int)(midPoint.x + dd), (int)(midPoint.y + dd));  


注意这里Rect的四个参数其实就是矩形框左上顶点的x 、y坐标和右下顶点的x、y坐标。

4、实际应用中发现,人脸识别会发生误判。所以增加函数checkFace(Rect rect)来判断,当人脸Rect的面积像素点太小时则视为无效人脸。这里阈值设为10000,实际上这个值可以通过整个图片的大小进行粗略估计到。

5、为了让用户看到正在识别的提醒,这里动态添加一个ProgressBar。代码如下:

 

[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. public void showProcessBar(){  
  2.     RelativeLayout mainLayout = (RelativeLayout)findViewById(R.id.layout_main);  
  3.     progressBar = new ProgressBar(MainActivity.this, null, android.R.attr.progressBarStyleLargeInverse); //ViewGroup.LayoutParams.WRAP_CONTENT  
  4.     RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);  
  5.     params.addRule(RelativeLayout.ALIGN_PARENT_TOP, RelativeLayout.TRUE);  
  6.     params.addRule(RelativeLayout.CENTER_HORIZONTAL, RelativeLayout.TRUE);  
  7.     progressBar.setVisibility(View.VISIBLE);  
  8.     //progressBar.setLayoutParams(params);  
  9.     mainLayout.addView(progressBar, params);  
  10.   
  11. }  


事实上这个ProgressBar视觉效果不是太好,用ProgressDialog会更好。这里只不过是提供动态添加ProgressBar的方法。

6、 程序中设置了checkFaceThread线程用来检测人脸,mainHandler用来控制UI的更新。这里重点说下Thread的构造方法,这里是 模仿源码中打开Camera的方法。如果一个线程只需执行一次,则通过这种方法是最好的,比较简洁。反之,如果这个Thread在执行后需要再次执行或重 新构造,不建议用这种方法,建议使用自定义Thread,程序逻辑会更容易 控制。在线程执行完毕后,设置button无法再点击,否则线程再次start便会挂掉。

 

[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. Thread checkFaceThread = new Thread(){  
  2.   
  3.     @Override  
  4.     public void run() {  
  5.         // TODO Auto-generated method stub  
  6.         Bitmap faceBitmap = detectFace();  
  7.         mainHandler.sendEmptyMessage(2);  
  8.         Message m = new Message();  
  9.         m.what = 0;  
  10.         m.obj = faceBitmap;  
  11.         mainHandler.sendMessage(m);  
  12.   
  13.     }  
  14.   
  15. };  

7、看下识别效果:

原图:


识别后:

最后特别交代下,当人眼距离少于100个像素时会识别不出来。如果静态图片尺寸较少,而手机的densityDpi又比较高的话,当图片放在drawable-hdpi文件夹下时会发生检测不到人脸的情况,同样的测试图片放在drawable-mdpi就可以正常检测。原因是不同的文件夹下,Bitmap加载进来后的尺寸大小不一样。

后续会推出Camera里实时检测并绘制人脸框,进一步研究眨眼检测,眨眼控制拍照的demo,敬请期待。如果您觉得笔者在认真的写博客,请为我投上一票。

CSDN2013博客之星评选:

http://vote.blog.csdn.net/blogstaritem/blogstar2013/yanzi1225627

本文demo下载链接:

http://download.csdn.net/detail/yanzi1225627/6783575

 

参考文献:

链接1:

链接2:

转载于:https://www.cnblogs.com/xgjblog/p/3853647.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/376823.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

图论:最短路径搜索--Dijkstra算法(c代码实现)

最近因为辞职&#xff0c;有不少闲功夫&#xff0c;重温下数据结构&#xff0c;顺便练练手。今天说说最短路径搜索算法中的Dijkstra原理和实现。 一&#xff1a;简介 这个算法用于解决图中单源最短路径问题。所谓单源节点是指给定源节点&#xff0c;求图中其它节点到此源节点的…

C++多线程快速入门(五)简单线程池设计

目录设计思路主线程运行逻辑task以及taskpool设计详细流程讲解完整代码打印结果往期回顾设计思路 线程池实际上就是一组线程&#xff0c;当我们需要异步执行一些任务时&#xff0c;经常要通过OS频繁创建和销毁线程&#xff0c;不如直接创建一组在程序生命周期内不会退出的线程…

C--函数

函数:具有特定功能的代码段,分为库函数,自定义函数. 函数定义: 函数返回值类型 函数名(形式参数列表) { 代码段; return 返回值; } 注意:每个函数返回值最多只有一个.return是一个函数结束的标志. 形式参数(形参):函数定义时使用的虚拟参数名,用以接收函数调用是传递过来的实际…

公式系统 - TradeBlazer公式基础 - Bar数据

Bar数据 在介绍Bar数据之前&#xff0c;首先&#xff0c;我们需要讨论一下TradeBlazer公式的计算方法&#xff0c;针对上面介绍的各种公式类型&#xff0c;包含公式应用&#xff0c;在公式进行计算时&#xff0c;都是建立在基本数据源(Bar数据)之上&#xff0c;我们这里所谓的B…

C++网络编程快速入门(一):TCP网络通信基本流程以及基础函数使用

目录流程概述服务器端代码实现客户端代码实现函数和结构讲解sockaddr_in和sockaddrsocket &#xff1a; 创建一个socket连接bind &#xff1a;绑定地址以及端口号问题流程概述 客户端与服务器之间的网络通信基本原理如下所示&#xff0c;复杂一点的架构可能会添加消息中间件。…

php 字符串处理

addcslashes — 为字符串里面的部分字符添加反斜线转义字符addslashes — 用指定的方式对字符串里面的字符进行转义bin2hex — 将二进制数据转换成十六进制表示chop — rtrim() 的别名函数chr — 返回一个字符的ASCII码chunk_split — 按一定的字符长度将字符串分割成小块conve…

使用前端框架Foundation 4来帮助简化响应式设计开发

日期&#xff1a;2013-3-12 来源&#xff1a;GBin1.com Foundation是一套使用广泛的前端开发套件&#xff0c;可以帮助你快速的网站。最近ZURB发布了一个新版本的Foundation 4前端框架&#xff0c;能够有效的帮助你快速的开发响应式的网站。 和另外一个套知名的前端框架BootSt…

C++网络编程快速入门(二):Linux下使用select演示简单服务端程序

目录select参数解释select使用规范select使用缺点基本流程实例代码通信效果演示往期文章select参数解释 extern int select (int __nfds, fd_set *__restrict __readfds,fd_set *__restrict __writefds,fd_set *__restrict __exceptfds,struct timeval *__restrict __timeout)…

Android转载一:Android文件命名规范

REF&#xff1a;http://blog.csdn.net/gulianchao/article/details/23391651 (一) Layout命名 1&#xff0e;contentview命名&#xff1a;activity_功能模块.xml 例如&#xff1a;activity_main.xml、activity_more.xml 2&#xff0e;Dialog命名&#xff1a;dialog_描述.xml …

[转]XBRL应用软件分类

1) 分类标准编辑软件(Taxonomy editor)&#xff1a; 分类标准是XBRL技术的应用基础&#xff0c;每一个采用XBRL技术的国家都必须先按各国的GAAP制订XBRL分类标准&#xff0c;上市公司才能据以编制实例文件。由于一套XBRL 2.0或2.1版分类标准必须包含至少一份XML Schema文…

C++网络编程快速入门(三):阻塞与非阻塞式调用网络通信函数

目录阻塞与非阻塞定义send与recvconnect一些问题为什么要将监听socket设置为非阻塞阻塞与非阻塞定义 阻塞模式指的是当前某个函数执行效果未达预期&#xff0c;该函数会阻塞当前的执行线程&#xff0c;程序执行流在超时时间到达或者执行成功后恢复原有流程。非阻塞模式相反&am…

css3实现头像旋转360度

css样式: .div a img{ width: 88px; height: 88px; border-radius: 88px; transition: all 1.2s ease-out 0s;}.div a img:hover{ -webkit-transform:rotate(360deg); -moz-transform:rotate(360deg); -o-transform:rotate(360deg); -ms-transform:rotate(360deg); transform:r…

POJ 2488 深搜

POJ 2488 深搜 要求字典序的顺序。 1 #include <iostream>2 #include <stdio.h>3 #include <string.h>4 using namespace std;5 int n,m,cnt;6 bool success;7 bool sign[30][30];8 int step[30][2];9 int dir[8][2]{ 10 -2,-1,-2,1, 11 …

socket 端口和地址复用

https://blog.csdn.net/weibo1230123/article/details/79978745 https://blog.csdn.net/weixin_42157432/article/details/115560824 在linux socket网络编程中&#xff0c;大规模并发TCP或UDP连接时&#xff0c;经常会用到端口复用&#xff1a; int opt 1; if (setsockopt…

MyEclipse老是弹出problem occurred窗口

有的时候是因为jsp页面中的java脚本有误&#xff0c;比如说<% String name"";>就会出现错误&#xff0c;因为结束标签少了一个百分号&#xff05;。转载于:https://www.cnblogs.com/passer1991/archive/2013/03/15/2961624.html

C++网络编程快速入门(四):EPOLL模型使用

目录基本使用方法step1:创建epollfdstep2:将fd绑定到epollfdstep3:调用epoll_wait检测事件epoll_wait与poll、select区别所在水平触发与边缘触发基本使用方法 step1:创建epollfd 创建一个epollfd&#xff0c;若epoll_create调用成功&#xff0c;则返回一个非负值的epollfd&am…

Mysql中代替like模糊查询的一种方法

使用Mysql的函数instr,可代替传统的like方式查询,并且速度更快。 instr函数&#xff0c;第一个参数是字段&#xff0c;第二个参数是要查询的串&#xff0c;返回串的位置&#xff0c;第一个是1&#xff0c;如果没找到就是0. 例如&#xff1a; select username from prefix_user …

两种大小端判断的方式

网络通信是按照字节流进行数据交换的&#xff0c;主机根据不同的CPU型号可能是大段存储&#xff0c;也可能是小端存储。而网络字节序在TCP/IP协议中已经规定好了&#xff0c;采用大端的排序方式。 所以网络通信中一般将需要传输的整数型值转换成网络字节序。 从本机字节序转换成…

把数据库复制成脚本(包含远程以及数据库数据)

1.启动VS 2.服务器资源管理器 3.连接需要的数据库 4.右键数据库 选择publist to provider.... 5.剩下的 选择数据库 选择存放地址 下一步 这方法应该是用在把08的数据还原到05上面 明天用这个方法去盗取哈公司的数据库 看行不行转载于:https://www.cnblogs.com/Rock-Lee/a…

代理模式用来初始化的延迟下载

package 设计模式; //代理模式实现延迟加载来减小启动时间 //数据库查询接口 interface IDBQery{ public String request(); }class DBQuery implements IDBQery {//创建一个DBQery非常耗时的&#xff0c;这里面我可以在需要DBQuery的时候在创建public DBQuery(){try {Thread.s…