原文 WPF 中那些可跨线程访问的 DispatcherObject(WPF Free Threaded Dispatcher Object)

众所周知的,WPF 中多数对象都继承自 DispatcherObject,而 DispatcherObject 带给这些对象一个特点:不能跨线程访问。

不过,WPF 中依然存在一些例外。本文将介绍 WPF 那些可跨线程访问的 DispatcherObject,如何充分利用这个特点提高应用程序的性能,以及如何自己编写这样的 DispatcherObject


什么样的 DispatcherObject 可以跨线程访问?

要了解什么样的 DispatcherObject 可以跨线程访问,需要知道 WPF 是如何限制对象的跨线程访问的。

Dispatcher 属性

DispatcherObject 类有一个 Dispatcher 属性,它长下面这样:

public Dispatcher Dispatcher
{
get
{
// This property is free-threaded.
return _dispatcher;
}
}

属性在 Dispatcher 的构造函数中被赋值:

protected DispatcherObject()
{
_dispatcher = Dispatcher.CurrentDispatcher;
}

CheckAccess 和 VerifyAccess

DispatcherObject 提供了两种验证 Dispatcher 的方法,CheckAccess 和 VerifyAccess;他们内部的实现是调用 Dispatcher 类型的 CheckAccess 和 VerifyAccess 方法。

CheckAccess 用于检查调用线程对此对象是否有访问权,如果有访问权,则返回 true,否则返回 false。而 VerifyAccess 也是用于检查调用线程对此对象是否有访问权,如果没有访问权会抛出异常。

你可以阅读这两个方法的代码来了解其实现原理。每个方法只有短短的一两行而已,非常容易理解。

/// <summary>
/// Checks that the calling thread has access to this object.
/// </summary>
/// <remarks>
/// Only the dispatcher thread may access DispatcherObjects.
/// <p/>
/// This method is public so that any thread can probe to
/// see if it has access to the DispatcherObject.
/// </remarks>
/// <returns>
/// True if the calling thread has access to this object.
/// </returns>
[System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)]
public bool CheckAccess()
{
return Thread == Thread.CurrentThread;
} /// <summary>
/// Verifies that the calling thread has access to this object.
/// </summary>
/// <remarks>
/// Only the dispatcher thread may access DispatcherObjects.
/// <p/>
/// This method is public so that derived classes can probe to
/// see if the calling thread has access to itself.
/// </remarks>
[System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)]
public void VerifyAccess()
{
if(!CheckAccess())
{
throw new InvalidOperationException(SR.Get(SRID.VerifyAccess));
}
}

需要说明的是,只有调用这两个方法才会对线程的访问权限进行检查。如果你写一个类继承自 DispatcherObject 而在你的属性和方法中不直接或间接调用 VerifyAccess,那么这是不受线程访问限制的。

只不过,WPF 封装的大多对象和属性都调用了 VerifyAccess(例如依赖项属性),所以很大程度上限制了 WPF UI 的线程访问权限。

_dispatcher 的重新赋值

Dispatcher 属性的获取实际上就是在拿 _dispatcher 字段的值。于是我们现在仔细寻找 _dispatcher 的所有赋值代码,只有三处,就是下面这三个方法:

  1. 构造函数,会赋值为当前线程的 Dispatcher
  2. DetachFromDispatcher,会赋值为 null
  3. MakeSentinel,会赋值为另一个线程的 Dispatcher 的值(即一个线程创建,但由另一个线程来使用)。
/// <summary>
/// Instantiate this object associated with the current Dispatcher.
/// </summary>
protected DispatcherObject()
{
_dispatcher = Dispatcher.CurrentDispatcher;
} // This method allows certain derived classes to break the dispatcher affinity
// of our objects.
[FriendAccessAllowed] // Built into Base, also used by Framework.
internal void DetachFromDispatcher()
{
_dispatcher = null;
} // Make this object a "sentinel" - it can be used in equality tests, but should
// not be used in any other way. To enforce this and catch bugs, use a special
// sentinel dispatcher, so that calls to CheckAccess and VerifyAccess will
// fail; this will catch most accidental uses of the sentinel.
[FriendAccessAllowed] // Built into Base, also used by Framework.
internal void MakeSentinel()
{
_dispatcher = EnsureSentinelDispatcher();
} private static Dispatcher EnsureSentinelDispatcher()
{
if (_sentinelDispatcher == null)
{
// lazy creation - the first thread reaching here creates the sentinel
// dispatcher, all other threads use it.
Dispatcher sentinelDispatcher = new Dispatcher(isSentinel:true);
Interlocked.CompareExchange<Dispatcher>(ref _sentinelDispatcher, sentinelDispatcher, null);
} return _sentinelDispatcher;
}

