How to display long messages in logcat

AndroidLogcat

Android Problem Overview


I am trying to display long message on logcat. If the length of message is more than 1000 characters, it gets broken.

What is the mechanism to show all characters of long message in logcat?

Android Solutions


Solution 1 - Android

If logcat is capping the length at 1000 then you can split the string you want to log with String.subString() and log it in pieces. For example:

int maxLogSize = 1000;
for(int i = 0; i <= veryLongString.length() / maxLogSize; i++) {
    int start = i * maxLogSize;
    int end = (i+1) * maxLogSize;
    end = end > veryLongString.length() ? veryLongString.length() : end;
    Log.v(TAG, veryLongString.substring(start, end));
}

Solution 2 - Android

As a follow on to spatulamania answer I wrote a wrapper class which handles this for you. You just need to change the import and it will log everything

public class Log {

    public static void d(String TAG, String message) {
        int maxLogSize = 2000;
        for(int i = 0; i <= message.length() / maxLogSize; i++) {
            int start = i * maxLogSize;
            int end = (i+1) * maxLogSize;
            end = end > message.length() ? message.length() : end;
            android.util.Log.d(TAG, message.substring(start, end));
        }
    }

}

Solution 3 - Android

This builds on spatulamania's answer, is a little more succinct, and won't add an empty log message at the end:

final int chunkSize = 2048;
for (int i = 0; i < s.length(); i += chunkSize) {
    Log.d(TAG, s.substring(i, Math.min(s.length(), i + chunkSize)));
}

Solution 4 - Android

Try this piece of code to show long message in logcat.

public void logLargeString(String str) {
    if(str.length() > 3000) {
        Log.i(TAG, str.substring(0, 3000));
        logLargeString(str.substring(3000));
    } else {
        Log.i(TAG, str); // continuation
    }
}

Solution 5 - Android

This is how OkHttp with HttpLoggingInterceptor does it:

public void log(String message) {
  // Split by line, then ensure each line can fit into Log's maximum length.
  for (int i = 0, length = message.length(); i < length; i++) {
    int newline = message.indexOf('\n', i);
    newline = newline != -1 ? newline : length;
    do {
      int end = Math.min(newline, i + MAX_LOG_LENGTH);
      Log.d("OkHttp", message.substring(i, end));
      i = end;
    } while (i < newline);
  }
}

MAX_LOG_LENGTH is 4000.

Here it use Log.d (debug) and hardcoded "OkHttp" tag.

It split the log at newlines or when it reach the max length.

This class below is an helper class you can use (if you have lambda support throw Jack & Jill or retrolambda) to do the same thing OkHttp does on any log:

/**
 * Help printing logs splitting text on new line and creating multiple logs for too long texts
 */

public class LogHelper {

	private static final int MAX_LOG_LENGTH = 4000;

	public static void v(@NonNull String tag, @Nullable String message) {
	    log(message, line -> Log.v(tag, line));
	}

	public static void d(@NonNull String tag, @Nullable String message) {
	    log(message, line -> Log.d(tag, line));
	}

	public static void i(@NonNull String tag, @Nullable String message) {
	    log(message, line -> Log.i(tag, line));
	}

	public static void w(@NonNull String tag, @Nullable String message) {
	    log(message, line -> Log.w(tag, line));
	}

	public static void e(@NonNull String tag, @Nullable String message) {
	    log(message, line -> Log.e(tag, line));
	}

	public static void v(@NonNull String tag, @Nullable String message, @Nullable Throwable throwable) {
	    log(message, throwable, line -> Log.v(tag, line));
	}

	public static void d(@NonNull String tag, @Nullable String message, @Nullable Throwable throwable) {
	    log(message, throwable, line -> Log.d(tag, line));
	}

	public static void i(@NonNull String tag, @Nullable String message, @Nullable Throwable throwable) {
	    log(message, throwable, line -> Log.i(tag, line));
	}

	public static void w(@NonNull String tag, @Nullable String message, @Nullable Throwable throwable) {
	    log(message, throwable, line -> Log.w(tag, line));
	}

	public static void e(@NonNull String tag, @Nullable String message, @Nullable Throwable throwable) {
	    log(message, throwable, line -> Log.e(tag, line));
	}

