Apr 09, 2018 Accessing Files and Directories. Before you can open a file, you first have to locate it in the file system. The system frameworks provide many routines for obtaining references to many well-known directories, such as the Library directory and its contents. You can also specify locations manually by building a URL or string-based path from known directory names. For example, lets say you have a directory full of archives, containing sub-directories and files. Deleting each item individually from the Finder or the command line can take a long time.
Whether you need to do so for yourself or for someone else, there is sometimes a need to create a listing of the contents of a particular directory, set of directories, or a whole drive. With the size of hard drives today reaching multiple Terabytes, keeping track of all the stored files would be quite useful. It’s quite easy to find out what files are taking up the most drive space but it’s also handy to know what files are stored where.
There are very simple ways of getting a list of files in a folder without external software. One option is using the Windows command “Dir [folder] > File.txt” from the Command Prompt. However, messing around with console commands is not to everyone’s liking and Dir is also quite limited. Here’s a selection of 10 free tools to save or print out content lists of the files inside your folders including to plain text file, an Excel spreadsheet, or an HTML document. All tools were tested in Windows 10 and 7.
1. FileListCreator
What makes FileListCreator a potentially very useful program is the number of available options in specific areas. It has several different file formats you can save to, including text file, HTML page, comma separated value file (CSV), Excel spreadsheet, and even image files (PNG, JPG, or BMP). FileListCreator has a portable version, is multi-platform, and free to use (donations are welcomed by the author).
You can add multiple files and folders from different directories or drives by simply using the buttons or dropping them onto the window. The lower pane in the interface is a live preview of the output format which you can copy to the clipboard or save. There’s a huge amount of columns you can add to the list including specific columns for images, audio files, video files, text files, documents, attributes, and checksums. Limited output format options such as fonts and colors are found in the Settings.
Download FileListCreator
2. Snap2HTML
Snap2HTML is a portable tool that is able to generate a single HTML file that lists all folders and files from a predefined root folder. It’s the one tool in this list we actually use ourselves to make content lists of external drives. What makes Snap2HTML so useful is it creates the most functional and visually appealing HTML file from any tool we’ve seen. It’s very easy to navigate through the list of files and folders in your browser.
The configuration options are few with tick boxes to include hidden or system files. The Link files option turns file entries into clickable links so you can execute a file directly from the HTML page. Drives or folders are searched recursively. The HTML page looks and behaves a lot like Windows Explorer complete with an expandable tree view down the left and a search option. You can edit the template.html file yourself if you know some CSS. Snap2HTML has command line options for use with Command Prompt and batch files.
Download Snap2HTML
3. DirLister v2
DirLister is a tool that we have talked about before. The original from back in 2005 is available on the developer’s website which doesn’t have a .NET 4.6 requirement. The newer version 2 currently in beta is portable and more feature rich than the old version. One improvement is the ability to add multiple drives or folders to the list at once. Use the “Select folder” button or simply drop a folder onto the window.
In addition to HTML and plain text files, you have the option to save as a CSV file, an XML file, a JSON file, or a markdown file. Multiple output formats can be selected at once. There’s also a filter where you can add normal wildcards (like *.jpg) or Regex if you know how to use it. Optional Explorer integration makes it easy to create a file list from the right click context menu. The HTML page DirLister v2 creates is good and probably only second to Snap2HTML.
Download DirLister v2
4. Arclab Dir2HTML
Dir2HTML is a tool that used to be freeware but Arclab took the program full shareware back in 2013/2014. However, we found the last freeware version from 2011 and look at that here as it still works fine in Windows 10. While it might not have as many features as the latest paid version, the old free version of Dir2HTML has a number of useful options to aid in the creation and customization of the HTML index file.
It supports recursive folder indexing, a simple file mask, creating a sub-section for each subfolder, excluding hidden files and folders, including size and date, and many other options. The right side of the window allows you to tailor the output HTML file visually and you can add a customized title, alter the font sizes, colors, and resize different areas of the output. Many of the options can be left alone if the standard layout is acceptable.
Download Arclab Dir2HTML
5. Karen’s Directory Printer
Directory Printer has been a popular tool for several years for printing and saving file and folder lists. This is because there’s a lot of options you can configure to get the output to your liking. Files have 15 pieces of information that can be displayed including names, dates, sizes, attributes, and checksums. Also included is a file mask filter that already has five format presets built-in for images, music, documents, etc.
There are various save options for hidden and system files, sorting, and what to include/exclude. The separate Print and “Save to Disk” tabs have a few different options to allow you to set different output formatting. There’s a right click context menu function in the Settings tab. Sadly, the only truly useful output file format is a text file. You can extract the setup installer with 7-Zip to create a portable program.
Download Karen’s Directory Printer
12Next › View AllAny ideas of Android version of this kind of thing?
ReplyI’ve been looking for years. No. No Android app. There should be! The only solution is to use dir2html on a Windows machine and connect it to the phone or tablet.
ReplyIs there a program that includes all possible fields? Length for example?
ReplyThe Dir List and Print (#4 in the list above) prints a lot of fields in the free version, with a more complete list of properties which are available only in the purchased version
ReplyDirectory List & Print is the best but the free version is limited: No filters.
Karen’s is ok but lacks of preview and looks old.
FileListCreator is better and free as well.
The comments; including your own, say there is supposed to be a utility here called “DriveZ”…
So where is it?
If you look carefully you’ll see some nice black buttons that take you to page 2 where DriveZ is….;)
ReplyKRK’s DirLister (2.37) free edition only allows HTML saves.
Reply“Dir2Html is a tool which is free for personal and commercial use”
How? When I use it it changes every second entry to “TEST-VERSION” unless I pay.
How is that free???
ReplyBecause when the article was written, Dir2HTML was freeware, you can even see the title bar in the screenshot say “freeware”. Arclab have since taken the program full shareware.
I’ve tracked down the last freeware version and we now link to that instead.
ReplyDriveZ is so simple and exactly what I was looking for. You don’t need to edit the output again to, say, delete the extensions, file sizes or information etc..
Only the filenames and it’s directly on the clipboard. Just ready to use in one click. Just have a try if you’re looking for such a simple thing..
(Others are not bad too (JR DirPrinter, Dirlister) but you need to clean information etc.. when compared)
Important: When I downloaded DriveZ using the link here, it asked for password for the zip file and it was looking a bit suspicious (v. 1044). So I searched and downloaded from somewhere else (v. 1.0.19)
ReplyThe DriveZ download is hosted on our own server. We have to password selected files like DriveZ because they produce false positive alerts in some antivirus software and online services like Virustotal. If we don’t do that, just a few files can cause our entire server to be marked by an antivirus vendor as unsafe or malicious.
If you upload your 1.0.19 file to Virustotal, you’ll see that it is classed as malicious by at least one AV service.
ReplyThank you.
So what’s the password for the downloaded zip files? ( I’d tried blank (enter) but didn’t work.. )
ReplyThe password is listed on the download page.
ReplyKaren’s Directory Printer let’s you save output as a tsv file (tab separated values) which can be read by Excel
ReplyI have always like Directory Lister v0.9.1 by krksoft.com. There are portable versions online BTW. It looks like there is a newer paid/shareware version but v0.9.1 as of 2006 works fine on Windows 7 however I have not tried any newer version of windows.
It has all the options I need to configure the output and the output can be previewed, modified, previewed and saved or copied to clipboard in text, html or csv formats. It does everything I require, nothing more or less.
ReplyI’m using YourDir, which doesn’t have so many fancy options like the above, but is really easy and saves to either CSV or creates an image file, which I’m using a lot.
ReplyThere is another way, open elevated cmd and give the command dir > filelist.txt
Open the file filelist.txt with notepad and print it :D
You can use and dir/s > filelist.txt command also :)
ReplyAnd you can open the file in word when you are done :)
Thanks for the refresher… I miss the DOS days…
ReplyWill any of these suggestions work with Win 10?
ReplyDirectory List and Print is the only one for me. THE single clincher is its ability to output to Word. I’ve used it to produce lists, tables of contents, indices of hundreds of separate music, image, video files and folders which would have taken years to produce manually and which can be printed. And once in Word, editing is easy – using find and replace to produce a style uniformity. It makes the rest look amateur.
ReplyThanks for the helpful reviews. @ A King: Nobody is interested in your self-serving propaganda. A PC user has the right to control their own machine, including software made freely available for it,
ReplySnap2HTML is the BEST! It does exactly what I was looking for and I spent quite a bit of time searching! So happy I found it!
ReplyDir2Html is no longer free. :(
ReplyI agree with anyone who says that Snap2HTML is the best! the most important thing about it is that it’s FREE so you don’t have to waste your time for finding a working keygen or patch unlike Dir2HTML(spent 1 hour trying to find a working keygen but nothing) and it has a file explorer alike GUI so you won’t get lost easily while navigating huge database
ReplyStealing software is a crime
ReplySo informative ¬¬
ReplyThis was e-learning for me. Learnt something new
ReplyI’ve tested all of them and the Snap2HTML was the best for me. Its design is exactly what I was needing.
The DIR2HTML and ArcLab are good options too but for what I needed, the Snap2HTML is the first one.
DirLister works in a test install, but unlike stated in Raymond’s mini-review (thank you) I was unable to get a listing that included folder names without an unwanted recursive listing. Maybe folder names do list with recursion, but I didn’t have time to try that, and it still wouldn’t help me. Also file sizes is one of several future features not implemented after years of version 1.0, but the author said he would have to need these features before implementing them.
ReplyI have used SNAP2HTML.
It is indeed a great tool and creates HTML file of the directory in a whisker and handy for the web designers. It has built-in option to search the directory too.
I used it for creating a directory containing nearly 4 TB of Files in thousands of sub-directories. It is simply an awesome tool with clean interface and easy to use.
Special thanks to the Mr. Dan, the developer of SNAP2HTML.
Great app,
thanks to the Mr. Dan, the developer and you too,
and this article. <3
ReplyThe one I’ve been using for years is Rep-Listing 3.0 (Jean-Pierre Aigron, 2004) ; apart from a few minor bugs and quirks (incorrect file sizes above 2GB, annoying question asked every time the program is opened about « margins set outside the printable area ») it’s complete, fast and efficient. The listings are clear and include the file sizes in bytes written as continuous numbers (i.e. with no spaces, for instance a VOB file will be : 1073709056), which was the feature I was looking for back then. I wanted to have a list of the contents for each one of my hundreds of archived CD/DVDs, with the accurate file sizes, so that I could quickly verify if a file I was about to download was already there somewhere, only by typing its size in Windows XP’s search panel (dearly missed on later iterations), searching inside the directory containing all the listings for a given category of content (then if I wanted to be sure I could take the corresponding disc from the pile and look at the file, or calculate its CRC). Now I no longer archive files on CD/DVDs (neither reliable nor practical) but it’s still useful to keep tracks of things.
ReplyThanks a ton man………….it works like anything…..great work
ReplyThank you so much!
ReplyThe best one is missing :
mythicsoft.com/agentransack/Page.aspx?page=download
Replyfinally… free from DOS’ dir command, excellent tools!
ReplyBrilliant!!.. Just what Ive been looking for, Thanks!
ReplyThaks a ton man………….it works like anything…..great work……………
Replynice tools this is great! thanks Raymond.
ReplyBefore you can open a file, you first have to locate it in the file system. The system frameworks provide many routines for obtaining references to many well-known directories, such as the Library
directory and its contents. You can also specify locations manually by building a URL or string-based path from known directory names.
When you know the location of a file, you can then start planning the best way to access it. Depending on the type of file, you may have several options. For known file types, you typically use built-in system routines to read or write the file contents and give you an object that you can use. For custom file types, you may need to read the raw file data yourself.
Although you can open any file and read its contents as a stream of bytes, doing so is not always the right choice. macOS and iOS provide built-in support that makes opening many types of standard file formats (such as text files, images, sounds, and property lists) much easier. For these standard file formats, use the higher-level options for reading and writing the file contents. Table 2-1 lists the common file types supported by the system along with information about how you access them.
File Type | Examples | Description |
---|---|---|
Resource files | Nib files Image files Sound files Strings files Localized resources | Apps use resource files to store data that is independent of the code that uses it. Resource files are commonly used to store localizable content such as strings and images. The process for reading data from a resource file depends on the resource type. For information about how to read the contents of resource files, see Resource Programming Guide. |
Text files | Plain text file UTF-8 formatted file UTF-16 formatted file | A text file is an unstructured sequence of ASCII or Unicode characters. You typically load the contents of a text file into an For information about using the |
Structured data files | XML file Property list file Preference file | A structured data file usually consists of string-based data arranged using a set of special characters. For information about parsing XML, see Event-Driven XML Programming Guide. |
Archive files | Binary files created using a keyed archiver object | An archive is a file format used to store a persistent version of your app’s runtime objects. An archiver object encodes the state of the objects into a stream of bytes that can be written to disk all at once. An unarchiver reverses the process, using the stream of bytes to re-create the objects and restore them to their previous state. Archives are often a convenient alternative to implementing custom binary file formats for your documents or other data files. For information on how to create and read archive files, see Archives and Serializations Programming Guide. |
File packages | Custom document formats | A file package is a directory that contains any number of custom data files but which is presented to the user as if it were a single file. Apps can use file packages to implement complex file formats that contain multiple distinct files, or a mixture of different types of files. For example, you might use a file package if your file format includes both a binary data file and one or more image, video, or audio files. You access the contents of a file package using |
Bundles | Apps Plug-ins Frameworks | Bundles provide a structured environment for storing code and the resources used by that code. Most of the time, you do not work with the bundle itself but with its contents. However, you can locate bundles and obtain information about them as needed. For information about bundles and how you access them, see Bundle Programming Guide |
Code files | Binary code resources Dynamic shared libraries | Apps that work with plug-ins and shared libraries need to be able to load the associated code for that item to take advantage of its functionality. For information about how to load code resources into memory, see Code Loading Programming Topics. |
In situations where the standard file formats are insufficient, you can always create your own custom file formats. When reading and writing the content of custom files, you read and write data as a stream of bytes and apply those bytes to your app’s file-related data structures. You have complete control over how you read and write the bytes and how you manage your file-related data structures. For more information about the techniques for reading and writing files that use custom file formats, see Techniques for Reading and Writing Files Without File Coordinators.
The preferred way to specify the location of a file or directory is to use the NSURL
class. Although the NSString
class has many methods related to path creation, URLs offer a more robust way to locate files and directories. For apps that also work with network resources, URLs also mean that you can use one type of object to manage items located on a local file system or on a network server.
Note: In addition to NSURL
, you can also use the CFURLRef
opaque type to manipulate paths as URLs. The NSURL
class is toll-free bridged with the CFURLRef
type, which means you can use them interchangeably in your code. For more information about how to create and manipulate URLs using Core Foundation, see CFURL Reference.
For most URLs, you build the URL by concatenating directory and file names together using the appropriate NSURL
methods until you have the path to the item. A URL built in that way is referred to as a path-based URL because it stores the names needed to traverse the directory hierarchy to locate the item. (You also build string-based paths by concatenating directory and file-names together, with the results stored in a slightly different format than that used by the NSURL
class.) In addition to path-based URLs, you can also create a file reference URL, which identifies the location of the file or directory using a unique ID.
All of the following entries are valid references to a file called MyFile.txt
in a user’s Documents
directory:
Path-based URL:file://localhost/Users/steve/Documents/MyFile.txt
File reference URL:file:///.file/id=6571367.2773272/
String-based path:/Users/steve/Documents/MyFile.txt
You create URL objects using the NSURL
methods and convert them to file reference URLs only when needed. Path-based URLs are easier to manipulate, easier to debug, and are generally preferred by classes such as NSFileManager
. An advantage of file reference URLs is that they are less fragile than path-based URLs while your app is running. If the user moves a file in the Finder, any path-based URLs that refer to the file immediately become invalid and must be updated to the new path. However, as long as the file moved to another location on the same disk, its unique ID does not change and any file reference URLs remain valid.
Important: Although they are safe to use while your app is running, file reference URLs are not safe to store and reuse between launches of your app because a file’s ID may change if the system is rebooted. If you want to store the location of a file persistently between launches of your app, create a bookmark as described in Locating Files Using Bookmarks.
Of course, there are still times when you might need to use strings to refer to a file. Fortunately, the NSURL
class provides methods to convert path-based URLs to and from NSString
objects. You might use a string-based path when presenting that path to the user or when calling a system routine that accepts strings instead of URLs. The conversion between NSURL
objects and NSString
objects is done using the NSURL class’s method absoluteString
.
Because NSURL
and NSString
describe only the location of a file or directory, you can create them before the actual file or directory exists. Neither class attempts to validate the actual existence of the file or directory you specify. In fact, you must create the path to a nonexistent file or directory before you can create it on disk.
If you have an NSURL
object that refers to an actual file or directory on disk, and you want to get the user visible name of the volume on which it resides, you use the method getResourceValue:forKey:error:
in a two step process:
Use the NSURLVolumeURLKey
constant to obtain the volume URL from the resource URL.
Use the NSURLLocalizedNameKey
constant to obtain the user visible volume name from the volume URL.
Use the NSURLLocalizedNameKey
constant rather than the NSURLNameKey
constant. Using NSURLNameKey
returns the name of the volume in the file system, which may not be the same as the user visible name. Listing 2-1 demonstrates this process for the current user’s home directory.
Listing 2-1 Obtaining the user visible volume name for a resource URL
For more information about the methods you use to create and manipulate URLs and strings, see NSURL Class Reference and NSString Class Reference.
Before you can access a file or directory, you need to know its location. There are several ways to locate files and directories:
Find the file yourself.
Ask the user to specify a location.
Locating files in the standard system directories, in both sandboxed and non-sandboxed apps.
Using bookmarks.
The file systems of iOS and macOS impose specific guidelines on where you should place files, so most of the items your app creates or uses should be stored in a well-known place. Both systems provide interfaces for locating items in such well-known places, and your app can use these interfaces to locate items and build paths to specific files. An app should prompt the user to specify the location of a file or directory only in a limited number of situations that are described in Using the Open and Save Panels.
In macOS, user interactions with the file system should always be through the standard Open and Save panels. Because these panels involve interrupting the user, use them only in a limited number of situations:
To open user documents
To ask the user where to save a new document
To associate user files (or directories of files) with an open window
Never use the Open and Save panels to access any files that your app created and uses internally. Support files, caches, and app-generated data files should be placed in one of the standard directories dedicated to app-specific files.
For information on how to present and customize the Open and Save panels, see Using the Open and Save Panels.
Apps that need to locate resource files inside their bundle directory (or inside another known bundle) must do so using an NSBundle
object. Bundles eliminate the need for your app to remember the location of individual files by organizing those files in a specific way. The methods of the NSBundle
class understand that organization and use it to locate your app’s resources on demand. The advantage of this technique is that you can generally rearrange the contents of your bundle without rewriting the code you use to access it. Bundles also take advantage of the current user’s language settings to locate an appropriately localized version of a resource file.
The following code shows how to retrieve a URL object for an image named MyImage.png
that is located in the app’s main bundle. This code determines only the location of the file; it does not open the file. You would pass the returned URL to a method of the NSImage
class to load the image from disk so that you could use it.
For more information about bundles, including how to locate items in a bundle, see Bundle Programming Guide. For specific information about loading and using resources in your app, see Resource Programming Guide.
When you need to locate a file in one of the standard directories, use the system frameworks to locate the directory first and then use the resulting URL to build a path to the file. The Foundation framework includes several options for locating the standard system directories. By using these methods, the paths will be correct whether your app is sandboxed or not:
The URLsForDirectory:inDomains:
method of the NSFileManager
class returns a directory’s location packaged in an NSURL
object. The directory to search for is an NSSearchPathDirectory
constant. These constants provide URLs for the user’s home directory, as well as most of the standard directories.
The NSSearchPathForDirectoriesInDomains
function behaves like the URLsForDirectory:inDomains:
method but returns the directory’s location as a string-based path. Use the URLsForDirectory:inDomains:
method instead.
The NSHomeDirectory
function returns the path to either the user’s or app’s home directory. (Which home directory is returned depends on the platform and whether the app is in a sandbox.) When an app is sandboxed the home directory points to the app’s sandbox, otherwise it points to the User’s home directory on the file system. If constructing a file to a subdirectory of a user’s home directory, consider using the URLsForDirectory:inDomains:
method instead.
You can use the URL or path-based string you receive from the preceding routines to build new objects with the locations of the files you want. Both the NSURL
and NSString
classes provide path-related methods for adding and removing path components and making changes to the path in general. Listing 2-2 shows an example that searches for the standard Application Support
directory and creates a new URL for a directory containing the app’s data files.
Listing 2-2 Creating a URL for an item in the app support directory
If you want to save the location of a file persistently, use the bookmark capabilities of NSURL
. A bookmark is an opaque data structure, enclosed in an NSData
object, that describes the location of a file. Whereas path- and file reference URLs are potentially fragile between launches of your app, a bookmark can usually be used to re-create a URL to a file even in cases where the file was moved or renamed.
To create a bookmark for an existing URL, use the bookmarkDataWithOptions:includingResourceValuesForKeys:relativeToURL:error:
method of NSURL
. Specifying the NSURLBookmarkCreationSuitableForBookmarkFile
option creates an NSData
object suitable for saving to disk. Listing 2-3 shows a simple example implementation that uses this method to create a bookmark data object.
Listing 2-3 Converting a URL to a persistent form
If you write the persistent bookmark data to disk using the writeBookmarkData:toURL:options:error:
method of NSURL
, what the system creates on disk is an alias file. Aliases are similar to symbolic links but are implemented differently. Normally, users create aliases from the Finder when they want to create links to files elsewhere on the system.
To transform a bookmark data object back into a URL, use the URLByResolvingBookmarkData:options:relativeToURL:bookmarkDataIsStale:error:
method of NSURL
. Listing 2-4 shows the process for converting a bookmark back into a URL.
Listing 2-4 Returning a persistent bookmark to its URL form
The Core Foundation framework provides a set of C-based functions that parallel the bookmark interface provided by NSURL
. For more information about using those functions, see CFURL Reference.
To discover the contents of a given directory, you enumerate that directory. Cocoa supports enumerating a directory one file at a time or all at once. Regardless of which option you choose, enumerate directories sparingly. Each enumeration can touch a large number of files and subdirectories. This quickly becomes expensive.
Enumerating a directory one file at a time is recommended when you want to search for a specific file and stop enumerating when you find it. File-by-file enumeration uses the NSDirectoryEnumerator
class, which defines the methods for retrieving items. Because NSDirectoryEnumerator
itself is an abstract class, however, you do not create instances of it directly. Instead, you use either the enumeratorAtURL:includingPropertiesForKeys:options:errorHandler:
or the enumeratorAtPath:
method of NSFileManager
object to obtain a concrete instance of the class for use in your enumeration.
Enumerator objects return the paths of all files and directories contained within the enumerated directory. Because enumerations are recursive and cross device boundaries, the number of files and directories returned may be more than what is present in the starting directory. You can skip the contents of any directory you are not interested in by calling the enumerator object’s skipDescendents
method. Enumerator objects do not resolve symbolic links or attempt to traverse symbolic links that point to directories.
Listing 2-5 shows how to use the enumeratorAtURL:includingPropertiesForKeys:options:errorHandler:
method to list all the user-visible subdirectories of a given directory, noting whether they are directories or file packages. The keys
array tells the enumerator object to prefetch and cache information for each item. Prefetching this information improves efficiency by touching the disk only once. The options argument specifies that the enumeration should not list the contents of file packages and hidden files. The error handler is a block object that returns a Boolean value. If the block returns YES
, the enumeration continues after the error; if it returns NO
, the enumeration stops.
Listing 2-5 Enumerating the contents of a directory
You can use other methods declared by NSDirectoryEnumerator
to determine attributes of files during the enumeration—both of the parent directory and the current file or directory—and to control recursion into subdirectories. The code in Listing 2-6 enumerates the contents of a directory and lists files that have been modified within the last 24 hours; if, however, it comes across RTFD file packages, it skips recursion into them.
Listing 2-6 Looking for files that have been modified recently
If you know that you want to look at every item in a directory, you can retrieve a snapshot of the items and iterate over them at your convenience. Retrieving the contents of a directory in a batch operation is not the most efficient way to enumerate a directory, because the file manager must walk the entire directory contents every time. However, if you plan to look at all the items anyway, it is a much simpler way to retrieve the items.
There are two options for doing a batch enumeration of a directory using NSFileManager
:
To perform a shallow enumeration, use the contentsOfDirectoryAtURL:includingPropertiesForKeys:options:error:
or contentsOfDirectoryAtPath:error:
method.
To perform a deep enumeration and return only subdirectories, use the subpathsOfDirectoryAtPath:error:
method.
Listing 2-7 shows an example that uses the contentsOfDirectoryAtURL:includingPropertiesForKeys:options:error:
method to enumerate the contents of a directory. One of the benefits of using URLs is that you can also efficiently retrieve additional information about each item. This example retrieves the localized name, creation date, and type information for each item in the directory and stores that information in the corresponding NSURL
object. When the method returns, you can proceed to iterate over the items in the array
variable and do what you need with them.
Listing 2-7 Retrieving the list of items in a directory all at once
The Library
directory is the designated repository for files your app creates and manages on behalf of the user. Consider that these directories may be in different locations if your app is sandboxed. As a result, always use the NSFileManager
method URLsForDirectory:inDomains:
with the NSUserDomainMask
as the domain to locate the specific directory that you should use to store this data.
Use the Application Support
directory constant NSApplicationSupportDirectory
, appending your <bundle_ID> for:
Resource and data files that your app creates and manages for the user. You might use this directory to store app state information, computed or downloaded data, or even user created data that you manage on behalf of the user.
Autosave files.
Use the Caches
directory constant NSCachesDirectory
, appending your <bundle_ID> directory for cached data files or any files that your app can re-create easily.
Read and write preferences using the NSUserDefaults
class. This class automatically writes preferences to the appropriate location.
Use the NSFileManager
method URLsForDirectory:inDomains:
to get the directory for storing temporary files. Temporary files are files you intend to use immediately for some ongoing operation but then plan to discard later. Delete temporary files as soon as you are done with them.
Important: If you do not delete temporary files after three days, the system may delete them for you whether you are using them or not.
Note: Apps rarely share the files stored in the Application Support
, Caches
or Temp
directories. Therefore, you don’t typically need to use a file coordinator (NSFileCoordinator
) when reading or writing to these locations. For more information about file coordination, see The Role of File Coordinators and Presenters.
Because the file system is a resource shared by all processes, including system processes, use it carefully. Even on systems with solid-state disk drives, file operations tend to be a little slower because of the latency involved in retrieving data from the disk. And when you do access files, it is important that you do so in a way that is secure and does not interfere with other processes.
Operations involving the file system should be performed only when they are absolutely necessary. Accessing the file system usually takes a lot of time relative to other computer-wide tasks. So make sure you really need to access the disk before you do. Specifically:
Write data to disk only when you have something valuable to save. The definition of what is valuable is different for each app but should generally involve information that the user provides explicitly. For example, if your app creates some default data structures at launch time, do not save those structures to disk unless the user changes them.
Read data from disk only when you need it. In other words, load data that you need for your user interface now but do not load an entire file just to get one small piece of data from that file. For custom file formats, use file mapping or read only the few chunks of a file that you need to present your user interface. Read any remaining chunks as needed in response to user interactions with the data. For structured data formats, use Core Data to manage and optimize the reading of data for you.
There are several principles you can follow to help ensure that you do not have file-based security vulnerabilities in your program:
Check the result codes for any functions or methods you call. Result codes are there to let you know that there is a problem, so do pay attention to them. For example, if you try to delete a directory that you think is empty and get an error code, you might find that the directory is not empty after all.
When working in a directory to which your process does not have exclusive access, you must check to make sure a file does not exist before you create it. You must also verify that the file you intend to read from or write to is the same file you created.
Whenever possible use routines that operate on file descriptors rather than pathnames. In this way you can be sure you’re always dealing with the same file.
Intentionally create files as a separate step from opening them so that you can verify that you are opening a file you created rather than one that already existed
Know whether the function you are calling follows symbolic links. For example, the lstat
function gives you the status of a file regardless of whether it’s a normal file or a symbolic link, but the stat
function follows symbolic links and, if the specified file was a symbolic link, returns the status of the linked-to file. Therefore, if you use the stat
function, you might be accessing a different file than you expected.
Before you read a file, make sure that file has the owner and permissions you expect. Be prepared to fail gracefully (rather than hanging) if it does not.
Set your process’ file code creation mask (umask) to restrict access to files created by your process. The umask is a bitmask that alters the default permissions of a new file. You do not reset the umask for your app, your process inherits the umask from its parent process. For more information about setting the umask, see the umask(2)
man page.
For additional tips and coding practices, see Race Conditions and Secure File Operations in Secure Coding Guide.
When searching or comparing filenames, always assume that the underlying file system is case sensitive. macOS supports many file systems that use case to differentiate between files. Even on file systems (such as APFS and HFS+) that support case insensitivity, there are still times when case may be used to compare filenames. For example, the NSBundle
class and CFBundle APIs consider case when searching bundle directories for named resources.
All files should have a filename extension that reflects the type of content contained in the file. Filename extensions help the system determine how to open files and also make it easier to exchange files with other computers or other platforms. For example, network transfer programs often use filename extensions to determine the best way to to transfer files between computers.
Whenever you need to present the name of a file or directory in your user interface, always use the item’s display name. Using the display name ensures that what your app presents matches what the Finder and other apps are presenting. For example, if your app shows the path to a file, using the display name ensures that the path is localized in accordance with the current user’s language settings.
For more information about display names, see Files and Directories Can Have Alternate Names.
In general, the objects and functions used to access and manipulate files can be used from any thread of your app. However, as with all threaded programming, make sure you manipulate files safely from your background threads:
Avoid using the same file descriptor from multiple threads. Doing so could cause each thread to disrupt the state of the other.
Always use file coordinators to access files. File coordinators provide notifications when other threads in your program (or other processes) access the same file. You can use these notifications to clean up your local data structures or perform other relevant tasks.
Create a unique instance of NSFileManager
for each thread when copying, moving, linking, or deleting files. Although most file manager operations are thread-safe, operations involving a delegate require a unique instance of NSFileManager
, especially if you are performing those operations on background threads.
Copyright © 2018 Apple Inc. All Rights Reserved. Terms of Use | Privacy Policy | Updated: 2018-04-09