于是我们发现,实际上 Dispatcher 属性虽然在 DispatcherObject 对象创建的时候会赋值,但实际上提供了多种方法来修改值。有的是修改成另一个线程的 Dispatcher,而有的就是粗暴地赋值为 null

_dispatcher 赋值为 null

无论是 CheckAccess 还是 VerifyAccess 方法,实际上都对 null 进行了判断。

public bool CheckAccess()
{
// This method is free-threaded. bool accessAllowed = true;
Dispatcher dispatcher = _dispatcher; // Note: a DispatcherObject that is not associated with a
// dispatcher is considered to be free-threaded.
if(dispatcher != null)
{
accessAllowed = dispatcher.CheckAccess();
} return accessAllowed;
} public void VerifyAccess()
{
// This method is free-threaded. Dispatcher dispatcher = _dispatcher; // Note: a DispatcherObject that is not associated with a
// dispatcher is considered to be free-threaded.
if(dispatcher != null)
{
dispatcher.VerifyAccess();
}
}

注释中说明:

Note: a DispatcherObject that is not associated with a dispatcher is considered to be free-threaded.

也就是说,如果一个 DispatcherObject 对象没有任何被关联的 Dispatcher,那么就被认为这个 DispatcherObject 没有线程访问限制,此对象将允许被任何线程访问。

哪些 DispatcherObject 是可以跨线程访问的?

通过阅读 DispatcherObject 的源码,我们可以知道 DispatcherObject 其实是允许跨线程访问的,它只是在刚刚创建的时候如果没有其他额外的方法调用使得 Dispatcher 属性改变,那么就只能被创建它的线程访问。

但也需要注意,能够改变 Dispatcher 属性值的两个方法 DetachFromDispatcher 和 MakeSentinel 都是 internal 的。这意味着只有微软自己在 WindowsBase、PresentationCore 和 PresentationFramework 程序集中编写的类型才能修改其值。可是,有哪些类呢?

通过查找 DetachFromDispatcher 的引用,我找到了以下类型:

  • Freezable
  • Style
  • StyleHelper
  • TriggerBase
  • BeginStoryboard
  • ResourceDictionary

也就是说,这些类型的实例会在某种特定的条件下从单线程访问权限变为可被任意跨线程访问。(实际上 ResourceDictionary 并不是一个 DispatcherObject,不过它会访问 Owner,这是一个 DependencyObject;所以也会涉及到一点跨线程问题。)

而查找 MakeSentinel 的引用,又可以找到:

  • ItemsControl

也就是说,ItemsControl 类在某种情况下提供了一种在一个线程中创建对象,在另外一个线程中使用的特性。

Freezable

Freezable 是继承自 DispatcherObject 的一个抽象类,其出现的主要目的就是解决 WPF 单线程模型带来的负面性能影响。

Freezable 主要由那些与图形渲染强相关的 WPF 类型继承,比如 BrushTransformGeometryD3DImage 还有各种动画等。典型的,这些类型都对高性能渲染有要求。这些类型的刚开始创建的时候只能由创建的对象对它进行修改,而且在修改的时候还会引发 Changed 事件以便相关类型对其进行处理。不过,一旦 Freeze,这些类型将变成不可修改,这时不会也不需要再引发 Changed 事件,可以提升性能,同时对所有线程开放访问权限,这样能继续提升性能。

你可以对 Freezable 对象调用 Freeze() 方法,调用之后,其 Dispatcher 属性会被设为 null,于是对象可以跨线程访问。

在 XAML 中,你可以通过指定 PresentationOptions:Freeze 特性达到同样的目的。如下面的例子,SolidColorBrush 对象在创建完设置完所有的值之后,会调用 Freeze 冻结这个对象以便跨线程访问。

