Windows 系统 Unicode 文件名操作(新建、重命名、枚举、复制)全攻略

常见的那些文件操作函数都不支持,于是为了达到目的,需要各种方法配合,应该是不如其他语言方便。
我只是想看看Perl到底是否适合做这件事,于是折腾了一回。

文件的建立:

      模块:Win32

Code: [全选] [展开/收缩] [Download] (example.pl)

  1. use Win32;
  2. use utf8;
  3. use Encode;
  4.  
  5. #接受unicode传参
  6. 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格式):

Code: [全选] [展开/收缩] [Download] (example.pl)

  1. use Win32API::File qw(:ALL);
  2. use utf8;
  3. use Encode;
  4. $str="文tes?t.txt\x00";
  5. $hobject=CreateFileW(encode(‘UTF16-LE‘, $str), GENERIC_WRITE, 0, [], OPEN_ALWAYS,0,0);

目录的建立

      模块:Win32

Code: [全选] [展开/收缩] [Download] (example.pl)

  1. use Win32;
  2. use utf8;
  3.  
  4. 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进行复制:

Code: [全选] [展开/收缩] [Download] (example.pl)

  1. use Win32API::File qw‘:ALL‘;
  2. use Encode;
  3. use utf8;
  4.  
  5. my $src=encode(‘gbk‘,‘.\\测试目录‘);
  6. my $dst=‘.\\Target‘;
  7.  
  8. #该目录只有一层,/s开关是为了列出完整的路径
  9. my $all=`cmd /U /C dir /s /b \"$src\"`;
  10. my $fn;
  11.  
  12. foreach (split(/\x0d\x00\x0a\x00/, $all)) {
  13.     $fn = encode(‘gbk‘, decode(‘utf16-le‘,$_))."\n";
  14.     @xrr=split(/\x5c\x00/, $_);
  15.     CopyFileW(
  16.         $_ ."\x00",
  17.         encode(‘utf-16le‘, decode(‘utf8‘, "$dst\\")).$xrr[$#xrr]."\x00",
  18.         1
  19.     );
  20.     print "$^E\n" if ($^E);
  21. }
  22. <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

Code: [全选] [展开/收缩] [Download] (example.pl)

  1. MoveFileW(
  2.     encode(‘utf-16le‘, decode(‘utf8‘,$F))."\x00",
  3.     encode(‘utf-16le‘, decode(‘utf8‘,$newname))."\x00"
  4.     );

获取文件的日期信息:

      普通文件名的情况

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中的方法
      效率要高得多

Code: [全选] [展开/收缩] [Download] (example.pl)

  1. use utf8;
  2. use Encode;
  3. use Win32;
  4.  
  5. $filename=‘D:\测试目录\董贞 ? 01.剑如虹.[贞江湖].mp3‘;
  6. $filename=Win32::GetShortPathName($filename);
  7.  
  8. my $mtime = (stat $filename)[9];
  9. my($sec,$min,$hour,$mday,$mon,$year,$wday,$yday,$isdst) = localtime($mtime);
  10. $year+=1900;
  11. $mon+=1;
  12. print "$year-$mon-$mday\n";
  13. <STDIN>;

原文地址:https://www.cnblogs.com/catgatp/p/8443294.html

时间: 2024-11-05 20:41:55

Windows 系统 Unicode 文件名操作(新建、重命名、枚举、复制)全攻略的相关文章

delphi 文件的操作:重命名、复制、移动、删除

Delphi 文件的操作:重命名.复制.移动.删除第一种方法: RenameFile('Oldname', 'Newname'); CopyFile(PChar('Oldname'), PChar('Newname'), False); MoveFile(PChar('Oldname'), PChar('Newname')); DeleteFile(文件名); 第二种方法: 在Delphi可以使用ShellApi单元中的函数SHFileOperation来实现消息上传控件的制作, SHFileO

linux下的文件操作——批量重命名

概述:在日常工作中,我们经常需要对一批文件进行重命名操作,例如将所有的jpg文件改成bnp,将名字中的1改成one,等等.文本主要为你讲解如何实现这些操作 1.删除所有的 .bak 后缀: rename 's/\.bak$//' *.bak 注意,这个命令的格式组织如下:s/   \.bark$   / / 是s/para1/para2/ 这个有点想sed的语法,将para1匹配的字符串换成para2 2.把 .jpe 文件后缀修改为 .jpg: rename 's/\.jpe$/\.jpg/'

linux 下文件重命名/移动/复制命令(转)

linux下重命名文件:使用mv命令就可以了, 例:要把名为:abc   重命名为:123 可以这样操作: 重命名:MV命令 1.进入你的文件目录,运行命令:   /cd  dir  (dir指的是你文件所在目录) 2.然后命令:ls (例出你的所有文件夹),我们可以看到有  abc 这个文件了 3.最后命令:mv abc 123 但是要注意的是,如果当前目录下也有个123的文件的话,我们的这个文件是会将它覆盖的. 拷贝复制:CP命令 例:我要把 abc文件复制到 语法: cp [选项] /x/

Windows Socket五种I/O模型——代码全攻略(转)

Winsock 的I/O操作: 1. 两种I/O模式 阻塞模式:执行I/O操作完成前会一直进行等待,不会将控制权交给程序.套接字 默认为阻塞模式.可以通过多线程技术进行处理. 非阻塞模式:执行I/O操作时,Winsock函数会返回并交出控制权.这种模式使用 起来比较复杂,因为函数在没有运行完成就进行返回,会不断地返回 WSAEWOULDBLOCK错误.但功能强大.为了解决这个问题,提出了进行I/O操作的一些I/O模型,下面介绍最常见的三种: Windows Socket五种I/O模型——代码全攻

.Net常用技巧_VS2005[C#] 操作 Excel 全攻略(转)

using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Text; using System.Windows.Forms; using Microsoft.Office.Interop.Excel; using System.Data.SqlClient; using System.Data.

漏洞演示系统DVWA(Damn Vulnerable Web Application) V1.8渗透测试攻略

漏洞演示系统DVWA(Damn Vulnerable Web Application) V1.8攻略 测试环境: 操作系统:Windows 8.1 .Windows 7 运行时:.Net Framework 3.5 PHP+MySQL集成测试环境:XAMPP V3.2.1 首先,从http://www.dvwa.co.uk/ 下载DVWA,并将文件释放到c:\xampp\htdocs\DVWA 修改配置文件config\config.inc.php ,设置数据库连接账号及默认的安全级别: $_D

webBrowser中操作网页元素全攻略

webBrowser中操作网页元素全攻略 2012-12-20 14:21 188人阅读 评论(0) 收藏 举报 1.获取非input控件的值: webBrowser1.Document.All["控件ID"].InnerText; 或webBrowser1.Document.GetElementById("控件ID").InnerText; 或webBrowser1.Document.GetElementById("控件ID").GetAttr

javascript 操作 excel 全攻略

最近做一个项目,用到了javascript操纵excel以生成报表,下面是标有详细注解的实例 <html> <head><script language="javascript" type="text/javascript"> </script><script language="javascript" type="text/javascript">function

Perl Unicode全攻略

Perl Unicode全攻略 耐心看完本文,相信你今后在unicode处理上不会再有什么问题. 本文内容适用于perl 5.8及其以上版本. perl internal form 在Perl看来, 字符串只有两种形式. 一种是octets, 即8位序列, 也就是我们通常说的字节数组. 另一种utf8编码的字符串, perl管它叫string. 也就是说: Perl只认识两种编码: Ascii(octets)和utf8(string). utf8 flag 那么perl如何确定一个字符串是oct