http://blog.csdn.net/gz_huangzl/article/details/52486509

前言

在我们开发游戏的过程中,经常会碰到脚本引用丢失的情况,但是怎么把它们修复到我们的理想情况呢?先在这打个预防针,不是所有情况下的脚本引用丢失都能修复,但绝大多数情况下都是可行的,只要你知道原来脚本的GUID和FILEID(不知道也可以在prefab中找到),最重要的是你要有(必须有)用来做修复的脚本GUID和FILEID,要不然就没办法修复 了。

我举个极端情况,假如Prefab挂了A脚本,但是这个Prefab是第三方的,但是它却没有把A脚本给你,这种情况下你就没办法修复了,除非你通过其它途径知道了A脚本的实现,你自己在本地创建了一个类A的脚本,这样才可能被修复

脚本在Prefab中被引用的原理

脚本被引用有两种情况

a.prefab引用的是cs文件脚本

b.prefab引用的是dll文件中的脚本

区别一:

对于第一种情况,脚本的文件名必须和类名相同,且必须为MonoBehaviour类,不管脚本里面有一个或多个类,只有和文件名相同的类名的才能被挂接上

对于第二种情况,一个脚本可以包含多个MonoBehaviour类,如果把它打成dll后,它里面的所有类都是可以被挂接

区别二:

prefab挂了脚本,打成AssetBundle后,加载运行的时候,只有第一种情况的脚本是可以生效的,挂的dll是无效,因为bundle加载并初始化的时候,unity是从包内的脚本中去搜索的,并不会从包内的dll中去搜索(这也是脚本更新的拦路虎之一,解决方法要么动态挂脚本,要么挂的脚本不热更,跟包走)

引用的原理,如下图:


用文本编辑器打开prefab文件,如上图,挂载的脚本如上紫框内所示,fileID脚本中类的信息,guild表示脚本所在的文件ID

a.直接挂载脚本

    这种情况下,fileID的值永远是11500000,它实际上指的是MonoScript类型组件,它的值由ClassID * 10000所得,详情见官方文档,而guid能直接定位到MonoScript脚本文件

所以它是通过guid找到脚本文件,然后挂载这个脚本中与脚本文件名相同的类

b.直接挂载DLL中的脚本

   这种情况下,fileID的值是由"s\0\0\0" + type.Namespace + type.Name的值转换成MD4的值,guid指的是这个dll所对应的文件(MD4的算法贴在最后面)

所以它是通过guid找到dll文件,然后生成里面所有类的MD4,然后和这个值做比对,相同的则挂载上去

脚本引用修复

一、原来是挂的脚本(可以通过fileID等于1150000判断),要替换成新的脚本

a.把prefab中的guid通过文本工具统一替换成新脚本的guid

b.直接把新脚本的.mate文件里面的guid改成prefab中的

c.如果新脚本已经有地方有的,不能改.mate,就只能使用a方法

二、原来是挂的DLL中的脚本(可以通过fileID不等于1150000判断),要替换成新的脚本

a.把prefab中的guid改成新脚本的guid,把fileID统一改成1150000

三、原来是挂的脚本,要替换成dll中的脚本

a.把prefab中的guid改成dll的guid,把fileID统一改成DLL中类的Type算出来的MD4的值

四、原来是挂的DLL中的脚本,要替换成新DLL中的脚本

a.把prefab中的guid改成新dll的guid,把fileID统一改成新DLL中类的Type算出来的MD4的值

注:如写工具时,要获得一个本地脚本中包含哪一些类,可以使用AssetDatabase.Load,它返回的是一个MonoScript,通过它可以获得到所有类的信息

MD4算法如下:

using System;using System.Collections.Generic;using System.Linq;using System.Security.Cryptography;namespace cn.crashByNull{    public class MD4 : HashAlgorithm    {        private uint _a;        private uint _b;        private uint _c;        private uint _d;        private uint[] _x;        private int _bytesProcessed;        public MD4()        {            _x = new uint[16];            Initialize();        }        public override void Initialize()        {            _a = 0x67452301;            _b = 0xefcdab89;            _c = 0x98badcfe;            _d = 0x10325476;            _bytesProcessed = 0;        }        protected override void HashCore(byte[] array, int offset, int length)        {            ProcessMessage(Bytes(array, offset, length));        }        protected override byte[] HashFinal()        {            try            {                ProcessMessage(Padding());                return new[] { _a, _b, _c, _d }.SelectMany(word => Bytes(word)).ToArray();            }            finally            {                Initialize();            }        }        private void ProcessMessage(IEnumerable<byte> bytes)        {            foreach (byte b in bytes)            {                int c = _bytesProcessed & 63;                int i = c >> 2;                int s = (c & 3) << 3;                _x[i] = (_x[i] & ~((uint)255 << s)) | ((uint)b << s);                if (c == 63)                {                    Process16WordBlock();                }                _bytesProcessed++;            }        }        private static IEnumerable<byte> Bytes(byte[] bytes, int offset, int length)        {            for (int i = offset; i < length; i++)            {                yield return bytes[i];            }        }        private IEnumerable<byte> Bytes(uint word)        {            yield return (byte)(word & 255);            yield return (byte)((word >> 8) & 255);            yield return (byte)((word >> 16) & 255);            yield return (byte)((word >> 24) & 255);        }        private IEnumerable<byte> Repeat(byte value, int count)        {            for (int i = 0; i < count; i++)            {                yield return value;            }        }        private IEnumerable<byte> Padding()        {            return Repeat(128, 1)               .Concat(Repeat(0, ((_bytesProcessed + 8) & 0x7fffffc0) + 55 - _bytesProcessed))               .Concat(Bytes((uint)_bytesProcessed << 3))               .Concat(Repeat(0, 4));        }        private void Process16WordBlock()        {            uint aa = _a;            uint bb = _b;            uint cc = _c;            uint dd = _d;            foreach (int k in new[] { 0, 4, 8, 12 })            {                aa = Round1Operation(aa, bb, cc, dd, _x[k], 3);                dd = Round1Operation(dd, aa, bb, cc, _x[k + 1], 7);                cc = Round1Operation(cc, dd, aa, bb, _x[k + 2], 11);                bb = Round1Operation(bb, cc, dd, aa, _x[k + 3], 19);            }            foreach (int k in new[] { 0, 1, 2, 3 })            {                aa = Round2Operation(aa, bb, cc, dd, _x[k], 3);                dd = Round2Operation(dd, aa, bb, cc, _x[k + 4], 5);                cc = Round2Operation(cc, dd, aa, bb, _x[k + 8], 9);                bb = Round2Operation(bb, cc, dd, aa, _x[k + 12], 13);            }            foreach (int k in new[] { 0, 2, 1, 3 })            {                aa = Round3Operation(aa, bb, cc, dd, _x[k], 3);                dd = Round3Operation(dd, aa, bb, cc, _x[k + 8], 9);                cc = Round3Operation(cc, dd, aa, bb, _x[k + 4], 11);                bb = Round3Operation(bb, cc, dd, aa, _x[k + 12], 15);            }            unchecked            {                _a += aa;                _b += bb;                _c += cc;                _d += dd;            }        }        private static uint ROL(uint value, int numberOfBits)        {            return (value << numberOfBits) | (value >> (32 - numberOfBits));        }        private static uint Round1Operation(uint a, uint b, uint c, uint d, uint xk, int s)        {            unchecked            {                return ROL(a + ((b & c) | (~b & d)) + xk, s);            }        }        private static uint Round2Operation(uint a, uint b, uint c, uint d, uint xk, int s)        {            unchecked            {                return ROL(a + ((b & c) | (b & d) | (c & d)) + xk + 0x5a827999, s);            }        }        private static uint Round3Operation(uint a, uint b, uint c, uint d, uint xk, int s)        {            unchecked            {                return ROL(a + (b ^ c ^ d) + xk + 0x6ed9eba1, s);            }        }    }    public static class FileIDUtil    {        public static int Compute(Type t)        {            string toBeHashed = "s\0\0\0" + t.Namespace + t.Name;            using (HashAlgorithm hash = new MD4())            {                byte[] hashed = hash.ComputeHash(System.Text.Encoding.UTF8.GetBytes(toBeHashed));                int result = 0;                for (int i = 3; i >= 0; --i)                {                    result <<= 8;                    result |= hashed[i];                }                return result;            }        }    }}

Unity脚本引用原理,修复Unity脚本引用丢失,源码脚本与dll中的脚本引用互换 .的更多相关文章

  1. [Tomcat 源码分析系列] (二) : Tomcat 启动脚本-catalina.bat

    概述 Tomcat 的三个最重要的启动脚本: startup.bat catalina.bat setclasspath.bat 上一篇咱们分析了 startup.bat 脚本 这一篇咱们来分析 ca ...

  2. Hadoop之HDFS原理及文件上传下载源码分析(下)

    上篇Hadoop之HDFS原理及文件上传下载源码分析(上)楼主主要介绍了hdfs原理及FileSystem的初始化源码解析, Client如何与NameNode建立RPC通信.本篇将继续介绍hdfs文 ...

  3. vue双向绑定的原理及实现双向绑定MVVM源码分析

    vue双向绑定的原理及实现双向绑定MVVM源码分析 双向数据绑定的原理是:可以将对象的属性绑定到UI,具体的说,我们有一个对象,该对象有一个name属性,当我们给这个对象name属性赋新值的时候,新值 ...

  4. [Tomcat 源码分析系列] (附件) : catalina.bat 脚本

    摘自 apache-tomcat-8.0.39-src 源码包中的 catalina.bat 脚本内容 @echo off rem Licensed to the Apache Software Fo ...

  5. 【Spark2.0源码学习】-2.一切从脚本说起

    从脚本说起      在看源码之前,我们一般会看相关脚本了解其初始化信息以及Bootstrap类,Spark也不例外,而Spark我们启动三端使用的脚本如下: %SPARK_HOME%/sbin/st ...

  6. 源码编译安装Apache-附一键部署脚本

    1.进入apache官网https://www.apache.org/,点击Download 2.如图选择 3.选择httpd 4.下载两个包,2.2为CentOS6使用,2.4为CentOS7使用 ...

  7. Hadoop之HDFS原理及文件上传下载源码分析(上)

    HDFS原理 首先说明下,hadoop的各种搭建方式不再介绍,相信各位玩hadoop的同学随便都能搭出来. 楼主的环境: 操作系统:Ubuntu 15.10 hadoop版本:2.7.3 HA:否(随 ...

  8. 聊聊ThreadLocal原理以及使用场景-JAVA 8源码

    相信很多人知道ThreadLocal是针对每个线程的,但是其中的原理相信大家不是很清楚,那咱们就一块看一下源码. 首先,我们先看看它的set方法.非常简单,从当前Thread中获取map.那么这个ge ...

  9. Redis核心原理与实践--事务实践与源码分析

    Redis支持事务机制,但Redis的事务机制与传统关系型数据库的事务机制并不相同. Redis事务的本质是一组命令的集合(命令队列).事务可以一次执行多个命令,并提供以下保证: (1)事务中的所有命 ...

随机推荐

  1. Nodejs 中常见的加密算法:RSA(1)

    Linux用户(以Ubuntu为例) $ openssl 进入OpenSSL程序 OpenSSL> genrsa -out rsa_private_key.pem 1024 生成私钥 OpenS ...

  2. main方法的参数

    敲例子的时候无意中把主方法的参数给落下了,当时没有发现,保存之后就去编译,运行了,通常情况下编译没有错误那胜利就在掌握之中了,没想到这次我竟然在"不一般"的行列中,编译无误,运行出 ...

  3. 一个商品SKU是怎么生成的

    首先说一说什么是SKU.......自己百度去... 类似京东上面,未来人类S5这个台笔记本(没错,我刚入手了) 都是S5这个型号,但是因为CPU,显卡,内存,硬盘等不同,价格也不一样.CPU,显卡, ...

  4. android的GPS代码分析JNI如何HAL之间如何设置回调函数【转】

    本文转载自:http://blog.csdn.net/kmesg/article/details/6531577 本文只关注JNI和HAL的接口部分 在jni的android_location_Gps ...

  5. nginx+keepalived简单双机主从热备

    双机主从热备概述 可以两台机子互为热备,平时各自负责各自的服务.在做上线更新的时候,关闭一台服务器的tomcat后,nginx自动把流量切换到另外一台服务的后备机子上,从而实现无痛更新,保持服务的持续 ...

  6. JAVA- 成员变量与局部变量的区别

    成员变量与局部变量的区别 成员变量是定义在方法之外,类之内的局部变量是定义在方法之内的. 作用上的区别: 1.成员变量的作用是用于描述一类事物的公共属性的. 2.局部变量的作用就是提供一个变量给方法内 ...

  7. phpunit 入门

    话说,使用phpunit还是处于好奇之心,为什么以前一直没使用呢?主要是嫌麻烦,小项目,前后端都是自己写,几个人就完成的工作,没必要弄的那么麻烦.好了,废话不多说,让我们一起来感受下phpunit吧! ...

  8. window/body/img/iframe 的onload事件

    在html页面中,只有body,img,iframe这一类标签具有onload事件. onload事件表示在当前元素载入完成后发生的事件.其中,window也有onload事件,但是跟body的是同一 ...

  9. eclipse如何在不联网的情况下引入dtd约束文件

    1. 获取dtd文件,解压 F:\Java配置文件\Mybatis\mybatis-3.3.0\mybatis-3.3.0.jar\org\apache\ibatis\builder\xml\ 路径下 ...

  10. jsp基本概念

    服务器启动的时候执行初始化init方法,只执行一次 每次请求都会执行一次service方法 服务器停止的时候执行destroy方法,也是只执行一次 <%! //全局变量 int initNum= ...