<Page
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:PresentationOptions="http://schemas.microsoft.com/winfx/2006/xaml/presentation/options"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="PresentationOptions">
<!-- 如果你的 mc:Ignorable 有多个,请用空格隔开。 -->
<Page.Resources>
<!-- 注意,在 Resource 中的 SolidColorBrush 默认情况下是不会自动 Freeze 的, -->
<!-- 但是,你可以通过指定 PresentationOptions:Freeze 特性使得它在创建完后 Freeze。 -->
<!-- 对象在 Resources 中不会自动创建,它会在第一次被使用的时候创建, -->
<!-- 也就是说,你如果要验证它的跨线程访问,需要使用两个不同的线程访问它。 -->
<SolidColorBrush x:Key="Walterlv.Brush.Demo" PresentationOptions:Freeze="True" Color="Red" />
</Page.Resources>
</Page>

对于以上代码,有一些是需要说明的:

  1. 如果你的 mc:Ignorable 有多个,请用空格隔开。
  2. 在 Resource 中的 SolidColorBrush 默认情况下是不会自动 Freeze 的;但是,你可以通过指定 PresentationOptions:Freeze 特性使得它在创建完后 Freeze。
  3. 对象在 Resources 中不会自动创建,它会在第一次被使用的时候创建;也就是说,你如果要验证它的跨线程访问,需要使用两个不同的线程访问它(仅仅用一个后台线程去验证它,你会发现后台线程依然能够正常访问它的依赖项属性的值)。

Style

Style 是直接继承自 DispatcherObject 的类型,并没有 Freeze 相关的方法。不过这不重要,因为重要的是能够访问到内部的 DetachFromDispatcher 方法。

Style 访问 DetachFromDispatcher 的代码在 public 的 Seal 方法中,这是继承自 internal的 ISealable 接口的方法。

internal interface ISealable
{
bool CanSeal { get; }
void Seal();
bool IsSealed { get; }
}

为了方便理解,我把 Seal 方法进行简化后贴在下面:

/// <summary>
/// This Style and all factories/triggers are now immutable
/// </summary>
public void Seal()
{
// Verify Context Access
VerifyAccess(); // 99% case - Style is already sealed.
if (_sealed) return; // 省略一些验证代码。 // Seal setters
if (_setters != null) _setters.Seal(); // Seal triggers
if (_visualTriggers != null) _visualTriggers.Seal(); // Will throw InvalidOperationException if we find a loop of
// BasedOn references. (A.BasedOn = B, B.BasedOn = C, C.BasedOn = A)
CheckForCircularBasedOnReferences(); // Seal BasedOn Style chain
if (_basedOn != null) _basedOn.Seal(); // Seal the ResourceDictionary
if (_resources != null) _resources.IsReadOnly = true; //
// Build shared tables
// // Process all Setters set on the selfStyle. This stores all the property
// setters on the current styles into PropertyValues list, so it can be used
// by ProcessSelfStyle in the next step. The EventSetters for the current
// and all the basedOn styles are merged into the EventHandlersStore on the
// current style.
ProcessSetters(this); // Add an entry in the EventDependents list for
// the TargetType's EventHandlersStore. Notice
// that the childIndex is 0.
StyleHelper.AddEventDependent(0, this.EventHandlersStore, ref EventDependents); // Process all PropertyValues (all are "Self") in the Style
// chain (base added first)
ProcessSelfStyles(this); // Process all TriggerBase PropertyValues ("Self" triggers
// and child triggers) in the Style chain last (highest priority)
ProcessVisualTriggers(this); // Sort the ResourceDependents, to help avoid duplicate invalidations
StyleHelper.SortResourceDependents(ref ResourceDependents); // All done, seal self and call it a day.
_sealed = true; // Remove thread affinity so it can be accessed across threads
DetachFromDispatcher();
}

具体来说,就是将 Style 中的所有属性进行 Seal,将资源设为只读;然后,将自己的 Dispatcher 属性设为 null

Template

不过,我们通常使用 Style 的方式都是在 Style 中写控件模板。如果控件模板不支持 Seal,那么 Style 即便 Seal,多数情况下也是没有用的。

在 StyleHelper 类型中,处理了控件模板的 Seal

它处理的是 FrameworkTemplate,这是控件模板的基类,具体来说,有这些类型:

  • ControlTemplate
  • DataTemplate
  • ItemsPanelTemplate
  • ItemContainerTemplate
  • HierarchicalDataTemplate
  • ContentPresenter.DefaultTemplate
  • ContentPresenter.UseContentTemplate

以下是 StyleHelper.SealTemplate 方法。这里原本是 FrameworkTemplate 内部的 Seal 方法的实现,不过 Seal 内部调到了 StyleHelper.SealTemplate 静态方法了。

为了便于理解,我也对其进行了精简。

internal static void SealTemplate(
FrameworkTemplate frameworkTemplate,
ref bool isSealed,
FrameworkElementFactory templateRoot,
TriggerCollection triggers,
ResourceDictionary resources,
HybridDictionary childIndexFromChildID,
ref FrugalStructList<ChildRecord> childRecordFromChildIndex,
ref FrugalStructList<ItemStructMap<TriggerSourceRecord>> triggerSourceRecordFromChildIndex,
ref FrugalStructList<ContainerDependent> containerDependents,
ref FrugalStructList<ChildPropertyDependent> resourceDependents,
ref ItemStructList<ChildEventDependent> eventDependents,
ref HybridDictionary triggerActions,
ref HybridDictionary dataTriggerRecordFromBinding,
ref bool hasInstanceValues,
ref EventHandlersStore eventHandlersStore)
{
// This template has already been sealed. There is no more to do.
if (isSealed) return; // Seal template nodes (if exists) if (frameworkTemplate != null) frameworkTemplate.ProcessTemplateBeforeSeal(); if (templateRoot != null) templateRoot.Seal(frameworkTemplate); // Seal triggers
if (triggers != null) triggers.Seal(); // Seal Resource Dictionary
if (resources != null) resources.IsReadOnly = true; // Build shared tables StyleHelper.ProcessTemplateTriggers(
triggers,
frameworkTemplate,
ref childRecordFromChildIndex,
ref triggerSourceRecordFromChildIndex, ref containerDependents, ref resourceDependents, ref eventDependents,
ref dataTriggerRecordFromBinding, childIndexFromChildID, ref hasInstanceValues,
ref triggerActions, templateRoot, ref eventHandlersStore,
ref frameworkTemplate.PropertyTriggersWithActions,
ref frameworkTemplate.DataTriggersWithActions,
ref hasHandler ); frameworkTemplate.HasLoadedChangeHandler = hasHandler; frameworkTemplate.SetResourceReferenceState(); // All done, seal self and call it a day.
isSealed = true; // Remove thread affinity so it can be accessed across threads
frameworkTemplate.DetachFromDispatcher();
}

其中,frameworkTemplate.DetachFromDispatcher() 方法即调用基类 DispatcherObject 中的 DetachFromDispatcher 方法。

方法内部也是对各种属性进行了 Seal 和只读化处理。最后,将自己的 Dispatcher 属性设为 null

Style 和 Template

由于每次应用模板的时候,都是创建新的 UI 控件,所以实际上通过模板创建的 UI 对象并不会产生跨线程访问的问题。也就是说,当 Style 和 Template 设置为可跨线程访问之后,是可以被多个线程同时访问创建控件而不会产生跨线程访问的问题。

写在 XAML 中的 ISealable 在创建的时候就会执行 Seal()。也就是说,你只要在 XAML 中写下了这个对象,那么就会在创建完后 Seal。这点跟 Freezable 是不一样的,Freezable 是需要自己主动编写 XAML 或 C# 代码进行 Freeze 的。

从这里可以推论出,你在 XAML 中写的样式,可以被跨线程访问而不会出现线程安全问题。

强制让一个 DispatcherObject 跨线程访问

从前面的各种源码分析来看,使用常规方法让任意一个对象进行跨线程访问几乎是不可能的了。剩下的就只是做一些邪恶的事情了,比如 —— 反射。

可以反射调用 DetachFromDispatcher 方法,将 Dispatcher 属性值清空,这样的对象将可以跨所有线程访问。不过要小心,你随意写的对象可能实际上是不具备跨线程访问能力的,这样的修改可能导致线程安全问题,你需要自行承担后果。

可以反射直接修改 _dispatcher 字段的值,改为目标线程中的 Dispatcher。这样的做法只是切换了一个线程,效果和调用 MakeSentinel 是一样的。使用这样的方式可以让创建对象的线程和使用对象的线程分开,适用于创建对象需要花费大量时间的对象 —— 如 BitmapImage