	private static void log(@Nullable String message, @NonNull LogCB callback) {
	    if (message == null) {
	        callback.log("null");
	        return;
	    }
	    // Split by line, then ensure each line can fit into Log's maximum length.
	    for (int i = 0, length = message.length(); i < length; i++) {
	        int newline = message.indexOf('\n', i);
	        newline = newline != -1 ? newline : length;
	        do {
	            int end = Math.min(newline, i + MAX_LOG_LENGTH);
	            callback.log(message.substring(i, end));
	            i = end;
	        } while (i < newline);
	    }
	}

	private static void log(@Nullable String message, @Nullable Throwable throwable, @NonNull LogCB callback) {
	    if (throwable == null) {
	        log(message, callback);
	        return;
	    }
	    if (message != null) {
	        log(message + "\n" + Log.getStackTraceString(throwable), callback);
	    } else {
	        log(Log.getStackTraceString(throwable), callback);
	    }
	}

	private interface LogCB {
	    void log(@NonNull String message);
	}
}

Solution 6 - Android

In order not to minimize splitting lines across log messages, I take the large string and log each line separately.

void logMultilineString(String data) {
    for (String line : data.split("\n")) {
        logLargeString(line);
    }
}

void logLargeString(String data) {
    final int CHUNK_SIZE = 4076;  // Typical max logcat payload.
    int offset = 0;
    while (offset + CHUNK_SIZE <= data.length()) {
        Log.d(TAG, data.substring(offset, offset += CHUNK_SIZE));
    }
    if (offset < data.length()) {
        Log.d(TAG, data.substring(offset));
    }
}

Solution 7 - Android

With Kotlin we can make use of the stdlib chunked function:

fun logUnlimited(tag: String, string: String) {
    val maxLogSize = 1000
    string.chunked(maxLogSize).forEach { Log.v(tag, it) }
}

Solution 8 - Android

I consider Timber a good option for this issue. Timber automatically split and print chunks of message in logcat.

https://github.com/JakeWharton/timber

You can see log method implementation in timber.log.Timber.DebugTree static class.

Solution 9 - Android

Here is a Kotlin version for the @spatulamania answer (especially for lazy/smart peooples):

val maxLogSize = 1000
val stringLength = yourString.length
for (i in 0..stringLength / maxLogSize) {
    val start = i * maxLogSize
    var end = (i + 1) * maxLogSize
    end = if (end > yourString.length) yourString.length else end
    Log.v("YOURTAG", yourString.substring(start, end))
}

Solution 10 - Android

if print json string, can use code below

    @JvmStatic
    fun j(level: Int, tag: String? = null, msg: String) {
        if (debug) {
            if (TextUtils.isEmpty(msg)) {
                p(level, tag, msg)
            } else {
                val message: String
                message = try {
                    when {
                        msg.startsWith("{") -> {
                            val jsonObject = JSONObject(msg)
                            jsonObject.toString(4)
                        }
                        msg.startsWith("[") -> {
                            val jsonArray = JSONArray(msg)
                            jsonArray.toString(4)
                        }
                        else -> msg
                    }
                } catch (e: JSONException) {
                    e.printStackTrace()
                    msg
                }
                p(level, tag, "╔═══════════════════════════════════════════════════════════════════════════════════════", false)
                val lines = message.split(LINE_SEPARATOR.toRegex()).dropLastWhile { it.isEmpty() }.toTypedArray()
                for (line in lines) {
                    p(level, tag, "║ $line", false)
                }
                p(level, tag, "╚═══════════════════════════════════════════════════════════════════════════════════════", false)
            }
        }
    }

full code

> CXLogUtil.j("json-tag","{}")

preview result

Solution 11 - Android

For an easy solution, use Use soft wrap option in below attach point no 4 options might help you.

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionVasuView Question on Stackoverflow
Solution 1 - AndroidspatulamaniaView Answer on Stackoverflow
Solution 2 - AndroidjiduvahView Answer on Stackoverflow
Solution 3 - AndroidLarsHView Answer on Stackoverflow
Solution 4 - AndroidmaniView Answer on Stackoverflow
Solution 5 - AndroidDaniele SegatoView Answer on Stackoverflow
Solution 6 - AndroidvonWippersnapView Answer on Stackoverflow
Solution 7 - AndroidLukeView Answer on Stackoverflow
Solution 8 - AndroidJaelson WagnerView Answer on Stackoverflow
Solution 9 - AndroidFilipe BritoView Answer on Stackoverflow
Solution 10 - AndroidMichael MaoView Answer on Stackoverflow
Solution 11 - AndroidMohit SutharView Answer on Stackoverflow