sun.misc.Unsafe

[url]http://www.cnblogs.com/tianchi/archive/2013/02/27/2935695.html[/url]
Java不能直接访问操作系统底层,而是通过本地方法来访问,Unsafe提供了硬件级别的原子操作,提供了以下功能:
分配、释放内存

操作内存主要依靠下面三个方法:
allocateMemory:分配内存;
reallocateMemory:扩充内存;
freeMemory:释放内存;
操作对象的字段

  Java对象中字段的定位可以通过staticFieldOffset实现,而读取具体便宜位置的字段的值可以使用getLong(根据不同类型选择不同的函数)来完成,可以使用putLong(根据不同的类型选择不同的函数)来设置值,可以使用arrayBaseOffset(获取数组第一个元素的偏移地址)和arrayIndexScale(获取数组中元素增量的地址)来访问数组中的元素。
CAS操作

CAS操作(会在java.util.concurrent中大量地使用)包括三个操作数:
内存位置;
预期原值;
新值;
如果内存位置的值与预期的值相等,那么处理器会将该位置的值设置为新值,否则,处理器不做任何处理。int类型的CAS实现如下:
复制代码
static inline bool compareAndSwap (volatile jint *addr, jint old, jint new_val)
{
jboolean result = false;
spinlock lock;
if ((result = (*addr == old)))
*addr = new_val;
return result;
}
复制代码
这段代码看起来根本无法保证操作的原子性,而其中的关键就在于spinlock lock这段看似没有用的代码,这时会调用构造函数,其中不停检查lock的值是否为0,如果不是0则线程让步等到线程下次执行时再次检查;否则,将lock设置为1。在离开函数的时候会调用其析构函数,将lock的值再次设置为0。spinlock的代码如下:
复制代码
class spinlock
{
static volatile obj_addr_t lock;
public:
spinlock ()
{
while (!compare_and_swap(&lock, 0, 1))
_Jv_ThreadYield();
}
~spinlock ()
{
release_set(&lock, 0);
}
};
复制代码
而且,不管是在多核还是在单核的情况下,总是需要一些特殊的手段才能保证CAS操作成功吧,比如锁总线,compare_and_swap的实现如下:
复制代码
inline static bool compare_and_swap(volatile obj_addr_t *addr, obj_addr_t old, obj_addr_t new_val)
{
char result;
#ifdef __x86_64__ __asm__ __volatile__("lock; cmpxchgq %2, %0; setz %1"
: "=m"(*(addr)), "=q"(result)
: "r" (new_val), "a"(old), "m"(*addr)
: "memory");
#else
__asm__ __volatile__("lock; cmpxchgl %2, %0; setz %1"
: "=m"(*addr), "=q"(result)
: "r" (new_val), "a"(old), "m"(*addr)
: "memory");
#endif
return (bool) result;
}
复制代码
挂起和恢复

使用park方法将一个线程挂起,知道超时或者中断等条件出现,实现如下:
复制代码
void sun::misc::Unsafe::park(jboolean isAbsolute, jlong time)
{
using namespace ::java::lang;
Thread *thread = Thread::currentThread();
natThread *nt = (natThread *) thread->data;
nt->park_helper.park(isAbsolute, time);
}
复制代码
使用unpark来终止一个挂起的线程,使其恢复正常,实现如下:
sun::misc::Unsafe::unpark (::java::lang::Thread *thread)
{
natThread *nt = (natThread *) thread->data;
nt->park_helper.unpark ();
}
----- -- -
END
### 回答1: sun.misc.unsafe.park(native me) 是Java中的一个方法,它是用来阻塞当前线程的。具体来说,它会使当前线程进入等待状态,直到被唤醒或者被中断。这个方法通常用于实现线程的同步和互斥。 ### 回答2: sun.misc.unsafe.park(native me 是一个Java API中的方法,它通过使用Unsafe类的park方法来使当前线程进入休眠状态。在这个方法中,native关键字表示它是由本地代码实现的,也就是说具体的实现是由底层的操作系统提供的。 调用sun.misc.unsafe.park(native me 方法可以实现线程的等待和唤醒操作。当当前线程执行到这个方法时,会立即进入休眠状态,暂停自己的执行,直到其他线程通过调用Unsafe类的unpark方法唤醒它。 这个机制通常用于线程同步的场景,可以实现线程之间的协作。比如,一个生产者线程通过调用Unsafe类的park方法进入休眠状态,等待某个条件满足后再继续执行;而一个消费者线程在满足某个条件后,通过调用Unsafe类的unpark方法唤醒生产者线程,使其恢复执行。 需要注意的是,sun.misc.unsafe.park(native me 方法通常不建议直接使用,因为它是一个内部API,可能会在未来的版本中被移除或者修改。在实际应用中,可以使用更高级的并发工具,如Lock和Condition、CountDownLatch、Semaphore等来实现线程的等待和唤醒操作,这些API提供了更加安全和可靠的线程同步机制。 ### 回答3: sun.misc.unsafe.park(native me)是Java中的一个方法,它是由sun.misc.Unsafe类提供的。该方法主要用于线程的阻塞等待。 在Java中,线程可以通过调用park方法进入阻塞状态,直到某个条件满足或者其他线程唤醒它。park方法是一种低级的阻塞机制,它不会占用CPU资源,因此适用于一些需要较长等待时间的场景。 使用park方法时,我们需要传入一个native me参数。native me是指要阻塞的线程对象,也就是当前执行park方法的线程自身。 park方法可以通过其他线程的unpark方法来唤醒被阻塞的线程。unpark方法会给指定的线程一个许可证,使得park方法立即返回。 使用park方法进行线程的阻塞等待可以提高线程的效率和性能,避免了一直占用CPU资源。 需要注意的是,sun.misc.unsafe.park(native me)方法属于sun.misc包下的不稳定的、不建议直接使用的API。在实际开发中,应该尽量避免使用这些API,而是使用Java提供的高级并发类库,如java.util.concurrent包下的Locks和Conditions,或者使用更高层次的并发框架,如线程池。这些类库和框架提供了更稳定、易用且可扩展的线程同步和阻塞等待机制。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值