[总结]Perl在遇到Unicode字符文件名时的各种处理方法
环境 XP/WIN7 Perl v5.16
编辑整理:523066680
常见的那些文件操作函数都不支持,于是为了达到目的,需要各种方法配合,应该是不如其他语言方便。
我只是想看看Perl到底是否适合做这件事,于是折腾了一回。
文件的建立:
- 模块:Win32
- use Win32;
- use utf8;
- use Encode;
- #接受unicode传参
- Win32::CreateFile("W32CreateFile・测试");
复制代码
- 特性: 成功返回true,但不返回文件句柄
- Creates the FILE and returns a true value on success.
- Check $^E on failure for extended error information.
- 模块:Win32API::File
- 函数:$hObject= CreateFileW( $swPath, $uAccess, $uShare, $pSecAttr, $uCreate, $uFlags, $hModel )
- $hObject可以返回文件对象句柄
- 注意事项: 传入的文件路径的编码格式为:UTF16-LE ,必须以\x00结尾,示例(代码保存为utf8格式):
- use Win32API::File qw(:ALL);
- use utf8;
- use Encode;
- $str="文tes・t.txt\x00";
- $hobject=CreateFileW(encode('UTF16-LE', $str), GENERIC_WRITE, 0, [], OPEN_ALWAYS,0,0);
复制代码
目录的建立
- 模块:Win32
- use Win32;
- use utf8;
- Win32::CreateDirectory("Dir・测试");
复制代码
文件的枚举
- 在遇到unicode字符的时候,File::Find模块 以及 IO::Dir 模块都只能输出文件短名。
- 暂时用CMD /U Dir 的方法输出文件列表(郁闷吧,暂时没找到能完美操作的内置模块)
- 参考文章
http://www.perlmonks.org/?node_id=536223
- how to read unicode filename
复制某个文件夹内的文件(文件名含unicode字符)
- 模块:Win32API::File
- 如果先获取文件的短名,然后再复制,但是目标文件名也会变成短名。
- 于是暂时用cmd /U 模式获取文件列表,然后CopyFileW进行复制:
- use Win32API::File qw':ALL';
- use Encode;
- use utf8;
- my $src=encode('gbk','.\\测试目录');
- my $dst='.\\Target';
- #该目录只有一层,/s开关是为了列出完整的路径
- my $all=`cmd /U /C dir /s /b \"$src\"`;
- my $fn;
- foreach (split(/\x0d\x00\x0a\x00/, $all)) {
- $fn = encode('gbk', decode('utf16-le',$_))."\n";
- @xrr=split(/\x5c\x00/, $_);
- CopyFileW(
- $_ ."\x00",
- encode('utf-16le', decode('utf8', "$dst\\")).$xrr[$#xrr]."\x00",
- 1
- );
- print "$^E\n" if ($^E);
- }
- <STDIN>;
复制代码
- 细节一、
- 正确地使用 split $all 截断utf-16le字符段落,分隔符为0d 00 0a 00
- 参考枚举脚本
- 细节二、
- 如果用basename()分割路径,同样会遇到00被忽略的问题,'\\' 的U16LE
- 编码是5C 00,但是basename 只按5C截断,剩下的00造成了处理乱码。
- 测试basename的第二个参数设置为 "\x5c\x00" 并不能解决这个问题
- 解决方法一、
- 手工去掉开头处00
- 方法二、
- 先转为GBK,再获取basename,再转utf-16le
- 2014-12-12 备注这种方法在LongPath的情况下,会丢失unicode字符
- 可以考虑转为UTF-8,不管怎么说都有点绕
- 方法三、
- 自己用正则表达式获取
- /\x5C\x00([^\x5c]+)$/;
- $1
- 方法四、
- @xrr=split(/\x5c\x00/, $_);
- $xrr[$#xrr]
- 细节三、
- CopyFileW复制文件时,要在末尾加\x00作为字符串终止符
- 否则各种问题=_=
判断文件是否存在:
- 方法一:先转为短名再判断,不做赘述
- 方法二:渣方法,用CreateFileW测试建立同名文件,看是否有冲突
重命名:
- 模块:Win32API::File
- MoveFileW(
- encode('utf-16le', decode('utf8',$F))."\x00",
- encode('utf-16le', decode('utf8',$newname))."\x00"
- );
复制代码
获取文件的日期信息:
- 普通文件名的情况
http://stackoverflow.com/questions/1839877/
- how-can-i-get-a-files-modification-date-in-ddmmyy-format-in-perl
- 含有Unicode字符的文件名的情况
http://www.perlmonks.org/?node_id=741797
- How to stat a file with a Unicode (UTF16-LE) filename in Windows?
- 其中的方法是通过createfileW 获取文件句柄,然后用OsFHandleOpen获取通用的文件句柄对象,并传入state
- (感觉特别绕)
- 另一种就是先转为短名再获取日期,但是这种方法在处理文件量大的时候,效率非常低。前面perlmonks中的方法
- 效率要高得多
- use utf8;
- use Encode;
- use Win32;
- $filename='D:\测试目录\董贞 ・ 01.剑如虹.[贞江湖].mp3';
- $filename=Win32::GetShortPathName($filename);
- my $mtime = (stat $filename)[9];
- my($sec,$min,$hour,$mday,$mon,$year,$wday,$yday,$isdst) = localtime($mtime);
- $year+=1900;
- $mon+=1;
- print "$year-$mon-$mday\n";
- <STDIN>;
复制代码
[总结]Perl在遇到Unicode字符文件名时的各种处理方法的更多相关文章
- 扩展ASCII码,不同的国家有不同的字符集。Unicode转换为utf8的规则,utf8没有大小端的问题。超过0xFFFF的Unicode字符WINAPI也无能为力(附各种字符编码表及转换表)good
一.概念 1,ASCII ASCII(American Standard Code for Information Interchange),中文名称为美国信息交换标准代码.是 ...
- [Perl]Windows 系统 Unicode 文件名操作(新建、重命名、枚举、复制)全攻略
[Perl] Windows 系统 Unicode 文件名操作(新建.重命名.枚举.复制)全攻略 环境 XP/WIN7 Perl v5.16 编辑整理:PerlMonk.523066680 常见的那些 ...
- 转载一篇关于unicode字符编码的文章
很久很久以前,有一群人,他们决定用8个可以开合的晶体管来组合成不同的状态,以表示世界上的万物.他们认为8个开关状态作为原子单位很好,于是他们把这称为"字节". 再后来,他们又做了一 ...
- 在多字节的目标代码页中,没有此 Unicode 字符可以映射到的字符。 (#1113)
报错 在使用MySQL-Front导入sql文件时报错1113:在多字节的目标代码页中,没有此 Unicode 字符可以映射到的字符. (#1113) 解决方案 导入.sql文件时,单击 选择文件对话 ...
- Unicode字符列表(超完整)
Unicode字符列表(超完整)Unicode, 字符, 列表代码 显示 描述U+0020 空格 U+0021 ! 叹号 U+0022 " 双引号 U+0023 # 井号 U+0024 $ ...
- python print输出unicode字符
命令行提示符下,python print输出unicode字符时出现以下 UnicodeEncodeError: 'gbk' codec can't encode character '\u30fb ...
- 解决“在多字节的目标代码页中,没有此Unicode字符可以映射到的字符”
今天在处理Google网站管理员中的500错误时发现这样一些URL: http://www.cnblogs.com/Garnai/tag/3D%3F%96%CA/ http://www.cnblogs ...
- 转:Unicode字符集和多字节字符集关系
原文地址: http://my.oschina.net/alphajay/blog/5691 unicode.ucs-2.ucs-4.utf-16.utf-32.utf-8 http://stallm ...
- 字符和字符串处理-ANSI字符和Unicode字符
我们知道,C语言用char数据类型表示一个8位的ANSI字符,默认在代码中声明一个字符串时,C编译器会把字符串中的字符转换成由8位char数据类型构成的一个数组: // An 8-bit charac ...
随机推荐
- 自动生成缓存Key值的CacheKeyHelper
/// <summary> /// CacheKeyHelper /// </summary> public class CacheKeyHelper { /// <su ...
- [转] CSS 颜色名
[From] http://www.w3school.com.cn/cssref/css_colornames.asp CSS 颜色名 所有浏览器都支持的颜色名. HTML 和 CSS 颜色规范中定义 ...
- panda强化练习2
In [1]: import pandas as pd import numpy as np import matplotlib.pyplot as plt plt.rcParams['font.sa ...
- 文献综述八:基于JAVA的商品网站的研究
一.基本信息 标题:基于JAVA的商品网站的研究 时间:2015 出版源:信息技术 文件分类:对java语言的研究 二.研究背景 本文主要介绍了系统的分析,设计和开发的全部过程. 三.具体内容 文献的 ...
- 问题记录——java.lang.IllegalArgumentException: Illegal character in scheme name at index 0
以下http请求报错是因为,请求的地址前面有个空格.... 2019-01-09 03:30:23,154 ERROR [business.modules.merchantreportresult.s ...
- 016-hibernateutils模板
package ${enclosing_package}; import org.hibernate.HibernateException; import org.hibernate.Session; ...
- jquery 添加关键字小插件
模块化封装,兼容seajs /** * User: c3t * Date: 14-3-25 * Time: 下午4:16 */ define(function (require, exports, m ...
- mysql中字段类型是datetime时比较大小问题
select sum(studychj) as tofflinejz from afterline where studybegin >= '2010-01-01 00:00:00' and s ...
- CSS像素、设备独立像素、设备像素之间关系
CSS像素.设备独立像素.设备像素,三者联系紧密又有很大的区别,而我们主要是在做移动端开发的时候需要更多地用到这些概念,那他们分别是指什么呢? 概念 CSS像素(CSS Pixel):适用于web编程 ...
- Basic Data Structures and Algorithms in the Linux Kernel--reference
http://luisbg.blogalia.com/historias/74062 Thanks to Vijay D'Silva's brilliant answer in cstheory.st ...