特别的,如果你的对象中有子 DispatcherObject 对象,你需要像上面的源码那样将所有子对象的 Dispatcher 属性都进行替换才行。

为了方便,我写了一个辅助方法来完成这样的 Dispatcher 属性值切换。

using System;
using System.Reflection;
using System.Threading;
using System.Windows.Threading; namespace Walterlv.Windows.Threading
{
/// <summary>
/// 包含 <see cref="DispatcherObject"/> 及其派生类对象切换所属线程的相关方法。
/// </summary>
public static class DispatcherSwitcher
{
/// <summary>
/// 延迟初始化 <see cref="Dispatcher"/> 类型中的 _dispatcher 字段。
/// </summary>
private static readonly Lazy<FieldInfo> DispatcherFieldInfo =
new Lazy<FieldInfo>(() => typeof(DispatcherObject).GetField("_dispatcher", BindingFlags.NonPublic | BindingFlags.Instance),
LazyThreadSafetyMode.None); /// <summary>
/// 将指定的 <see cref="DispatcherObject"/> 对象的所属线程切换至指定调度器所属的线程。
/// 注意:如果此对象包含嵌套的其他对象,则极有可能会发生跨线程访问的异常,请谨慎使用!
/// </summary>
/// <param name="d"></param>
/// <param name="dispatcher"></param>
internal static void SwitchTo(DispatcherObject d, Dispatcher dispatcher)
{
if (d.Dispatcher == dispatcher) return; var field = DispatcherFieldInfo.Value;
if (field == null)
{
throw new FieldAccessException();
} field.SetValue(d, dispatcher);
}
}
}

总结

  1. 为什么 DispatcherObject 可以限制跨线程访问?

    • 因为内部有 CheckAccess 和 VerifyAccess 方法检查线程的访问权限
    • 众多子类的属性和方法在使用前调用了 VerifyAccess 来验证调用方的线程
  2. 在 XAML 中编写的代码时,定义在 FrameworkElement 的 Resources 中的对象,哪些可以跨线程访问,哪些不可以跨线程访问?
    • 非继承自 DispatcherObject 的对象可以跨线程访问(这里就不要钻牛角尖说自己写的烂类了)

      • 比如 <system:String>Walterlv</system:String>
    • 继承自 DispatcherObject 的对象,但是同时实现了内部 ISealable 接口的对象
      • 比如 StyleTemplateDataTemplateItemTemplate 等
    • 继承自 Freezable 的对象
      • 指定了 PresentationOptions:Freeze="True" 特性的对象可以跨线程访问
      • 没有指定此特性的对象不可以跨线程访问
      • 这些对象比如 BrushTransformGeometryD3DImage 还有各种动画等
    • 其他 DispatcherObject 对象
      • 不可以跨线程访问(当然你自己写的类型,没有访问基类的 VerifyAccess 的话就没事)
  3. 可以随意切换 DispatcherObject 关联的 Dispatcher 吗?
    • 不可以随意切换,因为切换关联 Dispatcher 的方法都是 internal 的
    • 不过我们可以使用反射来间接实现这个效果(当然,你需要自行承担线程安全后果,以及切换不完全造成的跨线程访问问题)

参考资料

本文会经常更新,请阅读原文: https://walterlv.com/post/wpf-free-threaded-dispatcher-object.html,以避免陈旧错误知识的误导,同时有更好的阅读体验。

