Tôi đã không tự mình thực hiện điều này, vì vậy câu trả lời này dựa trên một bài báo trước đó ( Tôi chỉ viết "biên dịch thời gian chạy thống nhất" và tìm thấy 4 câu trả lời ngay trên đầu, vì vậy đừng bỏ qua việc thử tự tìm kiếm!)
Những gì bạn mô tả là có thể, nhưng không phải không có cạm bẫy:
Sử dụng mức tương thích .Net 2.0, nếu không, một số không gian tên bạn cần có thể vắng mặt.
Không phải mọi nền tảng sẽ cho phép bạn biên dịch C # mới từ các chuỗi khi chạy - cụ thể là Android và iOS không cho phép điều này.
Ngoài hộp này hoạt động trên Windows, nhưng không phải trên máy Mac / Linux trừ khi người dùng đã cài đặt SDK Mono. Bạn có thể bao gồm mcs-ICodeCompiler của aeroson để xây dựng chức năng này vào trò chơi của bạn dưới dạng plugin.
mã ví dụ của exodrifter (được sửa đổi để thêm một thành phần)
(tham khảo bài viết để biết chi tiết đầy đủ - Tôi chỉ muốn bao gồm đủ ở đây rằng câu trả lời này là một biển chỉ dẫn hữu ích ngay cả khi bài viết được liên kết đi xuống)
using Microsoft.CSharp;
using System;
using System.CodeDom.Compiler;
using System.Reflection;
using System.Text;
using UnityEngine;
public class RuntimeCompileTest : MonoBehaviour
{
void Start()
{
var assembly = Compile(@"
using UnityEngine;
public class RuntimeCompiled : MonoBehaviour
{
public static RuntimeCompiled AddYourselfTo(GameObject host)
{
return host.AddComponent<RuntimeCompiled>();
}
void Start()
{
Debug.Log(""The runtime compiled component was successfully attached to"" + gameObject.name);
}
}");
var runtimeType = assembly.GetType("RuntimeCompiled");
var method = runtimeType.GetMethod("AddYourselfTo");
var del = (Func<GameObject, MonoBehaviour>)
Delegate.CreateDelegate(
typeof(Func<GameObject, MonoBehaviour>),
method
);
// We ask the compiled method to add its component to this.gameObject
var addedComponent = del.Invoke(gameObject);
// The delegate pre-bakes the reflection, so repeated calls don't
// cost us every time, as long as we keep re-using the delegate.
}
public static Assembly Compile(string source)
{
// Replace this Compiler.CSharpCodeProvider wth aeroson's version
// if you're targeting non-Windows platforms:
var provider = new CSharpCodeProvider();
var param = new CompilerParameters();
// Add ALL of the assembly references
foreach (var assembly in AppDomain.CurrentDomain.GetAssemblies())
{
param.ReferencedAssemblies.Add(assembly.Location);
}
// Or, uncomment just the assemblies you need...
// System namespace for common types like collections.
//param.ReferencedAssemblies.Add("System.dll");
// This contains methods from the Unity namespaces:
//param.ReferencedAssemblies.Add("UnityEngines.dll");
// This assembly contains runtime C# code from your Assets folders:
// (If you're using editor scripts, they may be in another assembly)
//param.ReferencedAssemblies.Add("CSharp.dll");
// Generate a dll in memory
param.GenerateExecutable = false;
param.GenerateInMemory = true;
// Compile the source
var result = provider.CompileAssemblyFromSource(param, source);
if (result.Errors.Count > 0) {
var msg = new StringBuilder();
foreach (CompilerError error in result.Errors) {
msg.AppendFormat("Error ({0}): {1}\n",
error.ErrorNumber, error.ErrorText);
}
throw new Exception(msg.ToString());
}
// Return the assembly
return result.CompiledAssembly;
}
}
Cảnh báo
Hãy rất cẩn thận khi để cho trò chơi của bạn biên dịch & chạy các kịch bản bạn đã không viết cho mình. Các tập lệnh này có thể được thực hiện để làm bất cứ điều gì mà tài khoản người dùng của người chơi có đặc quyền phải làm - xóa hoặc làm hỏng tệp người dùng, tải xuống và chạy phần mềm độc hại, rình mò thông tin cá nhân, tất cả điều đó.
Nhiều như chúng tôi có thể cố gắng chặn và phát hiện những trường hợp này, tôi không bao giờ đặc biệt tin rằng chúng tôi đã nắm bắt được tất cả - luôn có khả năng một số tính năng ngôn ngữ C # tôi không biết về việc cho phép chúng truy cập các tính năng mà tôi nghĩ đã bị cắt .
Nếu tập lệnh chỉ được tạo bởi người chơi cục bộ, thì họ có thể phá game theo cách tùy ý, nhưng không thể làm gì với máy tính của họ, dù sao họ cũng không được phép làm. Nhưng nếu người chơi có thể chia sẻ tập lệnh, tải xuống từ tài nguyên mạng hoặc có thể bị lừa sao chép và dán tập lệnh mà họ không viết, thì đây sẽ trở thành một vectơ tấn công có thể bị khai thác.
Nếu bạn muốn cho phép người chơi lập trình mà không cần mở loại lỗ hổng này, hãy xem xét việc viết máy ảo của riêng bạn để phân tích / giải thích / thực thi các tập lệnh của trình phát trong một môi trường hạn chế hơn mà bạn kiểm soát hoàn toàn.