Record a weird bug encountered in a project:
- This is a Java project packaged as an executable JAR that executes periodically on a Linux server with a cronjob.
- Each time, the system scans the directory specified by the server. If there are files in the directory, the system processes the files. Otherwise, the system exits
Here is the sample code, which is very simple:
public class App {
private static final String FOLDER = "/tmp/work";
public static void main(String[] args) {
File folder = new File(FOLDER);
File[] files = folder.listFiles();
if(files ! =null && files.length > 0) {
for(File file : files) { handleFile(file); }}}}Copy the code
One day a customer reported that the program was not importing files properly.
Once you get the log and the failed import file, start debugging:
-
First, the error message in the program log is FileNotFoundException, which means that when the program calls the handleFile method to read the File object passed in, the File is not found.
-
However, this file must exist. Firstly, the client can see it through LS-L, and secondly, the program can obtain it through Folder.listfiles.
-
After setting up the same test environment locally as the client environment, logging the application, placing the file in the directory, file.exists() returns false
Why does file.exists() return false when a file exists and can be listed?
- After looking at the file, it was found that there were Chinese characters and Chinese parentheses in the file name. It was presumed that there was a problem with character encoding. If you changed the file name to English, it could be imported normally.
However, the customer said that the same file can be imported after restarting the server without changing the startup script.
Where to reason…
-
Next, ask the customer to list the system environment variables when they can be imported, and then list the system environment variables when they cannot be imported
-
The locale of the system changed when the file could not be imported…
The problem was resolved. This also explains why restarting the server solved the problem. Restarting the server resets the default locale (UTF-8), and someone must have changed the system locale before the problem occurred…
The most confusing part of this is that when we scan the files under a folder for processing, we usually think that the files in the file list returned by folder.listfiles () are “existing”, but this bug shows that this is not necessarily the case
- First, folder must exist first, otherwise Folder.listfiles () will fail
- Second, the file must also be real, otherwise folder.listfiles () won’t list the file
- Again, folder.listfiles () returns only the file object corresponding to the abstract path in this folder, which consists of the folder path plus the name of each file
- When you actually operate on each file object, the system will search the local disk based on the abstract path, but if your system does not recognize special characters in the abstract path, such as Chinese and Chinese parentheses in this example, The system considers that the file corresponding to the abstract path (containing special characters) does not exist, and you get a FileNotFoundException
In addition, the comment to the file.exists() method reads:
/**
* Tests whether the file or directory denoted by this abstract pathname
* exists.
*
* @return <code>true</code> if and only if the file or directory denoted
* by this abstract pathname exists; <code>false</code> otherwise
*
* @throws SecurityException
* If a security manager exists and its <code>{@link
* java.lang.SecurityManager#checkRead(java.lang.String)}</code>
* method denies read access to the file or directory
*/
public boolean exists(a) {
Copy the code
Tests whether the file or directory denoted by this abstract pathname exists.
The key to this statement is that Tests will not know if the actual file represented by the file object exists until the file object is actually tested.
Reference links:
- Stackoverflow.com/questions/1…
- Bugs.java.com/bugdatabase…