/* * Java 图片转换成字符图 CharMaps (整理) * * 2016-1-2 深圳 南山平山村 曾剑锋 * * @(#)CharMaps.java 2014/1/16 * 1、这个一个Java程序,感谢您花费大量时间阅读本文档; * 2、本人知道大家并不喜欢看大量文字描述,但实属无奈,因为我们的沟通只能通过文字; * 3、当您在复制、粘贴的时候请注意包名为:practice,文件名为:CharMaps,以防止一些不必要的麻烦; * 4、下面这张由字符组成的图是直接由图片生成的,信与不信由您决定,另外可以肯定的是本人绝对不会一个字一个敲这幅图; * 5、如果想知道她是如何完成的,请看完CharMaps类前面的全部注释,因为她是工具,有些内容需要了解、素材需要准备; * 6、本次注释参考了MyEclipse的注释风格,相对比上次来说注释要规范一些,但由于本人经验、认知水平有限,可能很多地方没有 * 考虑完全,或者没有解释清楚,请谅解。 */ package demo; import java.awt.image.BufferedImage; import java.io.BufferedWriter; import java.io.File; import java.io.FileWriter; import java.io.IOException; import java.util.Arrays; import java.util.regex.Matcher; import java.util.regex.Pattern; import javax.imageio.ImageIO; import javax.swing.JFileChooser; import javax.swing.filechooser.FileNameExtensionFilter; /** * * <p> * <h4>一、软件声明:</h4><br><ol> * <li>该软件是为了解决TTS无法上图的原因才写出来的,同时感谢您花费宝贵的时间来阅读本文档;<br> * <li>如果您打算使用这个小工具,那么请阅读完这个这整段注释; * <li>如果您只是路过,那么您可以也可以猫一眼 ^_^,不过我相信她不会让您失望的;<br> * <li>该软件已被我命名为CharMaps,在后面的注释中,我说明了命名理由 ^_^;<br> * <li>TTS最小字号是9px阻碍了CharMaps的强大功能,但CharMaps至少能缓解一下我们对图片的向往,希望TTS以后会放宽规则;<br> * * 该小工具被我命名为:CharMaps,没有抄袭他人的思路,是根据自己的需求写的,希望她能给您带来乐趣<br> * </ol></p> * * <p> * <h4>二、CharMaps由来:</h4><br><ol> * <li>CharMaps制作初衷是因为TTS不能上图,之前上传《动态写轮眼(火影)——Java原创》系列软件时感觉实在不爽,所以自己通过 * 自己目前的认知水平写了这个CharMaps。<br> * <li>灵感来自硬件传感器——摄像头——的工作原理,有兴趣的朋友可以去找下度娘或者谷哥,听说他们很懂的样子。 ^_^<br> *</ol></p> * * <p> * <h4>三、CharMaps命名解析:</h4><br><ol> * <li>Char代表字符,因为她将图片里对一个的像素转换成了字符,为什么是字符而不是字节?原因在于字节宽高比大约是1:2.5, * 而字符的宽高大约是1:1,我说的是大约 ^_^;<br> * <li>Map是代表输出的就像每张图片一个映射(map);<br> * <li>s是因为能同时处理多张图片;<br> * <li>最终组合为CharMaps,很优雅的名字,^_^ ;<br> *</ol></p> * * <p> * <h4>四、CharMaps知识预备:</h4><br><ol> * <li>CharMaps采用了具有RGB(red,green,blue)三基色的图片,后缀名为png/PNG/jpeg/jpe/JPEG,但只能处理黑白 * 图,原因是黑白图在三基色上是平均分布的,而CharMaps仅是依靠blue分量进行图片处理。<br> * <li>如果您手上没有黑白图,您可以使用我们使用的操作系统自带的The GIMP,该软件自带了离线中文帮助文档,界面挺清爽 * 的,您可以通过以下途径打开:应用程序 ——> 图像 ——> The GIMP,如果您不会使用,那就看看帮助文档吧,如果 * 您以前使用Photoshop(PS),估计您会有很熟悉的感觉,您以前投资在PS上的时间终于得到了有效回报, ^_^。<br> * </ol></p> * * <p> * <h4>五、CharMaps缺陷:</h4><br><ol> * <li>目前只支持黑白图,其他的图不保证效果,主要是写复杂了不利于学习交流,简单并能提供一些思路,并且每个人可以根据 * 自己需要进行功能改进或者定制才是沟通学习的王道;<br> * <li>所能转换的图片目前默认支持宽:int charMapsCol = 2000;高:int charMapsRow = 1000;您自己可以修改;<br> * <li>没有提供图片的自动缩放功能,所以不要操作默认支持的宽高,或者你自己修改的宽高;<br> * <li>目前没有发现Bug,不保证您在使用的时候会不会出现Bug,就算遇到了,相信您也能够独立解决,TTS不提供沟通讨论,没办法, * 另外本软件代码行数150左右(不包括注释),您懂的 ;^_^<br> * <li>如果这个小软件给您带来了不必要的麻烦,本人向您表示歉意。<br> * </ol></P> * * <p> * <h4>六、CharMaps操作流程:</h4><br><ol> * <li>准备好一张或者多张后缀是png/PNG/jpeg/jpe/JPEG的文件,运行本程序;<br> * <li>在文件选择框选择要转换的后缀是png/PNG/jpeg/jpe/JPEG的文件,选择完成后点“打开”按钮;不选择就会得到一个空文件;<br> * <li>根据Console提示框中内容找到输出文件,默认是路径/home/soft1/charMaps.txt,您可以自己修改;<br> * <li>用文本编辑器打开,把字号(不是字体)改成2px,看看效果吧。<br> * </ol></p> * * <p> * <h4>七、CharMaps工作流程图(非代码分析的朋友可以略过):</h4><br><ol> * 本次注释参考了MyEclipse的注释风格,注释尽量简洁,没有添加过多的额外的注释,以下为main()函数工作流程层次:<br> * <ul> * <li>|--filesSelectInit() //初始化文件选择器,主要用于过滤掉一些后缀不是png/PNG/jpeg/jpe/JPEG的文件,方便选择文件;<br> * <li>|--charMapsInit() //初始化charMaps数组,并以字符‘虢’填充,主要是笔画多,作为黑色背景好,还有就是字里有老虎 ^_^;<br> * <li>|--ImageProcessing() //图片处理函数,里面包含了如何对图片进行处理的方法;<br> * <ul><li>|--singleImageProcessing() //单张图片处理函数;<br> * <ul><li>|--fileSuffixCheck() //文件后缀检查,采用了正则表达式,主要是为了防止有些朋友选错;<br> * <li>|--ImageSizeCheck() //图片大小检查,主要是因为charMaps大小已由charMapsRow和charMapsCol固定了;<br> * <li>|--ImageMapToCharMaps() //将图片(Image)映射(map)到charMaps,所有的图片都是以charMaps中心点进行映射 * 这个步骤中把每张图片的没有内容的边框裁减掉了;<br> * </ul> * </ul> * <li>|--addFrameForCharMaps()//如果转换的图片尺寸没有大于charMaps,那么给图片加个框,您可以通过注释掉这一行 * 看有这个函数和没有这个函数的区别;<br> * <li>|--fileSave() //将最终的结果保存起来,路径已经固定,如果您有需要的话可以自己修改里面的路径。<br> * </ul></ol></P> * @author 曾剑锋<br> * @date 2014-1-16<br> */ public class CharMaps{ /** 声明一个文件选择器引用 */ static JFileChooser jFileChooser = null; /** *        * 在很多时候,我们在转换成为字符的时候,上、下、左、右总有我们那些我们不需要的一些行,这行不 *是我们自己需要的,我们还让软件帮我们处理掉吧,默认赋值有那么点讲究,您应该能看懂的,其实跟后面 *的判断取值有关,初始化值是根据小的取大值,大的取小值的方式,您自己可以看着取。 * */ static int firstOfUpRow = Integer.MAX_VALUE; static int firstOfCol = Integer.MAX_VALUE; static int lastOfUpRow = Integer.MIN_VALUE; static int lastOfCol = Integer.MIN_VALUE; /** * charMapsRow用于定义charMaps的行数<br> * charMapsCol用于定义charMaps的列数<br> */ static int charMapsRow = 1000; static int charMapsCol = 2000; static char[][] charMaps = null; /** 用于保存您选择的单个或者多个文件路径集合, 初始化为null */ static File[] filePaths = null; /** 保存图片的宽、高 */ static int imageWidth = 0; static int imageHeight = 0; /** *         * 在多张图片处理时候,每张图片处理使用不同的字符区别,charSelectindex用于计数,10个字符循环, * 如果一个charMaps同一个位置上字符会被后面的字符替代,如果没有则保持之前的字符<br> */ static int charSelectindex = 0; static char[] charSelects = {'一','二','三','四','五','六','七','八','九','十'}; /** 图像缓冲引用 */ static BufferedImage bufferedImage = null; /** * main()函数,完成任务如下:<br><ol> * <li>对文件选择器进行初始化;<br> * <li>对charMaps二维数组进行初始化;<br> * <li>对已选择的图片集进行处理;<br> * <li>由于图片集处理函数对图片有效边界进行了记录,调用addFrameForCharMaps(),添加在上、下、左、右各 * 添加一个字符的边框,使输出的CharMaps转换出来的图更具视觉效果;<br> * <li>保存转换好的文件;<br> * <li>如果出现异常,给出提示信息。<br></ol> */ public static void main(String[] args) { try { filesSelectInit(); charMapsInit(); ImageProcessing(); addFrameForCharMaps(); fileSave(); } catch (Exception e) { System.out.println("请选择后缀为png/PNG/jpeg/jpe/JPEG的文件"); } } /** * 图像处理函数,完成任务如下:<br><ol> * <li>判断jFileChooser是否按下了打开;<br> * <li>获取一个或者多个文件,保存于filePaths中;<br> * <li>使用for循环迭代分成单张图片依次处理,调用singleImageProcessing()函数处理;<br> * <li>图片处理完给出提示。<br></ol> */ private static void ImageProcessing() throws IOException { //判断是否在文件选择框上点了确定 if (jFileChooser.showOpenDialog(null) == JFileChooser.APPROVE_OPTION) { filePaths = jFileChooser.getSelectedFiles(); for (File filePath : filePaths) { singleImageProcessing(filePath); } System.out.println("3、完成图像处理"); } } /** * 单张图片处理函数,完成任务如下:<br><ol> * <li>调用文件后缀检查函数fileSuffixCheck(),防止因操作失误而引发的异常;<br> * <li>调用图片尺寸检查函数ImageSizeCheck(),防止图片尺寸过大,因为没有提供图片缩放功能;<br> * <li>调用图片映射函数ImageMapToCharMaps(),将图片的像素点映射到CharMaps数组中;<br> * <li>如果是多张图片的charSelectindex完成对charSelects数组中10个字符的自动选择。<br></ol> * * @param filePath 传入filePath进行检查,在返回回来,如果不行就抛弃。<br> * @throws IOException<br> */ private static void singleImageProcessing(File filePath) throws IOException { filePath = fileSuffixCheck(filePath); ImageSizeCheck(filePath); ImageMapToCharMaps(); charSelectindex = charSelectindex++ % 10; } /** * 为charMaps添加边框,完成任务如下:<br> *         * 如果有效的图形区域小于charMaps数组大小,在上、下、左、右按条件添加一个字符的边框,使输出的CharMaps转换出来的图更具视觉效果;<br> */ private static void addFrameForCharMaps() { firstOfCol = firstOfCol > 0 ? firstOfCol-1 : 0; firstOfUpRow = firstOfUpRow > 0 ? firstOfUpRow-1 : 0; lastOfCol = lastOfCol < charMapsCol ? lastOfCol+1 : charMapsCol; lastOfUpRow = lastOfUpRow < charMapsRow ? lastOfUpRow+1 : charMapsRow; } /** * 图片映射函数,完成任务如下:<br><ol> * <li>双重循环获取图片的宽高;<br> * <li>读出图片对应的坐标的RGB值<br> * <li>判断对应坐标点的RGB是背景还是需要转换,这里之使用了B(blue)的值,判断,这也是为什么目前的 * CharMaps只能处理黑白图,当然应该也可以处理白蓝 ^_^<br> * <li>对符合要求的的像素点改变charMaps对应点的字符<br> * <li>对符合要求的像素进行边界检测,主要是完成对图片的边沿检查<br></ol> */ private static void ImageMapToCharMaps() { for (int i = 0; i < imageHeight; i++) { for (int j = 0; j < imageWidth; j++) { int rgb = bufferedImage.getRGB(j, i); if ((rgb&0xff)<128 ) { charMaps[charMaps.length/2-imageHeight/2+i][charMaps[0].length/2-imageWidth/2+j] = charSelects[charSelectindex]; boundedRangeOfImage(i,j); } } } } /** * 图片边界范围检查函数,完成任务如下:<br> *         * 主要是完成检查当前点是否是图片的有效边缘,对图片上、下、左、右有效区域进行查找,为后面保存时裁剪作准备。<br> * @param i 当前点的行号 * @param j 当前点的列号 */ private static void boundedRangeOfImage(int i, int j) { //记录图形中最上面开始出现图形行号 if (charMaps.length/2-imageHeight/2+i<firstOfUpRow) { firstOfUpRow = charMaps.length/2-imageHeight/2+i; } //记录图形中最下面开始不再出现图形行号 if (charMaps.length/2-imageHeight/2+i>lastOfUpRow) { lastOfUpRow = charMaps.length/2-imageHeight/2+i; } //记录图形中一行里面开始出现图形列号 if (charMaps[0].length/2-imageWidth/2+j<firstOfCol) { firstOfCol = charMaps[0].length/2-imageWidth/2+j; } //记录图形中一行里面开始不再出现图形列号 if (charMaps[0].length/2-imageWidth/2+j>lastOfCol) { lastOfCol = charMaps[0].length/2-imageWidth/2+j; } } /** * 图片尺寸检查函数,完成任务如下:<br> *         * 主要是完成对图片的尺寸进行检查,不要比默认设置或者自己定制的charMaps数组大。<br> */ private static void ImageSizeCheck(File filePath) throws IOException { bufferedImage = ImageIO.read(filePath); //得到图片的长宽 imageWidth = bufferedImage.getWidth(); imageHeight = bufferedImage.getHeight(); if ((charMapsRow < imageHeight) || (charMapsCol < imageWidth)) { System.out.println("图片宽因该小于2000,高小于1000"); return ; } } /** * 文件后缀检查函数,完成任务如下:<br> *         * 采用正则表达式对文件进行匹配。<br> */ private static File fileSuffixCheck(File filePath) { //使用正则表达式来防止选择目前不支持的文件,目前只支持png/PNG/jpeg/jpe/JPEG格式图片 Pattern pattern = Pattern.compile(".+[.][pPJj][nNpP][eEgGpP][gG]?"); Matcher matcher = pattern.matcher(filePath.getName()); if (matcher.matches() == false) { return null; } return filePath; } /** * charMaps初始化函数,Init是初始化的英文单词缩写,完成任务如下:<br> *         * 完成对charMaps初始化,并提示完成。<br> */ private static void charMapsInit() { charMaps = new char[charMapsRow][charMapsCol]; //记得需要初始化,否则好像出不来值 for (int i = 0; i < charMaps.length; i++) { Arrays.fill(charMaps[i], '虢'); } System.out.println("2、完成charMaps数组初始化"); } /** * 文件保存函数,完成任务如下:<br><ol> * <li>设置一个文件保存的路径,这个路径可以自己修改;<br> * <li>创建文件路径下的文件缓冲区;<br> * <li>将charMaps中的字符写好文件中,忽略在上、下、左、右边界之外的部分,只将边界内的字符输出;<br> * <li>每写完一行字符,进行换行;<br> * <li>最后关闭文件缓冲区,如果不关闭,文件缓冲区内的字符可能不会写到文件中,请注意;<br> * <li>提示完成以及提示文件路径。<br><ol> */ private static void fileSave() { JFileChooser jFileChooser = new JFileChooser(); jFileChooser.setFileSelectionMode(JFileChooser.DIRECTORIES_ONLY);//只能选择目录 jFileChooser.showOpenDialog(null); File saveFilePath = new File(jFileChooser.getSelectedFile().getPath()+"charMaps.txt"); try { BufferedWriter bufferedWriter = new BufferedWriter(new FileWriter(saveFilePath)); for (int i = 0; i < charMaps.length; i++) { if ((i >= firstOfUpRow) && (i <= lastOfUpRow)) { bufferedWriter.write(charMaps[i], firstOfCol, lastOfCol-firstOfCol+1); bufferedWriter.newLine(); } } bufferedWriter.close(); } catch (IOException e1) { e1.printStackTrace(); } System.out.println("4、完成文件保存\n"); System.out.println("CharMaps已完成完成工作,请到"+saveFilePath.getPath()+"中查看结果 ^_^\n"); } /** * 文件选择对话框初始化函数,Init是初始化的英文单词缩写,完成任务如下:<br><ol> * <li>提示欢迎使用CharMaps;<br> * <li>创建文件选择对话框;<br> * <li>创建文件选择过滤器;<br> * <li>将文件选择过滤器添加进入文件对话框,还句话说是:使文件选择过滤器有效;<br> * <li>将文件选择对话框设置为可以多选;<br> * <li>提示完成初始化。<br></ol> */ private static void filesSelectInit() { System.out.println("\n\t欢迎使用CharMaps"); jFileChooser = new JFileChooser(); FileNameExtensionFilter filter = new FileNameExtensionFilter( "Images", "jpg", "png","PNG","JPG","jpe","JPE"); jFileChooser.setFileFilter(filter); jFileChooser.setMultiSelectionEnabled(true); System.out.println("1、完成文件选择初始化"); } }
转换的图片:
转换后的图片: