style: apply dart format across project
This commit is contained in:
@@ -5,19 +5,20 @@ import 'dart:async';
|
||||
|
||||
/// 성능 최적화를 위한 유틸리티 클래스
|
||||
class PerformanceOptimizer {
|
||||
static final PerformanceOptimizer _instance = PerformanceOptimizer._internal();
|
||||
static final PerformanceOptimizer _instance =
|
||||
PerformanceOptimizer._internal();
|
||||
factory PerformanceOptimizer() => _instance;
|
||||
PerformanceOptimizer._internal();
|
||||
|
||||
|
||||
// 프레임 타이밍 정보
|
||||
final List<FrameTiming> _frameTimings = [];
|
||||
bool _isMonitoring = false;
|
||||
|
||||
|
||||
/// 프레임 성능 모니터링 시작
|
||||
void startFrameMonitoring() {
|
||||
if (_isMonitoring) return;
|
||||
_isMonitoring = true;
|
||||
|
||||
|
||||
SchedulerBinding.instance.addTimingsCallback((timings) {
|
||||
_frameTimings.addAll(timings);
|
||||
// 최근 100개 프레임만 유지
|
||||
@@ -26,27 +27,27 @@ class PerformanceOptimizer {
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
/// 프레임 성능 모니터링 중지
|
||||
void stopFrameMonitoring() {
|
||||
if (!_isMonitoring) return;
|
||||
_isMonitoring = false;
|
||||
SchedulerBinding.instance.addTimingsCallback((_) {});
|
||||
}
|
||||
|
||||
|
||||
/// 평균 FPS 계산
|
||||
double getAverageFPS() {
|
||||
if (_frameTimings.isEmpty) return 0.0;
|
||||
|
||||
|
||||
double totalDuration = 0;
|
||||
for (final timing in _frameTimings) {
|
||||
totalDuration += timing.totalSpan.inMicroseconds;
|
||||
}
|
||||
|
||||
|
||||
final averageDuration = totalDuration / _frameTimings.length;
|
||||
return 1000000 / averageDuration; // microseconds to FPS
|
||||
}
|
||||
|
||||
|
||||
/// 메모리 사용량 모니터링
|
||||
static Future<MemoryInfo> getMemoryInfo() async {
|
||||
// Flutter에서는 직접적인 메모리 사용량 측정이 제한적이므로
|
||||
@@ -57,7 +58,7 @@ class PerformanceOptimizer {
|
||||
capacity: imageCache.maximumSizeBytes,
|
||||
);
|
||||
}
|
||||
|
||||
|
||||
/// 위젯 재빌드 최적화를 위한 데바운서
|
||||
static Timer? _debounceTimer;
|
||||
static void debounce(
|
||||
@@ -67,7 +68,7 @@ class PerformanceOptimizer {
|
||||
_debounceTimer?.cancel();
|
||||
_debounceTimer = Timer(delay, callback);
|
||||
}
|
||||
|
||||
|
||||
/// 스로틀링 - 지정된 시간 간격으로만 실행
|
||||
static DateTime? _lastThrottleTime;
|
||||
static void throttle(
|
||||
@@ -81,7 +82,7 @@ class PerformanceOptimizer {
|
||||
callback();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/// 무거운 연산을 별도 Isolate에서 실행
|
||||
static Future<T> runInIsolate<T>(
|
||||
ComputeCallback<dynamic, T> callback,
|
||||
@@ -89,7 +90,7 @@ class PerformanceOptimizer {
|
||||
) async {
|
||||
return await compute(callback, parameter);
|
||||
}
|
||||
|
||||
|
||||
/// 레이지 로딩을 위한 페이지네이션 헬퍼
|
||||
static List<T> paginate<T>({
|
||||
required List<T> items,
|
||||
@@ -98,13 +99,14 @@ class PerformanceOptimizer {
|
||||
}) {
|
||||
final startIndex = page * pageSize;
|
||||
final endIndex = (startIndex + pageSize).clamp(0, items.length);
|
||||
|
||||
|
||||
if (startIndex >= items.length) return [];
|
||||
return items.sublist(startIndex, endIndex);
|
||||
}
|
||||
|
||||
|
||||
/// 이미지 최적화 - 메모리 효율적인 크기로 조정
|
||||
static double getOptimalImageSize(BuildContext context, {
|
||||
static double getOptimalImageSize(
|
||||
BuildContext context, {
|
||||
required double originalSize,
|
||||
double maxSize = 1000,
|
||||
}) {
|
||||
@@ -113,29 +115,29 @@ class PerformanceOptimizer {
|
||||
final maxDimension = screenSize.width > screenSize.height
|
||||
? screenSize.width
|
||||
: screenSize.height;
|
||||
|
||||
|
||||
final optimalSize = (maxDimension * devicePixelRatio).clamp(100.0, maxSize);
|
||||
return optimalSize < originalSize ? optimalSize : originalSize;
|
||||
}
|
||||
|
||||
|
||||
/// 위젯 키 최적화
|
||||
static Key generateOptimizedKey(String prefix, dynamic identifier) {
|
||||
return ValueKey('${prefix}_$identifier');
|
||||
}
|
||||
|
||||
|
||||
/// 애니메이션 최적화 - 보이지 않는 애니메이션 중지
|
||||
static bool shouldAnimateWidget(BuildContext context) {
|
||||
final mediaQuery = MediaQuery.of(context);
|
||||
return !mediaQuery.disableAnimations && mediaQuery.accessibleNavigation;
|
||||
}
|
||||
|
||||
|
||||
/// 스크롤 성능 최적화
|
||||
static ScrollPhysics getOptimizedScrollPhysics() {
|
||||
return const BouncingScrollPhysics(
|
||||
parent: AlwaysScrollableScrollPhysics(),
|
||||
);
|
||||
}
|
||||
|
||||
|
||||
/// 빌드 최적화를 위한 const 위젯 권장사항 체크
|
||||
static void checkConstOptimization() {
|
||||
if (kDebugMode) {
|
||||
@@ -147,16 +149,16 @@ class PerformanceOptimizer {
|
||||
print('5. 애니메이션은 AnimatedBuilder 사용');
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/// 메모리 누수 감지 헬퍼
|
||||
static final Map<String, int> _widgetCounts = {};
|
||||
|
||||
|
||||
static void trackWidget(String widgetName, bool isCreated) {
|
||||
if (!kDebugMode) return;
|
||||
|
||||
_widgetCounts[widgetName] = (_widgetCounts[widgetName] ?? 0) +
|
||||
(isCreated ? 1 : -1);
|
||||
|
||||
|
||||
_widgetCounts[widgetName] =
|
||||
(_widgetCounts[widgetName] ?? 0) + (isCreated ? 1 : -1);
|
||||
|
||||
// 위젯이 비정상적으로 많이 생성되면 경고
|
||||
if ((_widgetCounts[widgetName] ?? 0) > 100) {
|
||||
print('⚠️ 경고: $widgetName 위젯이 100개 이상 생성됨. 메모리 누수 가능성!');
|
||||
@@ -168,16 +170,18 @@ class PerformanceOptimizer {
|
||||
class MemoryInfo {
|
||||
final int currentUsage;
|
||||
final int capacity;
|
||||
|
||||
|
||||
MemoryInfo({
|
||||
required this.currentUsage,
|
||||
required this.capacity,
|
||||
});
|
||||
|
||||
|
||||
double get usagePercentage => (currentUsage / capacity) * 100;
|
||||
|
||||
String get formattedUsage => '${(currentUsage / 1024 / 1024).toStringAsFixed(2)} MB';
|
||||
String get formattedCapacity => '${(capacity / 1024 / 1024).toStringAsFixed(2)} MB';
|
||||
|
||||
String get formattedUsage =>
|
||||
'${(currentUsage / 1024 / 1024).toStringAsFixed(2)} MB';
|
||||
String get formattedCapacity =>
|
||||
'${(capacity / 1024 / 1024).toStringAsFixed(2)} MB';
|
||||
}
|
||||
|
||||
/// 성능 측정 데코레이터
|
||||
@@ -187,7 +191,7 @@ class PerformanceMeasure {
|
||||
required Future<T> Function() operation,
|
||||
}) async {
|
||||
if (!kDebugMode) return await operation();
|
||||
|
||||
|
||||
final stopwatch = Stopwatch()..start();
|
||||
try {
|
||||
final result = await operation();
|
||||
@@ -200,4 +204,4 @@ class PerformanceMeasure {
|
||||
rethrow;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user