本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名 吕毅 (包含链接: https://walterlv.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系 (walter.lv@qq.com)

WPF 中那些可跨线程访问的 DispatcherObject(WPF Free Threaded Dispatcher Object)的更多相关文章

  1. wpf(怎么跨线程访问wpf控件)

    在编写代码时,我们经常会碰到一些子线程中处理完的信息,需要通知另一个线程(我这边处理完了,该你了). 但是当我们通知WPF的UI线程时需要用到Dispatcher. 首先我们需要想好在UI控件上需要显 ...

  2. ASP.NET Boilerplate 学习 AspNet Core2 浏览器缓存使用 c#基础,单线程,跨线程访问和线程带参数 wpf 禁用启用webbroswer右键菜单 EF Core 2.0使用MsSql/MySql实现DB First和Code First ASP.NET Core部署到Windows IIS QRCode.js:使用 JavaScript 生成

    ASP.NET Boilerplate 学习   1.在http://www.aspnetboilerplate.com/Templates 网站下载ABP模版 2.解压后打开解决方案,解决方案目录: ...

  3. 如何跨线程访问Winform中的UI元素

    如何跨线程访问Winform中的UI元素 假如制作一个Socket聊天应用程序,很可能会用到多线程: 例如为Receive方法开辟单独一个线程,例如为Receive方法开辟单独一个线程(后台运行的线程 ...

  4. Winform之跨线程访问控件(在进度条上显示字体)

    此文章对于遇到必须使用线程但是没有办法在线程内操作控件的问题的处理  有很好的解决方案(个人认为的.有更好的方案欢迎交流.) 在做跨线程访问之前我们先了解下我们所做的需要达到的效果: 这个是批量的将x ...

  5. C# WinFrom 跨线程访问控件

    1.跨线程访问控件委托和类的定义 using System; using System.Windows.Forms; namespace ahwildlife.Utils { /// <summ ...

  6. InvokeHelper,让跨线程访问/修改主界面控件不再麻烦(转)

    http://bbs.csdn.net/topics/390162519 事实上,本文内容很简单且浅显,所以取消前戏,直接开始.. 源代码:在本文最后 这里是一张动画,演示在多线程(无限循环+Thre ...

  7. Winform 让跨线程访问变得更简单

    Winform 让跨线程访问变得更简单 前言 由于多线程可能导致对控件访问的不一致,导致出现问题.C#中默认是要线程安全的,即在访问控件时需要首先判断是否跨线程,如果是跨线程的直接访问,在运行时会抛出 ...

  8. winform 跨线程访问问题

    一.问题描述 进行winform 开发我们在进行数据交换时避免不了使用多线程或异步方法,这样操作也将避免不了跨线程对控件进行操作(赋值.修改属性). 下面通过一个测试说明一下问题 点击一个按钮异步对t ...

  9. c#基础,单线程,跨线程访问和线程带参数

    using System; using System.Collections.Generic; using System.Threading; using System.Windows.Forms; ...

随机推荐

  1. DIV+CSS学习笔记

    第十五章 定位 static静态定位(不对它的位置进行改变,在哪里就在那里) 默认值.没有定位,元素出现在正常的流中(忽略 top, bottom, left, right 或者 z-index 声明 ...

  2. Loadrunner11--输入license后提示违反许可证安全,禁止操作

    安装中文补丁包后,重新把mlr5lprg.dll.lm70.dll覆盖LR11安装目录下“bin”文件夹下mlr5lprg.dll.lm70.dll.运行deletelicense.exe.重新用管理 ...

  3. set_fix_multiple_port_nets

    set_fix_multiple_port_nets   -all    -buffer_constants 加上这个命令之后 综合之后的网表就不会出现assign语句 否则会出现

  4. 【例题 4-1 UVA - 1339】 Ancient Cipher

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 位置其实都没关系了. 只要每个字母都有对应的字母,它们的数量相同就可以了. 求出每种字母的数量. 排序之后. 肯定是要一一对应的. ...

  5. Android Studio插件推荐-GsonFormat,ButterKnifeZelezny

    原创文章.转载请注明 http://blog.csdn.net/leejizhou/article/details/50557786 本篇介绍的仅仅适用android studio和 Intellij ...

  6. C#调用oracle存储过程自定义表类型

    http://blog.csdn.net/studyzy/article/details/11524527

  7. C语言free函数的原理——————————【Badboy】

    今天在网上看到了这样一个问题,"假设malloc 了一块字符串的内存.然后,它改变了这个字符串的大小,问会不会有一部分内存没有被释放掉."这个问题,曾经的确没有细致想过. 当然.我 ...

  8. Java与IOS日期格式

    //JAVA日期格式 Date date = new Date(); SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM- ...

  9. NSArray NSDictionary一些用法

    //从字符串分割到数组- componentsSeparatedByString: NSString *str = [NSString alloc] initWithString:@"a,b ...

  10. Apache与weblogic整合实战(独家研究)

    用apache来处理外界的请求,再把请求转发给wls,这样就行突破wls express版本号的5用户限制 详细配置例如以下 copy ${WLS_Server}/server/lib下的mod_wl ...